From 73ba1aca1e127447cf42c9e70752921078dbb2e7 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Tue, 9 Jan 2024 21:00:16 +0000 Subject: [PATCH] Update unreleased documentation (#552) * Update versions.json * Deployed 5b660186a to unreleased in versions with MkDocs 1.5.3 and mike 2.0.0 * Sort docs versions --------- Co-authored-by: GitHub Actions Bot --- versions/unreleased/404.html | 28 + versions/unreleased/api-ref/index.html | 28 + .../api-ref/prefect/agent/index.html | 28 + .../api-ref/prefect/artifacts/index.html | 28 + .../api-ref/prefect/blocks/core/index.html | 28 + .../api-ref/prefect/blocks/fields/index.html | 28 + .../prefect/blocks/kubernetes/index.html | 28 + .../prefect/blocks/notifications/index.html | 28 + .../api-ref/prefect/blocks/system/index.html | 28 + .../api-ref/prefect/blocks/webhook/index.html | 28 + .../api-ref/prefect/cli/agent/index.html | 28 + .../api-ref/prefect/cli/artifact/index.html | 28 + .../api-ref/prefect/cli/block/index.html | 28 + .../prefect/cli/cloud-webhook/index.html | 28 + .../api-ref/prefect/cli/cloud/index.html | 28 + .../prefect/cli/concurrency_limit/index.html | 28 + .../api-ref/prefect/cli/config/index.html | 28 + .../api-ref/prefect/cli/deploy/index.html | 28 + .../api-ref/prefect/cli/deployment/index.html | 28 + .../api-ref/prefect/cli/dev/index.html | 28 + .../api-ref/prefect/cli/flow/index.html | 28 + .../api-ref/prefect/cli/flow_run/index.html | 28 + .../api-ref/prefect/cli/kubernetes/index.html | 28 + .../api-ref/prefect/cli/profile/index.html | 28 + .../api-ref/prefect/cli/project/index.html | 28 + .../api-ref/prefect/cli/root/index.html | 28 + .../api-ref/prefect/cli/server/index.html | 28 + .../api-ref/prefect/cli/variable/index.html | 28 + .../api-ref/prefect/cli/work_pool/index.html | 28 + .../api-ref/prefect/cli/work_queue/index.html | 28 + .../api-ref/prefect/cli/worker/index.html | 28 + .../api-ref/prefect/client/base/index.html | 28 + .../api-ref/prefect/client/cloud/index.html | 28 + .../prefect/client/orchestration/index.html | 872 +- .../api-ref/prefect/client/schemas/index.html | 309 + .../prefect/client/utilities/index.html | 28 + .../prefect/concurrency/asyncio/index.html | 28 + .../prefect/concurrency/common/index.html | 28 + .../prefect/concurrency/events/index.html | 28 + .../prefect/concurrency/services/index.html | 28 + .../prefect/concurrency/sync/index.html | 28 + .../api-ref/prefect/context/index.html | 28 + .../prefect/deployments/base/index.html | 28 + .../deployments/deployments/index.html | 28 + .../prefect/deployments/runner/index.html | 28 + .../prefect/deployments/steps/core/index.html | 28 + .../prefect/deployments/steps/pull/index.html | 28 + .../deployments/steps/utility/index.html | 28 + .../api-ref/prefect/engine/index.html | 28 + .../api-ref/prefect/events/index.html | 28 + .../api-ref/prefect/exceptions/index.html | 28 + .../api-ref/prefect/filesystems/index.html | 28 + .../api-ref/prefect/flows/index.html | 28 + .../api-ref/prefect/futures/index.html | 28 + .../api-ref/prefect/infrastructure/index.html | 28 + .../api-ref/prefect/input/actions/index.html | 28 + .../prefect/input/run_input/index.html | 388 +- .../prefect/logging/configuration/index.html | 28 + .../prefect/logging/formatters/index.html | 28 + .../prefect/logging/handlers/index.html | 28 + .../prefect/logging/highlighters/index.html | 28 + .../api-ref/prefect/logging/index.html | 28 + .../prefect/logging/loggers/index.html | 28 + .../api-ref/prefect/manifests/index.html | 28 + .../api-ref/prefect/packaging/base/index.html | 28 + .../prefect/packaging/docker/index.html | 28 + .../api-ref/prefect/packaging/file/index.html | 28 + .../prefect/packaging/serializers/index.html | 28 + .../api-ref/prefect/runner/runner/index.html | 28 + .../api-ref/prefect/runner/server/index.html | 28 + .../api-ref/prefect/runner/storage/index.html | 28 + .../api-ref/prefect/runner/utils/index.html | 28 + .../prefect/runtime/deployment/index.html | 28 + .../prefect/runtime/flow_run/index.html | 28 + .../prefect/runtime/task_run/index.html | 28 + .../api-ref/prefect/serializers/index.html | 28 + .../api-ref/prefect/settings/index.html | 28 + .../api-ref/prefect/software/index.html | 28 + .../api-ref/prefect/states/index.html | 28 + .../api-ref/prefect/task-runners/index.html | 28 + .../api-ref/prefect/tasks/index.html | 28 + .../api-ref/prefect/testing/index.html | 28 + .../prefect/utilities/annotations/index.html | 28 + .../prefect/utilities/asyncutils/index.html | 28 + .../prefect/utilities/callables/index.html | 28 + .../prefect/utilities/collections/index.html | 28 + .../prefect/utilities/compat/index.html | 28 + .../prefect/utilities/context/index.html | 28 + .../prefect/utilities/dispatch/index.html | 28 + .../prefect/utilities/dockerutils/index.html | 28 + .../prefect/utilities/filesystem/index.html | 28 + .../prefect/utilities/hashing/index.html | 28 + .../prefect/utilities/importtools/index.html | 28 + .../api-ref/prefect/utilities/math/index.html | 28 + .../prefect/utilities/names/index.html | 28 + .../prefect/utilities/processutils/index.html | 28 + .../prefect/utilities/pydantic/index.html | 28 + .../utilities/render_swagger/index.html | 28 + .../prefect/utilities/services/index.html | 28 + .../prefect/utilities/slugify/index.html | 28 + .../prefect/utilities/templating/index.html | 28 + .../api-ref/prefect/utilities/text/index.html | 28 + .../prefect/utilities/validation/index.html | 28 + .../utilities/visualization/index.html | 28 + .../api-ref/prefect/variables/index.html | 28 + .../api-ref/prefect/workers/base/index.html | 28 + .../api-ref/prefect/workers/block/index.html | 28 + .../prefect/workers/process/index.html | 28 + .../api-ref/prefect/workers/server/index.html | 28 + .../prefect/workers/utilities/index.html | 28 + versions/unreleased/api-ref/python/index.html | 28 + .../api-ref/rest-api-reference/index.html | 28 + .../unreleased/api-ref/rest-api/index.html | 28 + .../api-ref/server/api/admin/index.html | 28 + .../server/api/dependencies/index.html | 28 + .../api-ref/server/api/deployments/index.html | 28 + .../server/api/flow_run_states/index.html | 28 + .../api-ref/server/api/flow_runs/index.html | 28 + .../api-ref/server/api/flows/index.html | 28 + .../api-ref/server/api/run_history/index.html | 28 + .../server/api/saved_searches/index.html | 28 + .../api-ref/server/api/server/index.html | 28 + .../server/api/task_run_states/index.html | 28 + .../api-ref/server/api/task_runs/index.html | 28 + versions/unreleased/api-ref/server/index.html | 28 + .../server/models/deployments/index.html | 28 + .../server/models/flow_run_states/index.html | 28 + .../server/models/flow_runs/index.html | 28 + .../api-ref/server/models/flows/index.html | 28 + .../server/models/saved_searches/index.html | 28 + .../server/models/task_run_states/index.html | 28 + .../server/models/task_runs/index.html | 28 + .../orchestration/core_policy/index.html | 28 + .../orchestration/global_policy/index.html | 28 + .../server/orchestration/policies/index.html | 28 + .../server/orchestration/rules/index.html | 28 + .../api-ref/server/schemas/actions/index.html | 28 + .../api-ref/server/schemas/core/index.html | 28 + .../api-ref/server/schemas/filters/index.html | 28 + .../server/schemas/responses/index.html | 28 + .../server/schemas/schedules/index.html | 28 + .../api-ref/server/schemas/sorting/index.html | 28 + .../api-ref/server/schemas/states/index.html | 28 + .../server/services/late_runs/index.html | 28 + .../server/services/loop_service/index.html | 28 + .../server/services/scheduler/index.html | 28 + .../server/utilities/database/index.html | 28 + .../server/utilities/schemas/index.html | 28 + .../server/utilities/server/index.html | 28 + .../images/social/guides/automations.png | Bin 0 -> 50061 bytes .../cloud/cloud-quickstart/index.html | 28 + .../unreleased/cloud/connecting/index.html | 28 + .../unreleased/cloud/incidents/index.html | 28 + versions/unreleased/cloud/index.html | 28 + .../unreleased/cloud/rate-limits/index.html | 28 + .../cloud/users/api-keys/index.html | 28 + .../cloud/users/audit-log/index.html | 28 + versions/unreleased/cloud/users/index.html | 28 + .../object-access-control-lists/index.html | 28 + .../unreleased/cloud/users/roles/index.html | 28 + .../cloud/users/service-accounts/index.html | 28 + .../unreleased/cloud/users/sso/index.html | 28 + .../unreleased/cloud/users/teams/index.html | 28 + .../unreleased/cloud/workspaces/index.html | 28 + versions/unreleased/community/index.html | 28 + .../unreleased/concepts/agents/index.html | 28 + .../unreleased/concepts/artifacts/index.html | 28 + .../concepts/automations/index.html | 28 + .../unreleased/concepts/blocks/index.html | 28 + .../deployments-block-based/index.html | 28 + .../concepts/deployments/index.html | 28 + .../unreleased/concepts/events/index.html | 28 + .../concepts/filesystems/index.html | 28 + versions/unreleased/concepts/flows/index.html | 28 + versions/unreleased/concepts/index.html | 28 + .../concepts/infrastructure/index.html | 28 + .../unreleased/concepts/results/index.html | 28 + .../unreleased/concepts/schedules/index.html | 28 + .../unreleased/concepts/states/index.html | 28 + .../unreleased/concepts/storage/index.html | 28 + .../concepts/task-runners/index.html | 28 + versions/unreleased/concepts/tasks/index.html | 28 + .../unreleased/concepts/work-pools/index.html | 28 + .../contributing/overview/index.html | 28 + .../unreleased/contributing/style/index.html | 28 + .../contributing/versioning/index.html | 28 + versions/unreleased/faq/index.html | 28 + .../getting-started/installation/index.html | 28 + .../getting-started/quickstart/index.html | 28 + .../unreleased/guides/automations/index.html | 9294 +++++++++++++++++ .../unreleased/guides/big-data/index.html | 28 + versions/unreleased/guides/ci-cd/index.html | 28 + .../index.html | 30 +- .../guides/dask-ray-task-runners/index.html | 28 + .../guides/deployment/aci/index.html | 28 + .../guides/deployment/daemonize/index.html | 28 + .../developing-a-new-worker-type/index.html | 28 + .../guides/deployment/kubernetes/index.html | 28 + .../deployment/push-work-pools/index.html | 28 + .../deployment/serverless-workers/index.html | 28 + .../deployment/storage-guide/index.html | 28 + versions/unreleased/guides/docker/index.html | 28 + .../global-concurrency-limits/index.html | 28 + versions/unreleased/guides/host/index.html | 28 + versions/unreleased/guides/index.html | 32 + versions/unreleased/guides/logs/index.html | 28 + .../guides/managed-execution/index.html | 28 + .../guides/migration-guide/index.html | 28 + .../unreleased/guides/moving-data/index.html | 28 + .../guides/prefect-deploy/index.html | 28 + .../guides/runtime-context/index.html | 28 + .../unreleased/guides/settings/index.html | 28 + .../guides/state-change-hooks/index.html | 28 + versions/unreleased/guides/testing/index.html | 28 + .../guides/troubleshooting/index.html | 28 + .../index.html | 28 + .../guides/using-the-client/index.html | 28 + .../unreleased/guides/variables/index.html | 28 + .../unreleased/guides/webhooks/index.html | 30 +- .../img/guides/automation-custom.png | Bin 0 -> 169674 bytes .../unreleased/img/guides/automation-list.png | Bin 0 -> 84671 bytes .../img/guides/automation-triggers.png | Bin 0 -> 46229 bytes versions/unreleased/img/guides/block-list.png | Bin 0 -> 483683 bytes .../img/guides/final-automation.png | Bin 0 -> 36884 bytes .../img/guides/notification-block.png | Bin 0 -> 114616 bytes .../img/guides/notify-auto-block.png | Bin 0 -> 191122 bytes .../img/guides/webhook-automate.png | Bin 0 -> 50890 bytes .../unreleased/img/guides/webhook-created.png | Bin 0 -> 50277 bytes .../unreleased/img/guides/webhook-simple.png | Bin 0 -> 51651 bytes versions/unreleased/index.html | 28 + .../integrations/contribute/index.html | 28 + versions/unreleased/integrations/index.html | 28 + .../unreleased/integrations/usage/index.html | 28 + versions/unreleased/objects.inv | Bin 22714 -> 22745 bytes .../unreleased/recipes/recipes/index.html | 28 + versions/unreleased/search/search_index.json | 2 +- versions/unreleased/sitemap.xml | 459 +- versions/unreleased/sitemap.xml.gz | Bin 1759 -> 1764 bytes .../tutorial/deployments/index.html | 28 + versions/unreleased/tutorial/flows/index.html | 28 + versions/unreleased/tutorial/index.html | 28 + versions/unreleased/tutorial/tasks/index.html | 28 + .../unreleased/tutorial/work-pools/index.html | 28 + .../unreleased/tutorial/workers/index.html | 28 + 244 files changed, 16886 insertions(+), 746 deletions(-) create mode 100644 versions/unreleased/assets/images/social/guides/automations.png create mode 100644 versions/unreleased/guides/automations/index.html create mode 100644 versions/unreleased/img/guides/automation-custom.png create mode 100644 versions/unreleased/img/guides/automation-list.png create mode 100644 versions/unreleased/img/guides/automation-triggers.png create mode 100644 versions/unreleased/img/guides/block-list.png create mode 100644 versions/unreleased/img/guides/final-automation.png create mode 100644 versions/unreleased/img/guides/notification-block.png create mode 100644 versions/unreleased/img/guides/notify-auto-block.png create mode 100644 versions/unreleased/img/guides/webhook-automate.png create mode 100644 versions/unreleased/img/guides/webhook-created.png create mode 100644 versions/unreleased/img/guides/webhook-simple.png diff --git a/versions/unreleased/404.html b/versions/unreleased/404.html index 0bc1bb0ce9..c1f616b3eb 100644 --- a/versions/unreleased/404.html +++ b/versions/unreleased/404.html @@ -822,6 +822,8 @@ + + @@ -1078,6 +1080,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/index.html b/versions/unreleased/api-ref/index.html index 2ccfbe0a22..e9571d9df2 100644 --- a/versions/unreleased/api-ref/index.html +++ b/versions/unreleased/api-ref/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/agent/index.html b/versions/unreleased/api-ref/prefect/agent/index.html index 9a2aada450..0a51412a9b 100644 --- a/versions/unreleased/api-ref/prefect/agent/index.html +++ b/versions/unreleased/api-ref/prefect/agent/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/artifacts/index.html b/versions/unreleased/api-ref/prefect/artifacts/index.html index bb4d59d164..03bf62e31d 100644 --- a/versions/unreleased/api-ref/prefect/artifacts/index.html +++ b/versions/unreleased/api-ref/prefect/artifacts/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/blocks/core/index.html b/versions/unreleased/api-ref/prefect/blocks/core/index.html index 85a66a8a9d..0c840d7f34 100644 --- a/versions/unreleased/api-ref/prefect/blocks/core/index.html +++ b/versions/unreleased/api-ref/prefect/blocks/core/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/blocks/fields/index.html b/versions/unreleased/api-ref/prefect/blocks/fields/index.html index bf9964f8c7..8171c7d924 100644 --- a/versions/unreleased/api-ref/prefect/blocks/fields/index.html +++ b/versions/unreleased/api-ref/prefect/blocks/fields/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/blocks/kubernetes/index.html b/versions/unreleased/api-ref/prefect/blocks/kubernetes/index.html index d6036e2beb..651a6f3b5a 100644 --- a/versions/unreleased/api-ref/prefect/blocks/kubernetes/index.html +++ b/versions/unreleased/api-ref/prefect/blocks/kubernetes/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/blocks/notifications/index.html b/versions/unreleased/api-ref/prefect/blocks/notifications/index.html index 798ae0a617..862acf11d2 100644 --- a/versions/unreleased/api-ref/prefect/blocks/notifications/index.html +++ b/versions/unreleased/api-ref/prefect/blocks/notifications/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/blocks/system/index.html b/versions/unreleased/api-ref/prefect/blocks/system/index.html index 11157eafb6..6fe429129e 100644 --- a/versions/unreleased/api-ref/prefect/blocks/system/index.html +++ b/versions/unreleased/api-ref/prefect/blocks/system/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/blocks/webhook/index.html b/versions/unreleased/api-ref/prefect/blocks/webhook/index.html index fbc7ae3a55..7fd27c5055 100644 --- a/versions/unreleased/api-ref/prefect/blocks/webhook/index.html +++ b/versions/unreleased/api-ref/prefect/blocks/webhook/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/agent/index.html b/versions/unreleased/api-ref/prefect/cli/agent/index.html index 5948cc48b3..bf1a36af72 100644 --- a/versions/unreleased/api-ref/prefect/cli/agent/index.html +++ b/versions/unreleased/api-ref/prefect/cli/agent/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/artifact/index.html b/versions/unreleased/api-ref/prefect/cli/artifact/index.html index 0117789dcb..115c9fc1a6 100644 --- a/versions/unreleased/api-ref/prefect/cli/artifact/index.html +++ b/versions/unreleased/api-ref/prefect/cli/artifact/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/block/index.html b/versions/unreleased/api-ref/prefect/cli/block/index.html index ce743cb76a..74a30ae58a 100644 --- a/versions/unreleased/api-ref/prefect/cli/block/index.html +++ b/versions/unreleased/api-ref/prefect/cli/block/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/cloud-webhook/index.html b/versions/unreleased/api-ref/prefect/cli/cloud-webhook/index.html index 9beb376dfb..2935d0e4bb 100644 --- a/versions/unreleased/api-ref/prefect/cli/cloud-webhook/index.html +++ b/versions/unreleased/api-ref/prefect/cli/cloud-webhook/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/cloud/index.html b/versions/unreleased/api-ref/prefect/cli/cloud/index.html index bbe4f98c5e..c3ce8513ba 100644 --- a/versions/unreleased/api-ref/prefect/cli/cloud/index.html +++ b/versions/unreleased/api-ref/prefect/cli/cloud/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/concurrency_limit/index.html b/versions/unreleased/api-ref/prefect/cli/concurrency_limit/index.html index 90b0c693d2..411b3ff40d 100644 --- a/versions/unreleased/api-ref/prefect/cli/concurrency_limit/index.html +++ b/versions/unreleased/api-ref/prefect/cli/concurrency_limit/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/config/index.html b/versions/unreleased/api-ref/prefect/cli/config/index.html index b4ce2c70de..963150a070 100644 --- a/versions/unreleased/api-ref/prefect/cli/config/index.html +++ b/versions/unreleased/api-ref/prefect/cli/config/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/deploy/index.html b/versions/unreleased/api-ref/prefect/cli/deploy/index.html index e89b0e7f41..61c9b2fa11 100644 --- a/versions/unreleased/api-ref/prefect/cli/deploy/index.html +++ b/versions/unreleased/api-ref/prefect/cli/deploy/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/deployment/index.html b/versions/unreleased/api-ref/prefect/cli/deployment/index.html index 0aa26753e0..1edc8fb2c0 100644 --- a/versions/unreleased/api-ref/prefect/cli/deployment/index.html +++ b/versions/unreleased/api-ref/prefect/cli/deployment/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/dev/index.html b/versions/unreleased/api-ref/prefect/cli/dev/index.html index 76e9f89f7c..04e88af7b8 100644 --- a/versions/unreleased/api-ref/prefect/cli/dev/index.html +++ b/versions/unreleased/api-ref/prefect/cli/dev/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/flow/index.html b/versions/unreleased/api-ref/prefect/cli/flow/index.html index 93e8d0b109..f7c6658877 100644 --- a/versions/unreleased/api-ref/prefect/cli/flow/index.html +++ b/versions/unreleased/api-ref/prefect/cli/flow/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/flow_run/index.html b/versions/unreleased/api-ref/prefect/cli/flow_run/index.html index 302e38e153..9c573742a7 100644 --- a/versions/unreleased/api-ref/prefect/cli/flow_run/index.html +++ b/versions/unreleased/api-ref/prefect/cli/flow_run/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/kubernetes/index.html b/versions/unreleased/api-ref/prefect/cli/kubernetes/index.html index 05ad41f022..0cf52a6d73 100644 --- a/versions/unreleased/api-ref/prefect/cli/kubernetes/index.html +++ b/versions/unreleased/api-ref/prefect/cli/kubernetes/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/profile/index.html b/versions/unreleased/api-ref/prefect/cli/profile/index.html index 00e58ab0bb..74fb5d947f 100644 --- a/versions/unreleased/api-ref/prefect/cli/profile/index.html +++ b/versions/unreleased/api-ref/prefect/cli/profile/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/project/index.html b/versions/unreleased/api-ref/prefect/cli/project/index.html index 0136da75dd..91f97a91de 100644 --- a/versions/unreleased/api-ref/prefect/cli/project/index.html +++ b/versions/unreleased/api-ref/prefect/cli/project/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/root/index.html b/versions/unreleased/api-ref/prefect/cli/root/index.html index ab21ac43cc..8b729cff69 100644 --- a/versions/unreleased/api-ref/prefect/cli/root/index.html +++ b/versions/unreleased/api-ref/prefect/cli/root/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/server/index.html b/versions/unreleased/api-ref/prefect/cli/server/index.html index bc9cc0b86c..d002e96320 100644 --- a/versions/unreleased/api-ref/prefect/cli/server/index.html +++ b/versions/unreleased/api-ref/prefect/cli/server/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/variable/index.html b/versions/unreleased/api-ref/prefect/cli/variable/index.html index 68e9f1e78b..b0e22ab0f2 100644 --- a/versions/unreleased/api-ref/prefect/cli/variable/index.html +++ b/versions/unreleased/api-ref/prefect/cli/variable/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/work_pool/index.html b/versions/unreleased/api-ref/prefect/cli/work_pool/index.html index b018f5a6a0..6eae766004 100644 --- a/versions/unreleased/api-ref/prefect/cli/work_pool/index.html +++ b/versions/unreleased/api-ref/prefect/cli/work_pool/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/work_queue/index.html b/versions/unreleased/api-ref/prefect/cli/work_queue/index.html index 51054923c0..595e0ae11d 100644 --- a/versions/unreleased/api-ref/prefect/cli/work_queue/index.html +++ b/versions/unreleased/api-ref/prefect/cli/work_queue/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/cli/worker/index.html b/versions/unreleased/api-ref/prefect/cli/worker/index.html index e5f5a7916d..f02fc53c5a 100644 --- a/versions/unreleased/api-ref/prefect/cli/worker/index.html +++ b/versions/unreleased/api-ref/prefect/cli/worker/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/client/base/index.html b/versions/unreleased/api-ref/prefect/client/base/index.html index 0872fab79c..c726322b34 100644 --- a/versions/unreleased/api-ref/prefect/client/base/index.html +++ b/versions/unreleased/api-ref/prefect/client/base/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/client/cloud/index.html b/versions/unreleased/api-ref/prefect/client/cloud/index.html index d36966c2a1..0c5c52bce1 100644 --- a/versions/unreleased/api-ref/prefect/client/cloud/index.html +++ b/versions/unreleased/api-ref/prefect/client/cloud/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/client/orchestration/index.html b/versions/unreleased/api-ref/prefect/client/orchestration/index.html index 5db97f894b..c1fbe1263e 100644 --- a/versions/unreleased/api-ref/prefect/client/orchestration/index.html +++ b/versions/unreleased/api-ref/prefect/client/orchestration/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • @@ -10608,9 +10636,7 @@

    Source code in prefect/client/orchestration.py -
     168
    - 169
    - 170
    +              
     170
      171
      172
      173
    @@ -13283,7 +13309,69 @@ 

    2840 2841 2842 -2843

    class PrefectClient:
    +2843
    +2844
    +2845
    +2846
    +2847
    +2848
    +2849
    +2850
    +2851
    +2852
    +2853
    +2854
    +2855
    +2856
    +2857
    +2858
    +2859
    +2860
    +2861
    +2862
    +2863
    +2864
    +2865
    +2866
    +2867
    +2868
    +2869
    +2870
    +2871
    +2872
    +2873
    +2874
    +2875
    +2876
    +2877
    +2878
    +2879
    +2880
    +2881
    +2882
    +2883
    +2884
    +2885
    +2886
    +2887
    +2888
    +2889
    +2890
    +2891
    +2892
    +2893
    +2894
    +2895
    +2896
    +2897
    +2898
    +2899
    +2900
    +2901
    +2902
    +2903
    +2904
    +2905
    class PrefectClient:
         """
         An asynchronous client for interacting with the [Prefect REST API](/api-ref/rest-api/).
     
    @@ -15840,6 +15928,66 @@ 

    }, ) + async def create_global_concurrency_limit( + self, concurrency_limit: GlobalConcurrencyLimitCreate + ) -> UUID: + response = await self._client.post( + "/v2/concurrency_limits/", + json=concurrency_limit.dict(json_compatible=True, exclude_unset=True), + ) + return UUID(response.json()["id"]) + + async def update_global_concurrency_limit( + self, name: str, concurrency_limit: GlobalConcurrencyLimitUpdate + ) -> httpx.Response: + try: + response = await self._client.patch( + f"/v2/concurrency_limits/{name}", + json=concurrency_limit.dict(json_compatible=True, exclude_unset=True), + ) + return response + except httpx.HTTPStatusError as e: + if e.response.status_code == status.HTTP_404_NOT_FOUND: + raise prefect.exceptions.ObjectNotFound(http_exc=e) from e + else: + raise + + async def delete_global_concurrency_limit_by_name( + self, name: str + ) -> httpx.Response: + try: + response = await self._client.delete(f"/v2/concurrency_limits/{name}") + return response + except httpx.HTTPStatusError as e: + if e.response.status_code == status.HTTP_404_NOT_FOUND: + raise prefect.exceptions.ObjectNotFound(http_exc=e) from e + else: + raise + + async def read_global_concurrency_limit_by_name( + self, name: str + ) -> Dict[str, object]: + try: + response = await self._client.get(f"/v2/concurrency_limits/{name}") + return response.json() + except httpx.HTTPStatusError as e: + if e.response.status_code == status.HTTP_404_NOT_FOUND: + raise prefect.exceptions.ObjectNotFound(http_exc=e) from e + else: + raise + + async def read_global_concurrency_limits( + self, limit: int = 10, offset: int = 0 + ) -> List[Dict[str, object]]: + response = await self._client.post( + "/v2/concurrency_limits/filter", + json={ + "limit": limit, + "offset": offset, + }, + ) + return response.json() + async def create_flow_run_input( self, flow_run_id: UUID, key: str, value: str, sender: Optional[str] = None ): @@ -16018,9 +16166,7 @@

    Source code in prefect/client/orchestration.py -
    338
    -339
    -340
    +            
    340
     341
     342
     343
    @@ -16029,7 +16175,9 @@ 

    346 347 348 -349

    async def api_healthcheck(self) -> Optional[Exception]:
    +349
    +350
    +351
    async def api_healthcheck(self) -> Optional[Exception]:
         """
         Attempts to connect to the API and returns the encountered exception if not
         successful.
    @@ -16068,11 +16216,11 @@ 

    Source code in prefect/client/orchestration.py -
    351
    -352
    -353
    +            
    353
     354
    -355
    async def hello(self) -> httpx.Response:
    +355
    +356
    +357
    async def hello(self) -> httpx.Response:
         """
         Send a GET request to /hello for testing purposes.
         """
    @@ -16182,9 +16330,7 @@ 

    Source code in prefect/client/orchestration.py -
    357
    -358
    -359
    +            
    359
     360
     361
     362
    @@ -16195,7 +16341,9 @@ 

    367 368 369 -370

    async def create_flow(self, flow: "FlowObject") -> UUID:
    +370
    +371
    +372
    async def create_flow(self, flow: "FlowObject") -> UUID:
         """
         Create a flow in the Prefect API.
     
    @@ -16314,9 +16462,7 @@ 

    Source code in prefect/client/orchestration.py -
    372
    -373
    -374
    +            
    374
     375
     376
     377
    @@ -16337,7 +16483,9 @@ 

    392 393 394 -395

    async def create_flow_from_name(self, flow_name: str) -> UUID:
    +395
    +396
    +397
    async def create_flow_from_name(self, flow_name: str) -> UUID:
         """
         Create a flow in the Prefect API.
     
    @@ -16442,9 +16590,7 @@ 

    Source code in prefect/client/orchestration.py -
    397
    -398
    -399
    +            
    399
     400
     401
     402
    @@ -16453,7 +16599,9 @@ 

    405 406 407 -408

    async def read_flow(self, flow_id: UUID) -> Flow:
    +408
    +409
    +410
    async def read_flow(self, flow_id: UUID) -> Flow:
         """
         Query the Prefect API for a flow by id.
     
    @@ -16659,9 +16807,7 @@ 

    Source code in prefect/client/orchestration.py -
    410
    -411
    -412
    +            
    412
     413
     414
     415
    @@ -16721,7 +16867,9 @@ 

    469 470 471 -472

    async def read_flows(
    +472
    +473
    +474
    async def read_flows(
         self,
         *,
         flow_filter: FlowFilter = None,
    @@ -16865,9 +17013,7 @@ 

    Source code in prefect/client/orchestration.py -
    474
    -475
    -476
    +            
    476
     477
     478
     479
    @@ -16879,7 +17025,9 @@ 

    485 486 487 -488

    async def read_flow_by_name(
    +488
    +489
    +490
    async def read_flow_by_name(
         self,
         flow_name: str,
     ) -> Flow:
    @@ -17120,9 +17268,7 @@ 

    Source code in prefect/client/orchestration.py -
    490
    -491
    -492
    +            
    492
     493
     494
     495
    @@ -17185,7 +17331,9 @@ 

    552 553 554 -555

    async def create_flow_run_from_deployment(
    +555
    +556
    +557
    async def create_flow_run_from_deployment(
         self,
         deployment_id: UUID,
         *,
    @@ -17442,9 +17590,7 @@ 

    Source code in prefect/client/orchestration.py -
    557
    -558
    -559
    +            
    559
     560
     561
     562
    @@ -17504,7 +17650,9 @@ 

    616 617 618 -619

    async def create_flow_run(
    +619
    +620
    +621
    async def create_flow_run(
         self,
         flow: "FlowObject",
         name: str = None,
    @@ -17736,9 +17884,7 @@ 

    Source code in prefect/client/orchestration.py -
    621
    -622
    -623
    +            
    623
     624
     625
     626
    @@ -17784,7 +17930,9 @@ 

    666 667 668 -669

    async def update_flow_run(
    +669
    +670
    +671
    async def update_flow_run(
         self,
         flow_run_id: UUID,
         flow_version: Optional[str] = None,
    @@ -17891,9 +18039,7 @@ 

    Source code in prefect/client/orchestration.py -
    671
    -672
    -673
    +            
    673
     674
     675
     676
    @@ -17910,7 +18056,9 @@ 

    687 688 689 -690

    async def delete_flow_run(
    +690
    +691
    +692
    async def delete_flow_run(
         self,
         flow_run_id: UUID,
     ) -> None:
    @@ -18050,9 +18198,7 @@ 

    Source code in prefect/client/orchestration.py -
    692
    -693
    -694
    +            
    694
     695
     696
     697
    @@ -18084,7 +18230,9 @@ 

    723 724 725 -726

    async def create_concurrency_limit(
    +726
    +727
    +728
    async def create_concurrency_limit(
         self,
         tag: str,
         concurrency_limit: int,
    @@ -18233,9 +18381,7 @@ 

    Source code in prefect/client/orchestration.py -
    728
    -729
    -730
    +            
    730
     731
     732
     733
    @@ -18266,7 +18412,9 @@ 

    758 759 760 -761

    async def read_concurrency_limit_by_tag(
    +761
    +762
    +763
    async def read_concurrency_limit_by_tag(
         self,
         tag: str,
     ):
    @@ -18394,9 +18542,7 @@ 

    Source code in prefect/client/orchestration.py -
    763
    -764
    -765
    +            
    765
     766
     767
     768
    @@ -18416,7 +18562,9 @@ 

    782 783 784 -785

    async def read_concurrency_limits(
    +785
    +786
    +787
    async def read_concurrency_limits(
         self,
         limit: int,
         offset: int,
    @@ -18547,9 +18695,7 @@ 

    Source code in prefect/client/orchestration.py -
    787
    -788
    -789
    +            
    789
     790
     791
     792
    @@ -18579,7 +18725,9 @@ 

    816 817 818 -819

    async def reset_concurrency_limit_by_tag(
    +819
    +820
    +821
    async def reset_concurrency_limit_by_tag(
         self,
         tag: str,
         slot_override: Optional[List[Union[UUID, str]]] = None,
    @@ -18703,9 +18851,7 @@ 

    Source code in prefect/client/orchestration.py -
    821
    -822
    -823
    +            
    823
     824
     825
     826
    @@ -18726,7 +18872,9 @@ 

    841 842 843 -844

    async def delete_concurrency_limit_by_tag(
    +844
    +845
    +846
    async def delete_concurrency_limit_by_tag(
         self,
         tag: str,
     ):
    @@ -18949,9 +19097,7 @@ 

    Source code in prefect/client/orchestration.py -
    846
    -847
    -848
    +            
    848
     849
     850
     851
    @@ -19015,7 +19161,9 @@ 

    909 910 911 -912

    async def create_work_queue(
    +912
    +913
    +914
    async def create_work_queue(
         self,
         name: str,
         tags: Optional[List[str]] = None,
    @@ -19212,9 +19360,7 @@ 

    Source code in prefect/client/orchestration.py -
    914
    -915
    -916
    +            
    916
     917
     918
     919
    @@ -19245,7 +19391,9 @@ 

    944 945 946 -947

    async def read_work_queue_by_name(
    +947
    +948
    +949
    async def read_work_queue_by_name(
         self,
         name: str,
         work_pool_name: Optional[str] = None,
    @@ -19393,9 +19541,7 @@ 

    Source code in prefect/client/orchestration.py -
    949
    -950
    -951
    +            
    951
     952
     953
     954
    @@ -19417,7 +19563,9 @@ 

    970 971 972 -973

    async def update_work_queue(self, id: UUID, **kwargs):
    +973
    +974
    +975
    async def update_work_queue(self, id: UUID, **kwargs):
         """
         Update properties of a work queue.
     
    @@ -19586,9 +19734,7 @@ 

    Source code in prefect/client/orchestration.py -
     975
    - 976
    - 977
    +            
     977
      978
      979
      980
    @@ -19624,7 +19770,9 @@ 

    1010 1011 1012 -1013

    async def get_runs_in_work_queue(
    +1013
    +1014
    +1015
    async def get_runs_in_work_queue(
         self,
         id: UUID,
         limit: int = 10,
    @@ -19778,9 +19926,7 @@ 

    Source code in prefect/client/orchestration.py -
    1015
    -1016
    -1017
    +            
    1017
     1018
     1019
     1020
    @@ -19802,7 +19948,9 @@ 

    1036 1037 1038 -1039

    async def read_work_queue(
    +1039
    +1040
    +1041
    async def read_work_queue(
         self,
         id: UUID,
     ) -> WorkQueue:
    @@ -19942,9 +20090,7 @@ 

    Source code in prefect/client/orchestration.py -
    1041
    -1042
    -1043
    +            
    1043
     1044
     1045
     1046
    @@ -19966,7 +20112,9 @@ 

    1062 1063 1064 -1065

    async def read_work_queue_status(
    +1065
    +1066
    +1067
    async def read_work_queue_status(
         self,
         id: UUID,
     ) -> WorkQueueStatusDetail:
    @@ -20087,9 +20235,7 @@ 

    Source code in prefect/client/orchestration.py -
    1067
    -1068
    -1069
    +            
    1069
     1070
     1071
     1072
    @@ -20121,7 +20267,9 @@ 

    1098 1099 1100 -1101

    async def match_work_queues(
    +1101
    +1102
    +1103
    async def match_work_queues(
         self,
         prefixes: List[str],
         work_pool_name: Optional[str] = None,
    @@ -20247,9 +20395,7 @@ 

    Source code in prefect/client/orchestration.py -
    1103
    -1104
    -1105
    +            
    1105
     1106
     1107
     1108
    @@ -20269,7 +20415,9 @@ 

    1122 1123 1124 -1125

    async def delete_work_queue_by_id(
    +1125
    +1126
    +1127
    async def delete_work_queue_by_id(
         self,
         id: UUID,
     ):
    @@ -20319,9 +20467,7 @@ 

    Source code in prefect/client/orchestration.py -
    1127
    -1128
    -1129
    +            
    1129
     1130
     1131
     1132
    @@ -20335,7 +20481,9 @@ 

    1140 1141 1142 -1143

    async def create_block_type(self, block_type: BlockTypeCreate) -> BlockType:
    +1143
    +1144
    +1145
    async def create_block_type(self, block_type: BlockTypeCreate) -> BlockType:
         """
         Create a block type in the Prefect API.
         """
    @@ -20379,9 +20527,7 @@ 

    Source code in prefect/client/orchestration.py -
    1145
    -1146
    -1147
    +            
    1147
     1148
     1149
     1150
    @@ -20397,7 +20543,9 @@ 

    1160 1161 1162 -1163

    async def create_block_schema(self, block_schema: BlockSchemaCreate) -> BlockSchema:
    +1163
    +1164
    +1165
    async def create_block_schema(self, block_schema: BlockSchemaCreate) -> BlockSchema:
         """
         Create a block schema in the Prefect API.
         """
    @@ -20477,9 +20625,7 @@ 

    Source code in prefect/client/orchestration.py -
    1165
    -1166
    -1167
    +            
    1167
     1168
     1169
     1170
    @@ -20517,7 +20663,9 @@ 

    1202 1203 1204 -1205

    async def create_block_document(
    +1205
    +1206
    +1207
    async def create_block_document(
         self,
         block_document: Union[BlockDocument, BlockDocumentCreate],
         include_secrets: bool = True,
    @@ -20585,9 +20733,7 @@ 

    Source code in prefect/client/orchestration.py -
    1207
    -1208
    -1209
    +            
    1209
     1210
     1211
     1212
    @@ -20607,7 +20753,9 @@ 

    1226 1227 1228 -1229

    async def update_block_document(
    +1229
    +1230
    +1231
    async def update_block_document(
         self,
         block_document_id: UUID,
         block_document: BlockDocumentUpdate,
    @@ -20657,9 +20805,7 @@ 

    Source code in prefect/client/orchestration.py -
    1231
    -1232
    -1233
    +            
    1233
     1234
     1235
     1236
    @@ -20667,7 +20813,9 @@ 

    1238 1239 1240 -1241

    async def delete_block_document(self, block_document_id: UUID):
    +1241
    +1242
    +1243
    async def delete_block_document(self, block_document_id: UUID):
         """
         Delete a block document.
         """
    @@ -20705,9 +20853,7 @@ 

    Source code in prefect/client/orchestration.py -
    1243
    -1244
    -1245
    +            
    1245
     1246
     1247
     1248
    @@ -20716,7 +20862,9 @@ 

    1251 1252 1253 -1254

    async def read_block_type_by_slug(self, slug: str) -> BlockType:
    +1254
    +1255
    +1256
    async def read_block_type_by_slug(self, slug: str) -> BlockType:
         """
         Read a block type by its slug.
         """
    @@ -20755,9 +20903,7 @@ 

    Source code in prefect/client/orchestration.py -
    1256
    -1257
    -1258
    +            
    1258
     1259
     1260
     1261
    @@ -20771,7 +20917,9 @@ 

    1269 1270 1271 -1272

    async def read_block_schema_by_checksum(
    +1272
    +1273
    +1274
    async def read_block_schema_by_checksum(
         self, checksum: str, version: Optional[str] = None
     ) -> BlockSchema:
         """
    @@ -20815,9 +20963,7 @@ 

    Source code in prefect/client/orchestration.py -
    1274
    -1275
    -1276
    +            
    1276
     1277
     1278
     1279
    @@ -20833,7 +20979,9 @@ 

    1289 1290 1291 -1292

    async def update_block_type(self, block_type_id: UUID, block_type: BlockTypeUpdate):
    +1292
    +1293
    +1294
    async def update_block_type(self, block_type_id: UUID, block_type: BlockTypeUpdate):
         """
         Update a block document in the Prefect API.
         """
    @@ -20879,9 +21027,7 @@ 

    Source code in prefect/client/orchestration.py -
    1294
    -1295
    -1296
    +            
    1296
     1297
     1298
     1299
    @@ -20897,7 +21043,9 @@ 

    1309 1310 1311 -1312

    async def delete_block_type(self, block_type_id: UUID):
    +1312
    +1313
    +1314
    async def delete_block_type(self, block_type_id: UUID):
         """
         Delete a block type.
         """
    @@ -20969,9 +21117,7 @@ 

    Source code in prefect/client/orchestration.py -
    1314
    -1315
    -1316
    +            
    1316
     1317
     1318
     1319
    @@ -20979,7 +21125,9 @@ 

    1321 1322 1323 -1324

    async def read_block_types(self) -> List[BlockType]:
    +1324
    +1325
    +1326
    async def read_block_types(self) -> List[BlockType]:
         """
         Read all block types
         Raises:
    @@ -21043,9 +21191,7 @@ 

    Source code in prefect/client/orchestration.py -
    1326
    -1327
    -1328
    +            
    1328
     1329
     1330
     1331
    @@ -21053,7 +21199,9 @@ 

    1333 1334 1335 -1336

    async def read_block_schemas(self) -> List[BlockSchema]:
    +1336
    +1337
    +1338
    async def read_block_schemas(self) -> List[BlockSchema]:
         """
         Read all block schemas
         Raises:
    @@ -21169,9 +21317,7 @@ 

    Source code in prefect/client/orchestration.py -
    1338
    -1339
    -1340
    +            
    1340
     1341
     1342
     1343
    @@ -21195,7 +21341,9 @@ 

    1361 1362 1363 -1364

    async def get_most_recent_block_schema_for_block_type(
    +1364
    +1365
    +1366
    async def get_most_recent_block_schema_for_block_type(
         self,
         block_type_id: UUID,
     ) -> Optional[BlockSchema]:
    @@ -21345,9 +21493,7 @@ 

    Source code in prefect/client/orchestration.py -
    1366
    -1367
    -1368
    +            
    1368
     1369
     1370
     1371
    @@ -21381,7 +21527,9 @@ 

    1399 1400 1401 -1402

    async def read_block_document(
    +1402
    +1403
    +1404
    async def read_block_document(
         self,
         block_document_id: UUID,
         include_secrets: bool = True,
    @@ -21557,9 +21705,7 @@ 

    Source code in prefect/client/orchestration.py -
    1404
    -1405
    -1406
    +            
    1406
     1407
     1408
     1409
    @@ -21593,7 +21739,9 @@ 

    1437 1438 1439 -1440

    async def read_block_document_by_name(
    +1440
    +1441
    +1442
    async def read_block_document_by_name(
         self,
         name: str,
         block_type_slug: str,
    @@ -21757,9 +21905,7 @@ 

    Source code in prefect/client/orchestration.py -
    1442
    -1443
    -1444
    +            
    1444
     1445
     1446
     1447
    @@ -21790,7 +21936,9 @@ 

    1472 1473 1474 -1475

    async def read_block_documents(
    +1475
    +1476
    +1477
    async def read_block_documents(
         self,
         block_schema_type: Optional[str] = None,
         offset: Optional[int] = None,
    @@ -21947,9 +22095,7 @@ 

    Source code in prefect/client/orchestration.py -
    1477
    -1478
    -1479
    +            
    1479
     1480
     1481
     1482
    @@ -21974,7 +22120,9 @@ 

    1501 1502 1503 -1504

    async def read_block_documents_by_type(
    +1504
    +1505
    +1506
    async def read_block_documents_by_type(
         self,
         block_type_slug: str,
         offset: Optional[int] = None,
    @@ -22193,9 +22341,7 @@ 

    Source code in prefect/client/orchestration.py -
    1506
    -1507
    -1508
    +            
    1508
     1509
     1510
     1511
    @@ -22284,7 +22430,9 @@ 

    1594 1595 1596 -1597

    async def create_deployment(
    +1597
    +1598
    +1599
    async def create_deployment(
         self,
         flow_id: UUID,
         name: str,
    @@ -22457,9 +22605,7 @@ 

    Source code in prefect/client/orchestration.py -
    1659
    -1660
    -1661
    +            
    1661
     1662
     1663
     1664
    @@ -22477,7 +22623,9 @@ 

    1676 1677 1678 -1679

    async def read_deployment(
    +1679
    +1680
    +1681
    async def read_deployment(
         self,
         deployment_id: UUID,
     ) -> DeploymentResponse:
    @@ -22613,9 +22761,7 @@ 

    Source code in prefect/client/orchestration.py -
    1681
    -1682
    -1683
    +            
    1683
     1684
     1685
     1686
    @@ -22638,7 +22784,9 @@ 

    1703 1704 1705 -1706

    async def read_deployment_by_name(
    +1706
    +1707
    +1708
    async def read_deployment_by_name(
         self,
         name: str,
     ) -> DeploymentResponse:
    @@ -22845,9 +22993,7 @@ 

    Source code in prefect/client/orchestration.py -
    1708
    -1709
    -1710
    +            
    1710
     1711
     1712
     1713
    @@ -22907,7 +23053,9 @@ 

    1767 1768 1769 -1770

    async def read_deployments(
    +1770
    +1771
    +1772
    async def read_deployments(
         self,
         *,
         flow_filter: FlowFilter = None,
    @@ -23028,9 +23176,7 @@ 

    Source code in prefect/client/orchestration.py -
    1772
    -1773
    -1774
    +            
    1774
     1775
     1776
     1777
    @@ -23047,7 +23193,9 @@ 

    1788 1789 1790 -1791

    async def delete_deployment(
    +1791
    +1792
    +1793
    async def delete_deployment(
         self,
         deployment_id: UUID,
     ):
    @@ -23148,9 +23296,7 @@ 

    Source code in prefect/client/orchestration.py -
    1793
    -1794
    -1795
    +            
    1795
     1796
     1797
     1798
    @@ -23165,7 +23311,9 @@ 

    1807 1808 1809 -1810

    async def read_flow_run(self, flow_run_id: UUID) -> FlowRun:
    +1810
    +1811
    +1812
    async def read_flow_run(self, flow_run_id: UUID) -> FlowRun:
         """
         Query the Prefect API for a flow run by id.
     
    @@ -23278,9 +23426,7 @@ 

    Source code in prefect/client/orchestration.py -
    1812
    -1813
    -1814
    +            
    1814
     1815
     1816
     1817
    @@ -23298,7 +23444,9 @@ 

    1829 1830 1831 -1832

    async def resume_flow_run(
    +1832
    +1833
    +1834
    async def resume_flow_run(
         self, flow_run_id: UUID, run_input: Optional[Dict] = None
     ) -> OrchestrationResult:
         """
    @@ -23514,9 +23662,7 @@ 

    Source code in prefect/client/orchestration.py -
    1834
    -1835
    -1836
    +            
    1836
     1837
     1838
     1839
    @@ -23577,7 +23723,9 @@ 

    1894 1895 1896 -1897

    async def read_flow_runs(
    +1897
    +1898
    +1899
    async def read_flow_runs(
         self,
         *,
         flow_filter: FlowFilter = None,
    @@ -23751,9 +23899,7 @@ 

    Source code in prefect/client/orchestration.py -
    1899
    -1900
    -1901
    +            
    1901
     1902
     1903
     1904
    @@ -23782,7 +23928,9 @@ 

    1927 1928 1929 -1930

    async def set_flow_run_state(
    +1930
    +1931
    +1932
    async def set_flow_run_state(
         self,
         flow_run_id: UUID,
         state: "prefect.states.State",
    @@ -23896,9 +24044,7 @@ 

    Source code in prefect/client/orchestration.py -
    1932
    -1933
    -1934
    +            
    1934
     1935
     1936
     1937
    @@ -23912,7 +24058,9 @@ 

    1945 1946 1947 -1948

    async def read_flow_run_states(
    +1948
    +1949
    +1950
    async def read_flow_run_states(
         self, flow_run_id: UUID
     ) -> List[prefect.states.State]:
         """
    @@ -24096,9 +24244,7 @@ 

    Source code in prefect/client/orchestration.py -
    1957
    -1958
    -1959
    +            
    1959
     1960
     1961
     1962
    @@ -24156,7 +24302,9 @@ 

    2014 2015 2016 -2017

    async def create_task_run(
    +2017
    +2018
    +2019
    async def create_task_run(
         self,
         task: "TaskObject",
         flow_run_id: UUID,
    @@ -24298,9 +24446,7 @@ 

    Source code in prefect/client/orchestration.py -
    2019
    -2020
    -2021
    +            
    2021
     2022
     2023
     2024
    @@ -24309,7 +24455,9 @@ 

    2027 2028 2029 -2030

    async def read_task_run(self, task_run_id: UUID) -> TaskRun:
    +2030
    +2031
    +2032
    async def read_task_run(self, task_run_id: UUID) -> TaskRun:
         """
         Query the Prefect API for a task run by id.
     
    @@ -24488,9 +24636,7 @@ 

    Source code in prefect/client/orchestration.py -
    2032
    -2033
    -2034
    +            
    2034
     2035
     2036
     2037
    @@ -24536,7 +24682,9 @@ 

    2077 2078 2079 -2080

    async def read_task_runs(
    +2080
    +2081
    +2082
    async def read_task_runs(
         self,
         *,
         flow_filter: FlowFilter = None,
    @@ -24695,9 +24843,7 @@ 

    Source code in prefect/client/orchestration.py -
    2082
    -2083
    -2084
    +            
    2084
     2085
     2086
     2087
    @@ -24719,7 +24865,9 @@ 

    2103 2104 2105 -2106

    async def set_task_run_state(
    +2106
    +2107
    +2108
    async def set_task_run_state(
         self,
         task_run_id: UUID,
         state: prefect.states.State,
    @@ -24825,9 +24973,7 @@ 

    Source code in prefect/client/orchestration.py -
    2108
    -2109
    -2110
    +            
    2110
     2111
     2112
     2113
    @@ -24840,7 +24986,9 @@ 

    2120 2121 2122 -2123

    async def read_task_run_states(
    +2123
    +2124
    +2125
    async def read_task_run_states(
         self, task_run_id: UUID
     ) -> List[prefect.states.State]:
         """
    @@ -24913,9 +25061,7 @@ 

    Source code in prefect/client/orchestration.py -
    2125
    -2126
    -2127
    +            
    2127
     2128
     2129
     2130
    @@ -24924,7 +25070,9 @@ 

    2133 2134 2135 -2136

    async def create_logs(self, logs: Iterable[Union[LogCreate, dict]]) -> None:
    +2136
    +2137
    +2138
    async def create_logs(self, logs: Iterable[Union[LogCreate, dict]]) -> None:
         """
         Create logs for a flow or task run
     
    @@ -25049,9 +25197,7 @@ 

    Source code in prefect/client/orchestration.py -
    2138
    -2139
    -2140
    +            
    2140
     2141
     2142
     2143
    @@ -25088,7 +25234,9 @@ 

    2174 2175 2176 -2177

    async def create_flow_run_notification_policy(
    +2177
    +2178
    +2179
    async def create_flow_run_notification_policy(
         self,
         block_document_id: UUID,
         is_active: bool = True,
    @@ -25239,9 +25387,7 @@ 

    Source code in prefect/client/orchestration.py -
    2179
    -2180
    -2181
    +            
    2181
     2182
     2183
     2184
    @@ -25270,7 +25416,9 @@ 

    2207 2208 2209 -2210

    async def read_flow_run_notification_policies(
    +2210
    +2211
    +2212
    async def read_flow_run_notification_policies(
         self,
         flow_run_notification_policy_filter: FlowRunNotificationPolicyFilter,
         limit: Optional[int] = None,
    @@ -25329,9 +25477,7 @@ 

    Source code in prefect/client/orchestration.py -
    2212
    -2213
    -2214
    +            
    2214
     2215
     2216
     2217
    @@ -25347,7 +25493,9 @@ 

    2227 2228 2229 -2230

    async def read_logs(
    +2230
    +2231
    +2232
    async def read_logs(
         self,
         log_filter: LogFilter = None,
         limit: int = None,
    @@ -25448,9 +25596,7 @@ 

    Source code in prefect/client/orchestration.py -
    2232
    -2233
    -2234
    +            
    2234
     2235
     2236
     2237
    @@ -25477,7 +25623,9 @@ 

    2258 2259 2260 -2261

    async def resolve_datadoc(self, datadoc: DataDocument) -> Any:
    +2261
    +2262
    +2263
    async def resolve_datadoc(self, datadoc: DataDocument) -> Any:
         """
         Recursively decode possibly nested data documents.
     
    @@ -25578,9 +25726,7 @@ 

    Source code in prefect/client/orchestration.py -
    2263
    -2264
    -2265
    +            
    2265
     2266
     2267
     2268
    @@ -25597,7 +25743,9 @@ 

    2279 2280 2281 -2282

    async def send_worker_heartbeat(
    +2282
    +2283
    +2284
    async def send_worker_heartbeat(
         self,
         work_pool_name: str,
         worker_name: str,
    @@ -25717,9 +25865,7 @@ 

    Source code in prefect/client/orchestration.py -
    2284
    -2285
    -2286
    +            
    2286
     2287
     2288
     2289
    @@ -25747,7 +25893,9 @@ 

    2311 2312 2313 -2314

    async def read_workers_for_work_pool(
    +2314
    +2315
    +2316
    async def read_workers_for_work_pool(
         self,
         work_pool_name: str,
         worker_filter: Optional[WorkerFilter] = None,
    @@ -25860,9 +26008,7 @@ 

    Source code in prefect/client/orchestration.py -
    2316
    -2317
    -2318
    +            
    2318
     2319
     2320
     2321
    @@ -25878,7 +26024,9 @@ 

    2331 2332 2333 -2334

    async def read_work_pool(self, work_pool_name: str) -> WorkPool:
    +2334
    +2335
    +2336
    async def read_work_pool(self, work_pool_name: str) -> WorkPool:
         """
         Reads information for a given work pool
     
    @@ -26006,9 +26154,7 @@ 

    Source code in prefect/client/orchestration.py -
    2336
    -2337
    -2338
    +            
    2338
     2339
     2340
     2341
    @@ -26034,7 +26180,9 @@ 

    2361 2362 2363 -2364

    async def read_work_pools(
    +2364
    +2365
    +2366
    async def read_work_pools(
         self,
         limit: Optional[int] = None,
         offset: int = 0,
    @@ -26144,9 +26292,7 @@ 

    Source code in prefect/client/orchestration.py -
    2366
    -2367
    -2368
    +            
    2368
     2369
     2370
     2371
    @@ -26168,7 +26314,9 @@ 

    2387 2388 2389 -2390

    async def create_work_pool(
    +2390
    +2391
    +2392
    async def create_work_pool(
         self,
         work_pool: WorkPoolCreate,
     ) -> WorkPool:
    @@ -26264,9 +26412,7 @@ 

    Source code in prefect/client/orchestration.py -
    2392
    -2393
    -2394
    +            
    2394
     2395
     2396
     2397
    @@ -26285,7 +26431,9 @@ 

    2410 2411 2412 -2413

    async def update_work_pool(
    +2413
    +2414
    +2415
    async def update_work_pool(
         self,
         work_pool_name: str,
         work_pool: WorkPoolUpdate,
    @@ -26364,9 +26512,7 @@ 

    Source code in prefect/client/orchestration.py -
    2415
    -2416
    -2417
    +            
    2417
     2418
     2419
     2420
    @@ -26380,7 +26526,9 @@ 

    2428 2429 2430 -2431

    async def delete_work_pool(
    +2431
    +2432
    +2433
    async def delete_work_pool(
         self,
         work_pool_name: str,
     ):
    @@ -26520,9 +26668,7 @@ 

    Source code in prefect/client/orchestration.py -
    2433
    -2434
    -2435
    +            
    2435
     2436
     2437
     2438
    @@ -26563,7 +26709,9 @@ 

    2473 2474 2475 -2476

    async def read_work_queues(
    +2476
    +2477
    +2478
    async def read_work_queues(
         self,
         work_pool_name: Optional[str] = None,
         work_queue_filter: Optional[WorkQueueFilter] = None,
    @@ -26729,9 +26877,7 @@ 

    Source code in prefect/client/orchestration.py -
    2497
    -2498
    -2499
    +            
    2499
     2500
     2501
     2502
    @@ -26760,7 +26906,9 @@ 

    2525 2526 2527 -2528

    async def get_scheduled_flow_runs_for_work_pool(
    +2528
    +2529
    +2530
    async def get_scheduled_flow_runs_for_work_pool(
         self,
         work_pool_name: str,
         work_queue_names: Optional[List[str]] = None,
    @@ -26850,9 +26998,7 @@ 

    Source code in prefect/client/orchestration.py -
    2530
    -2531
    -2532
    +            
    2532
     2533
     2534
     2535
    @@ -26868,7 +27014,9 @@ 

    2545 2546 2547 -2548

    async def create_artifact(
    +2548
    +2549
    +2550
    async def create_artifact(
         self,
         artifact: ArtifactCreate,
     ) -> Artifact:
    @@ -26924,9 +27072,7 @@ 

    Source code in prefect/client/orchestration.py -
    2550
    -2551
    -2552
    +            
    2552
     2553
     2554
     2555
    @@ -26962,7 +27108,9 @@ 

    2585 2586 2587 -2588

    async def read_artifacts(
    +2588
    +2589
    +2590
    async def read_artifacts(
         self,
         *,
         artifact_filter: ArtifactFilter = None,
    @@ -27038,9 +27186,7 @@ 

    Source code in prefect/client/orchestration.py -
    2590
    -2591
    -2592
    +            
    2592
     2593
     2594
     2595
    @@ -27076,7 +27222,9 @@ 

    2625 2626 2627 -2628

    async def read_latest_artifacts(
    +2628
    +2629
    +2630
    async def read_latest_artifacts(
         self,
         *,
         artifact_filter: ArtifactCollectionFilter = None,
    @@ -27172,9 +27320,7 @@ 

    Source code in prefect/client/orchestration.py -
    2630
    -2631
    -2632
    +            
    2632
     2633
     2634
     2635
    @@ -27185,7 +27331,9 @@ 

    2640 2641 2642 -2643

    async def delete_artifact(self, artifact_id: UUID) -> None:
    +2643
    +2644
    +2645
    async def delete_artifact(self, artifact_id: UUID) -> None:
         """
         Deletes an artifact with the provided id.
     
    @@ -27226,16 +27374,16 @@ 

    Source code in prefect/client/orchestration.py -
    2645
    -2646
    -2647
    +            
    2647
     2648
     2649
     2650
     2651
     2652
     2653
    -2654
    async def read_variable_by_name(self, name: str) -> Optional[Variable]:
    +2654
    +2655
    +2656
    async def read_variable_by_name(self, name: str) -> Optional[Variable]:
         """Reads a variable by name. Returns None if no variable is found."""
         try:
             response = await self._client.get(f"/variables/name/{name}")
    @@ -27272,15 +27420,15 @@ 

    Source code in prefect/client/orchestration.py -
    2656
    -2657
    -2658
    +            
    2658
     2659
     2660
     2661
     2662
     2663
    -2664
    async def delete_variable_by_name(self, name: str):
    +2664
    +2665
    +2666
    async def delete_variable_by_name(self, name: str):
         """Deletes a variable by name."""
         try:
             await self._client.delete(f"/variables/name/{name}")
    @@ -27316,10 +27464,10 @@ 

    Source code in prefect/client/orchestration.py -
    2666
    -2667
    -2668
    -2669
    async def read_variables(self, limit: int = None) -> List[Variable]:
    +            
    2668
    +2669
    +2670
    +2671
    async def read_variables(self, limit: int = None) -> List[Variable]:
         """Reads all variables."""
         response = await self._client.post("/variables/filter", json={"limit": limit})
         return pydantic.parse_obj_as(List[Variable], response.json())
    @@ -27350,11 +27498,11 @@ 

    Source code in prefect/client/orchestration.py -
    2671
    -2672
    -2673
    +            
    2673
     2674
    -2675
    async def read_worker_metadata(self) -> Dict[str, Any]:
    +2675
    +2676
    +2677
    async def read_worker_metadata(self) -> Dict[str, Any]:
         """Reads worker metadata stored in Prefect collection registry."""
         response = await self._client.get("collections/views/aggregate-worker-metadata")
         response.raise_for_status()
    @@ -27386,9 +27534,7 @@ 

    Source code in prefect/client/orchestration.py -
    2677
    -2678
    -2679
    +            
    2679
     2680
     2681
     2682
    @@ -27396,7 +27542,9 @@ 

    2684 2685 2686 -2687

    async def create_automation(self, automation: Automation) -> UUID:
    +2687
    +2688
    +2689
    async def create_automation(self, automation: Automation) -> UUID:
         """Creates an automation in Prefect Cloud."""
         if self.server_type != ServerType.CLOUD:
             raise RuntimeError("Automations are only supported for Prefect Cloud.")
    @@ -27506,27 +27654,27 @@ 

    Source code in prefect/client/orchestration.py -
    2725
    -2726
    -2727
    -2728
    -2729
    -2730
    -2731
    -2732
    -2733
    -2734
    -2735
    -2736
    -2737
    -2738
    -2739
    -2740
    -2741
    -2742
    -2743
    -2744
    -2745
    async def create_flow_run_input(
    +            
    2787
    +2788
    +2789
    +2790
    +2791
    +2792
    +2793
    +2794
    +2795
    +2796
    +2797
    +2798
    +2799
    +2800
    +2801
    +2802
    +2803
    +2804
    +2805
    +2806
    +2807
    async def create_flow_run_input(
         self, flow_run_id: UUID, key: str, value: str, sender: Optional[str] = None
     ):
         """
    @@ -27618,17 +27766,17 @@ 

    Source code in prefect/client/orchestration.py -
    2761
    -2762
    -2763
    -2764
    -2765
    -2766
    -2767
    -2768
    -2769
    -2770
    -2771
    async def read_flow_run_input(self, flow_run_id: UUID, key: str) -> str:
    +            
    2823
    +2824
    +2825
    +2826
    +2827
    +2828
    +2829
    +2830
    +2831
    +2832
    +2833
    async def read_flow_run_input(self, flow_run_id: UUID, key: str) -> str:
         """
         Reads a flow run input.
     
    @@ -27710,16 +27858,16 @@ 

    Source code in prefect/client/orchestration.py -
    2773
    -2774
    -2775
    -2776
    -2777
    -2778
    -2779
    -2780
    -2781
    -2782
    async def delete_flow_run_input(self, flow_run_id: UUID, key: str):
    +            
    2835
    +2836
    +2837
    +2838
    +2839
    +2840
    +2841
    +2842
    +2843
    +2844
    async def delete_flow_run_input(self, flow_run_id: UUID, key: str):
         """
         Deletes a flow run input.
     
    @@ -27766,9 +27914,7 @@ 

    Source code in prefect/client/orchestration.py -
    141
    -142
    -143
    +            
    143
     144
     145
     146
    @@ -27790,7 +27936,9 @@ 

    162 163 164 -165

    def get_client(httpx_settings: Optional[dict] = None) -> "PrefectClient":
    +165
    +166
    +167
    def get_client(httpx_settings: Optional[dict] = None) -> "PrefectClient":
         """
         Retrieve a HTTP client for communicating with the Prefect REST API.
     
    diff --git a/versions/unreleased/api-ref/prefect/client/schemas/index.html b/versions/unreleased/api-ref/prefect/client/schemas/index.html
    index aff2722a43..7ebd5a1a7e 100644
    --- a/versions/unreleased/api-ref/prefect/client/schemas/index.html
    +++ b/versions/unreleased/api-ref/prefect/client/schemas/index.html
    @@ -848,6 +848,8 @@
             
           
             
    +      
    +        
           
             
           
    @@ -1104,6 +1106,32 @@
       
       
       
    +    
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • @@ -4000,6 +4028,28 @@ +
  • + +
  • + + + + GlobalConcurrencyLimitCreate + + + + +
  • + +
  • + + + + GlobalConcurrencyLimitUpdate + + + +
  • @@ -5477,6 +5527,17 @@ +
  • + +
  • + + + + GlobalConcurrencyLimit + + + +
  • @@ -11098,6 +11159,28 @@ +
  • + +
  • + + + + GlobalConcurrencyLimitCreate + + + + +
  • + +
  • + + + + GlobalConcurrencyLimitUpdate + + + +
  • @@ -12575,6 +12658,17 @@ +
  • + +
  • + + + + GlobalConcurrencyLimit + + + +
  • @@ -15429,6 +15523,128 @@

    +

  • + + + + + + +
    + + + +

    + GlobalConcurrencyLimitCreate + + +

    + + +
    +

    + Bases: ActionBaseModel

    + + +

    Data used by the Prefect REST API to create a global concurrency limit.

    + +
    + Source code in prefect/client/schemas/actions.py +
    628
    +629
    +630
    +631
    +632
    +633
    +634
    +635
    +636
    @copy_model_fields
    +class GlobalConcurrencyLimitCreate(ActionBaseModel):
    +    """Data used by the Prefect REST API to create a global concurrency limit."""
    +
    +    name: str = FieldFrom(objects.GlobalConcurrencyLimit)
    +    limit: int = FieldFrom(objects.GlobalConcurrencyLimit)
    +    active: Optional[bool] = FieldFrom(objects.GlobalConcurrencyLimit)
    +    active_slots: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)
    +    slot_decay_per_second: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)
    +
    +
    + + + +
    + + + + + + + + + + + +
    + +
    + + +
    + +
    + + + +

    + GlobalConcurrencyLimitUpdate + + +

    + + +
    +

    + Bases: ActionBaseModel

    + + +

    Data used by the Prefect REST API to update a global concurrency limit.

    + +
    + Source code in prefect/client/schemas/actions.py +
    639
    +640
    +641
    +642
    +643
    +644
    +645
    +646
    +647
    @copy_model_fields
    +class GlobalConcurrencyLimitUpdate(ActionBaseModel):
    +    """Data used by the Prefect REST API to update a global concurrency limit."""
    +
    +    name: Optional[str] = FieldFrom(objects.GlobalConcurrencyLimit)
    +    limit: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)
    +    active: Optional[bool] = FieldFrom(objects.GlobalConcurrencyLimit)
    +    active_slots: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)
    +    slot_decay_per_second: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)
    +
    +
    + + + +
    + + + + + + + + + + +
    @@ -25416,6 +25632,99 @@

    + + + +

    + GlobalConcurrencyLimit + + +

    + + +
    +

    + Bases: ObjectBaseModel

    + + +

    An ORM representation of a global concurrency limit

    + +
    + Source code in prefect/client/schemas/objects.py +
    1566
    +1567
    +1568
    +1569
    +1570
    +1571
    +1572
    +1573
    +1574
    +1575
    +1576
    +1577
    +1578
    +1579
    +1580
    +1581
    +1582
    +1583
    +1584
    +1585
    +1586
    +1587
    +1588
    +1589
    +1590
    class GlobalConcurrencyLimit(ObjectBaseModel):
    +    """An ORM representation of a global concurrency limit"""
    +
    +    name: str = Field(description="The name of the global concurrency limit.")
    +    limit: int = Field(
    +        description=(
    +            "The maximum number of slots that can be occupied on this concurrency"
    +            " limit."
    +        )
    +    )
    +    active: Optional[bool] = Field(
    +        default=True,
    +        description="Whether or not the concurrency limit is in an active state.",
    +    )
    +    active_slots: Optional[int] = Field(
    +        default=0,
    +        description="Number of tasks currently using a concurrency slot.",
    +    )
    +    slot_decay_per_second: Optional[int] = Field(
    +        default=0,
    +        description=(
    +            "Controls the rate at which slots are released when the concurrency limit"
    +            " is used as a rate limit."
    +        ),
    +    )
    +
    +
    + + + +
    + + + + + + + + + + +
    diff --git a/versions/unreleased/api-ref/prefect/client/utilities/index.html b/versions/unreleased/api-ref/prefect/client/utilities/index.html index 4b2fae3df9..9a5438333b 100644 --- a/versions/unreleased/api-ref/prefect/client/utilities/index.html +++ b/versions/unreleased/api-ref/prefect/client/utilities/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/concurrency/asyncio/index.html b/versions/unreleased/api-ref/prefect/concurrency/asyncio/index.html index 3d77aac370..a5b3da779f 100644 --- a/versions/unreleased/api-ref/prefect/concurrency/asyncio/index.html +++ b/versions/unreleased/api-ref/prefect/concurrency/asyncio/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/concurrency/common/index.html b/versions/unreleased/api-ref/prefect/concurrency/common/index.html index 1aa171a6c3..365f560fce 100644 --- a/versions/unreleased/api-ref/prefect/concurrency/common/index.html +++ b/versions/unreleased/api-ref/prefect/concurrency/common/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/concurrency/events/index.html b/versions/unreleased/api-ref/prefect/concurrency/events/index.html index e46f04c018..c5cdfd9ee3 100644 --- a/versions/unreleased/api-ref/prefect/concurrency/events/index.html +++ b/versions/unreleased/api-ref/prefect/concurrency/events/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/concurrency/services/index.html b/versions/unreleased/api-ref/prefect/concurrency/services/index.html index 0fc0fd4be7..5f075e2008 100644 --- a/versions/unreleased/api-ref/prefect/concurrency/services/index.html +++ b/versions/unreleased/api-ref/prefect/concurrency/services/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/concurrency/sync/index.html b/versions/unreleased/api-ref/prefect/concurrency/sync/index.html index d614884c51..28c631df9a 100644 --- a/versions/unreleased/api-ref/prefect/concurrency/sync/index.html +++ b/versions/unreleased/api-ref/prefect/concurrency/sync/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/context/index.html b/versions/unreleased/api-ref/prefect/context/index.html index 331d5a0d86..1434002a2a 100644 --- a/versions/unreleased/api-ref/prefect/context/index.html +++ b/versions/unreleased/api-ref/prefect/context/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/deployments/base/index.html b/versions/unreleased/api-ref/prefect/deployments/base/index.html index 11dbbf2fb1..1b385d54e9 100644 --- a/versions/unreleased/api-ref/prefect/deployments/base/index.html +++ b/versions/unreleased/api-ref/prefect/deployments/base/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/deployments/deployments/index.html b/versions/unreleased/api-ref/prefect/deployments/deployments/index.html index 586444e8a1..3645e83581 100644 --- a/versions/unreleased/api-ref/prefect/deployments/deployments/index.html +++ b/versions/unreleased/api-ref/prefect/deployments/deployments/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/deployments/runner/index.html b/versions/unreleased/api-ref/prefect/deployments/runner/index.html index f17d0b1bcf..124a1c2cf9 100644 --- a/versions/unreleased/api-ref/prefect/deployments/runner/index.html +++ b/versions/unreleased/api-ref/prefect/deployments/runner/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/deployments/steps/core/index.html b/versions/unreleased/api-ref/prefect/deployments/steps/core/index.html index a4653e08f7..50db5b8612 100644 --- a/versions/unreleased/api-ref/prefect/deployments/steps/core/index.html +++ b/versions/unreleased/api-ref/prefect/deployments/steps/core/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/deployments/steps/pull/index.html b/versions/unreleased/api-ref/prefect/deployments/steps/pull/index.html index 850068b8a8..08af6803ff 100644 --- a/versions/unreleased/api-ref/prefect/deployments/steps/pull/index.html +++ b/versions/unreleased/api-ref/prefect/deployments/steps/pull/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/deployments/steps/utility/index.html b/versions/unreleased/api-ref/prefect/deployments/steps/utility/index.html index 2130f6ec58..85dcdb8696 100644 --- a/versions/unreleased/api-ref/prefect/deployments/steps/utility/index.html +++ b/versions/unreleased/api-ref/prefect/deployments/steps/utility/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/engine/index.html b/versions/unreleased/api-ref/prefect/engine/index.html index 2651eb0f80..4f06b66eba 100644 --- a/versions/unreleased/api-ref/prefect/engine/index.html +++ b/versions/unreleased/api-ref/prefect/engine/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/events/index.html b/versions/unreleased/api-ref/prefect/events/index.html index 6f00b00eef..cdd57502ad 100644 --- a/versions/unreleased/api-ref/prefect/events/index.html +++ b/versions/unreleased/api-ref/prefect/events/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/exceptions/index.html b/versions/unreleased/api-ref/prefect/exceptions/index.html index 460e950a9d..325d9967e1 100644 --- a/versions/unreleased/api-ref/prefect/exceptions/index.html +++ b/versions/unreleased/api-ref/prefect/exceptions/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/filesystems/index.html b/versions/unreleased/api-ref/prefect/filesystems/index.html index 59fe560222..6105db213e 100644 --- a/versions/unreleased/api-ref/prefect/filesystems/index.html +++ b/versions/unreleased/api-ref/prefect/filesystems/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/flows/index.html b/versions/unreleased/api-ref/prefect/flows/index.html index 0a9853b70e..dc52acaedc 100644 --- a/versions/unreleased/api-ref/prefect/flows/index.html +++ b/versions/unreleased/api-ref/prefect/flows/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/futures/index.html b/versions/unreleased/api-ref/prefect/futures/index.html index 3c6de5df00..00c9296c53 100644 --- a/versions/unreleased/api-ref/prefect/futures/index.html +++ b/versions/unreleased/api-ref/prefect/futures/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/infrastructure/index.html b/versions/unreleased/api-ref/prefect/infrastructure/index.html index f25b381e7c..f53d2f8ede 100644 --- a/versions/unreleased/api-ref/prefect/infrastructure/index.html +++ b/versions/unreleased/api-ref/prefect/infrastructure/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/input/actions/index.html b/versions/unreleased/api-ref/prefect/input/actions/index.html index 463f1ebed8..1b569e2cfe 100644 --- a/versions/unreleased/api-ref/prefect/input/actions/index.html +++ b/versions/unreleased/api-ref/prefect/input/actions/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/input/run_input/index.html b/versions/unreleased/api-ref/prefect/input/run_input/index.html index b613bb3d62..5ee48d6e2a 100644 --- a/versions/unreleased/api-ref/prefect/input/run_input/index.html +++ b/versions/unreleased/api-ref/prefect/input/run_input/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • @@ -8784,6 +8812,58 @@

    + +

    This module contains functions that allow sending type-checked RunInput data +to flows at runtime. Flows can send back responses, establishing two-way +channels with senders. These functions are particularly useful for systems that +require ongoing data transfer or need to react to input quickly. +real-time interaction and efficient data handling. It's designed to facilitate +dynamic communication within distributed or microservices-oriented systems, +making it ideal for scenarios requiring continuous data synchronization and +processing. It's particularly useful for systems that require ongoing data +input and output.

    +

    The following is an example of two flows. One sends a random number to the +other and waits for a response. The other receives the number, squares it, and +sends the result back. The sender flow then prints the result.

    +

    Sender flow:

    +
    import random
    +from uuid import UUID
    +from prefect import flow, get_run_logger
    +from prefect.input import RunInput
    +
    +class NumberData(RunInput):
    +    number: int
    +
    +
    +@flow
    +async def sender_flow(receiver_flow_run_id: UUID):
    +    logger = get_run_logger()
    +
    +    the_number = random.randint(1, 100)
    +
    +    await NumberData(number=the_number).send_to(receiver_flow_run_id)
    +
    +    receiver = NumberData.receive(flow_run_id=receiver_flow_run_id)
    +    squared = await receiver.next()
    +
    +    logger.info(f"{the_number} squared is {squared.number}")
    +
    +

    Receiver flow: +

    import random
    +from uuid import UUID
    +from prefect import flow, get_run_logger
    +from prefect.input import RunInput
    +
    +class NumberData(RunInput):
    +    number: int
    +
    +
    +@flow
    +async def receiver_flow():
    +    async for data in NumberData.receive():
    +        squared = data.number ** 2
    +        data.respond(NumberData(number=squared))
    +

    @@ -8814,68 +8894,7 @@

    Source code in prefect/input/run_input.py -
     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
    +              
    137
     138
     139
     140
    @@ -8955,7 +8974,68 @@ 

    214 215 216 -217

    class RunInput(pydantic.BaseModel):
    +217
    +218
    +219
    +220
    +221
    +222
    +223
    +224
    +225
    +226
    +227
    +228
    +229
    +230
    +231
    +232
    +233
    +234
    +235
    +236
    +237
    +238
    +239
    +240
    +241
    +242
    +243
    +244
    +245
    +246
    +247
    +248
    +249
    +250
    +251
    +252
    +253
    +254
    +255
    +256
    +257
    +258
    +259
    +260
    +261
    +262
    +263
    +264
    +265
    +266
    +267
    +268
    +269
    +270
    +271
    +272
    +273
    +274
    +275
    +276
    +277
    +278
    class RunInput(pydantic.BaseModel):
         class Config:
             extra = "forbid"
     
    @@ -9178,23 +9258,23 @@ 

    Source code in prefect/input/run_input.py -
    110
    -111
    -112
    -113
    -114
    -115
    -116
    -117
    -118
    -119
    -120
    -121
    -122
    -123
    -124
    -125
    -126
    @classmethod
    +            
    171
    +172
    +173
    +174
    +175
    +176
    +177
    +178
    +179
    +180
    +181
    +182
    +183
    +184
    +185
    +186
    +187
    @classmethod
     @sync_compatible
     async def load(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):
         """
    @@ -9268,21 +9348,21 @@ 

    Source code in prefect/input/run_input.py -
    128
    -129
    -130
    -131
    -132
    -133
    -134
    -135
    -136
    -137
    -138
    -139
    -140
    -141
    -142
    @classmethod
    +            
    189
    +190
    +191
    +192
    +193
    +194
    +195
    +196
    +197
    +198
    +199
    +200
    +201
    +202
    +203
    @classmethod
     def load_from_flow_run_input(cls, flow_run_input: "FlowRunInput"):
         """
         Load the run input from a FlowRunInput object.
    @@ -9369,25 +9449,25 @@ 

    Source code in prefect/input/run_input.py -
     90
    - 91
    - 92
    - 93
    - 94
    - 95
    - 96
    - 97
    - 98
    - 99
    -100
    -101
    -102
    -103
    -104
    -105
    -106
    -107
    -108
    @classmethod
    +            
    151
    +152
    +153
    +154
    +155
    +156
    +157
    +158
    +159
    +160
    +161
    +162
    +163
    +164
    +165
    +166
    +167
    +168
    +169
    @classmethod
     @sync_compatible
     async def save(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):
         """
    @@ -9464,19 +9544,19 @@ 

    Source code in prefect/input/run_input.py -
    144
    -145
    -146
    -147
    -148
    -149
    -150
    -151
    -152
    -153
    -154
    -155
    -156
    @classmethod
    +            
    205
    +206
    +207
    +208
    +209
    +210
    +211
    +212
    +213
    +214
    +215
    +216
    +217
    @classmethod
     def with_initial_data(cls: Type[T], **kwargs: Any) -> Type[T]:
         """
         Create a new `RunInput` subclass with the given initial data as field
    @@ -9578,20 +9658,20 @@ 

    Source code in prefect/input/run_input.py -
    54
    -55
    -56
    -57
    -58
    -59
    -60
    -61
    -62
    -63
    -64
    -65
    -66
    -67
    def keyset_from_base_key(base_key: str) -> Keyset:
    +            
    115
    +116
    +117
    +118
    +119
    +120
    +121
    +122
    +123
    +124
    +125
    +126
    +127
    +128
    def keyset_from_base_key(base_key: str) -> Keyset:
         """
         Get the keyset for the given base key.
     
    @@ -9658,19 +9738,19 @@ 

    Source code in prefect/input/run_input.py -
    39
    -40
    -41
    -42
    -43
    -44
    -45
    -46
    -47
    -48
    -49
    -50
    -51
    def keyset_from_paused_state(state: "State") -> Keyset:
    +            
    + + + + diff --git a/versions/unreleased/guides/logs/index.html b/versions/unreleased/guides/logs/index.html index 856d04026d..c65fe9c853 100644 --- a/versions/unreleased/guides/logs/index.html +++ b/versions/unreleased/guides/logs/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/managed-execution/index.html b/versions/unreleased/guides/managed-execution/index.html index b851fcf73b..36483721e8 100644 --- a/versions/unreleased/guides/managed-execution/index.html +++ b/versions/unreleased/guides/managed-execution/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/migration-guide/index.html b/versions/unreleased/guides/migration-guide/index.html index 60d0d31964..b782448990 100644 --- a/versions/unreleased/guides/migration-guide/index.html +++ b/versions/unreleased/guides/migration-guide/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/moving-data/index.html b/versions/unreleased/guides/moving-data/index.html index fd4f6e68b3..68acf065fc 100644 --- a/versions/unreleased/guides/moving-data/index.html +++ b/versions/unreleased/guides/moving-data/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/prefect-deploy/index.html b/versions/unreleased/guides/prefect-deploy/index.html index d6de78f1ce..9cfd4cb54a 100644 --- a/versions/unreleased/guides/prefect-deploy/index.html +++ b/versions/unreleased/guides/prefect-deploy/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/runtime-context/index.html b/versions/unreleased/guides/runtime-context/index.html index 51d58c710b..524449a654 100644 --- a/versions/unreleased/guides/runtime-context/index.html +++ b/versions/unreleased/guides/runtime-context/index.html @@ -852,6 +852,8 @@ + + @@ -1180,6 +1182,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/settings/index.html b/versions/unreleased/guides/settings/index.html index d5f6d3df0f..d4d994c3c3 100644 --- a/versions/unreleased/guides/settings/index.html +++ b/versions/unreleased/guides/settings/index.html @@ -852,6 +852,8 @@ + + @@ -1406,6 +1408,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/state-change-hooks/index.html b/versions/unreleased/guides/state-change-hooks/index.html index f30b003691..e88548652e 100644 --- a/versions/unreleased/guides/state-change-hooks/index.html +++ b/versions/unreleased/guides/state-change-hooks/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/testing/index.html b/versions/unreleased/guides/testing/index.html index 14aa973e13..3914059d91 100644 --- a/versions/unreleased/guides/testing/index.html +++ b/versions/unreleased/guides/testing/index.html @@ -852,6 +852,8 @@ + + @@ -1180,6 +1182,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/troubleshooting/index.html b/versions/unreleased/guides/troubleshooting/index.html index b3791d1437..cc51e411a7 100644 --- a/versions/unreleased/guides/troubleshooting/index.html +++ b/versions/unreleased/guides/troubleshooting/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/upgrade-guide-agents-to-workers/index.html b/versions/unreleased/guides/upgrade-guide-agents-to-workers/index.html index 3fef95c794..7f2b75d869 100644 --- a/versions/unreleased/guides/upgrade-guide-agents-to-workers/index.html +++ b/versions/unreleased/guides/upgrade-guide-agents-to-workers/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/using-the-client/index.html b/versions/unreleased/guides/using-the-client/index.html index b8aeeeea73..5847fecf7d 100644 --- a/versions/unreleased/guides/using-the-client/index.html +++ b/versions/unreleased/guides/using-the-client/index.html @@ -852,6 +852,8 @@ + + @@ -1212,6 +1214,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/variables/index.html b/versions/unreleased/guides/variables/index.html index 560387a4a1..e3c896917d 100644 --- a/versions/unreleased/guides/variables/index.html +++ b/versions/unreleased/guides/variables/index.html @@ -852,6 +852,8 @@ + + @@ -1257,6 +1259,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/webhooks/index.html b/versions/unreleased/guides/webhooks/index.html index 01393300d2..c263e7107c 100644 --- a/versions/unreleased/guides/webhooks/index.html +++ b/versions/unreleased/guides/webhooks/index.html @@ -13,7 +13,7 @@ - + @@ -852,6 +852,8 @@ + + @@ -1107,6 +1109,32 @@ + +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + diff --git a/versions/unreleased/img/guides/automation-custom.png b/versions/unreleased/img/guides/automation-custom.png new file mode 100644 index 0000000000000000000000000000000000000000..1854c3e751b3aea0aeaed005a41bbbf92539c9a1 GIT binary patch literal 169674 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 VbohN(Ed%&dxpehnuF}n){sWi5FPs1X literal 0 HcmV?d00001 diff --git a/versions/unreleased/img/guides/automation-triggers.png b/versions/unreleased/img/guides/automation-triggers.png new file mode 100644 index 0000000000000000000000000000000000000000..e44cc128f6e34052b5a4f5ae27e6485562204f22 GIT binary patch literal 46229 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 BgysMM literal 0 HcmV?d00001 diff --git a/versions/unreleased/img/guides/block-list.png b/versions/unreleased/img/guides/block-list.png new file mode 100644 index 0000000000000000000000000000000000000000..46563df7ffd854d436039fb1c49cedd2d4fdb161 GIT binary patch literal 483683 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%0AR)Rk5BKKcWP>K= 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 ?nomiO=r|IuyVgQ(Dh7E%ZR8Q4k5=wXBu6?r~VL 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%0D?j-1>ruZ>41w#0pELi=U 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 KcN7>>{{9b{hE@{* literal 0 HcmV?d00001 diff --git a/versions/unreleased/img/guides/final-automation.png b/versions/unreleased/img/guides/final-automation.png new file mode 100644 index 0000000000000000000000000000000000000000..e5f162e45cc74897f16b3a614180cc26e7c2c589 GIT binary patch literal 36884 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 XyWlNiDthV=_@l0>bGuCWVeo$dQ8k+j literal 0 HcmV?d00001 diff --git a/versions/unreleased/img/guides/notification-block.png b/versions/unreleased/img/guides/notification-block.png new file mode 100644 index 0000000000000000000000000000000000000000..ab8b2a51f63817f03dff5a2cb0e7373308f9e7b3 GIT binary patch literal 114616 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 z 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 literal 0 HcmV?d00001 diff --git a/versions/unreleased/img/guides/webhook-automate.png b/versions/unreleased/img/guides/webhook-automate.png new file mode 100644 index 0000000000000000000000000000000000000000..50bac511c5fd29d76e3c17acbbdd72290b2c0e6d GIT binary patch literal 50890 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 literal 0 HcmV?d00001 diff --git a/versions/unreleased/img/guides/webhook-created.png b/versions/unreleased/img/guides/webhook-created.png new file mode 100644 index 0000000000000000000000000000000000000000..f203570cbcf24e343edb30554eed24a76c4a00bf GIT binary patch literal 50277 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 ZCGJ)2yi0O&|2^<0E2$(=CT1M?e*m3?&+PyJ literal 0 HcmV?d00001 diff --git a/versions/unreleased/img/guides/webhook-simple.png b/versions/unreleased/img/guides/webhook-simple.png new file mode 100644 index 0000000000000000000000000000000000000000..dd3487f67e95de3606a4d66217cb3836008bdf52 GIT binary patch literal 51651 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 E1FXkQ9smFU literal 0 HcmV?d00001 diff --git a/versions/unreleased/index.html b/versions/unreleased/index.html index b705ff105d..a85c854e1c 100644 --- a/versions/unreleased/index.html +++ b/versions/unreleased/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ + + + + + + + + + +
  • diff --git a/versions/unreleased/integrations/contribute/index.html b/versions/unreleased/integrations/contribute/index.html index 5a7ede2aab..f7d213da63 100644 --- a/versions/unreleased/integrations/contribute/index.html +++ b/versions/unreleased/integrations/contribute/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/integrations/index.html b/versions/unreleased/integrations/index.html index 233b45e49f..8de4fa2ffd 100644 --- a/versions/unreleased/integrations/index.html +++ b/versions/unreleased/integrations/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/integrations/usage/index.html b/versions/unreleased/integrations/usage/index.html index 1bee308993..51be56926e 100644 --- a/versions/unreleased/integrations/usage/index.html +++ b/versions/unreleased/integrations/usage/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/objects.inv b/versions/unreleased/objects.inv index 5d00e3cac1732918d669756232346d1216e65b0f..a2f787ff2c85f111a1170aef94b05b7925dcbcad 100644 GIT binary patch delta 18861 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 AtpET3 delta 18830 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 + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/search/search_index.json b/versions/unreleased/search/search_index.json index e608e8f10d..58dc42032f 100644 --- a/versions/unreleased/search/search_index.json +++ b/versions/unreleased/search/search_index.json @@ -1 +1 @@ -{"config":{"lang":["en"],"separator":"[\\s\\-]+","pipeline":["stopWordFilter"],"fields":{"title":{"boost":1000.0},"text":{"boost":1.0},"tags":{"boost":1000000.0}}},"docs":[{"location":"","title":"Welcome to Prefect","text":"

    Prefect is a workflow orchestration tool empowering developers to build, observe, and react to data pipelines.

    It's the easiest way to transform any Python function into a unit of work that can be observed and orchestrated. Just bring your Python code, sprinkle in a few decorators, and go!

    With Prefect you gain:

    • scheduling
    • retries
    • logging
    • convenient async functionality
    • caching
    • notifications
    • observability
    • event-based orchestration

    ","tags":["getting started","quick start","overview"],"boost":2},{"location":"#new-to-prefect","title":"New to Prefect?","text":"

    Get up and running quickly with the quickstart guide.

    Want more hands on practice to productionize your workflows? Follow our tutorial.

    For deeper dives on common use cases, explore our guides.

    Take your understanding even further with Prefect's concepts and API reference.

    Join Prefect's vibrant community of over 26,000 engineers to learn with others and share your knowledge!

    Need help?

    Get your questions answered by a Prefect Product Advocate! Book a Meeting

    ","tags":["getting started","quick start","overview"],"boost":2},{"location":"faq/","title":"Frequently Asked Questions","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#prefect","title":"Prefect","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#how-is-prefect-licensed","title":"How is Prefect licensed?","text":"

    Prefect is licensed under the Apache 2.0 License, an OSI approved open-source license. If you have any questions about licensing, please contact us.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#is-the-prefect-v2-cloud-url-different-than-the-prefect-v1-cloud-url","title":"Is the Prefect v2 Cloud URL different than the Prefect v1 Cloud URL?","text":"

    Yes. Prefect Cloud for v2 is at app.prefect.cloud/ while Prefect Cloud for v1 is at cloud.prefect.io.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#the-prefect-orchestration-engine","title":"The Prefect Orchestration Engine","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#why-was-the-prefect-orchestration-engine-created","title":"Why was the Prefect orchestration engine created?","text":"

    The Prefect orchestration engine has three major objectives:

    • Embracing dynamic, DAG-free workflows
    • An extraordinary developer experience
    • Transparent and observable orchestration rules

    As Prefect has matured, so has the modern data stack. The on-demand, dynamic, highly scalable workflows that used to exist principally in the domain of data science and analytics are now prevalent throughout all of data engineering. Few companies have workflows that don\u2019t deal with streaming data, uncertain timing, runtime logic, complex dependencies, versioning, or custom scheduling.

    This means that the current generation of workflow managers are built around the wrong abstraction: the directed acyclic graph (DAG). DAGs are an increasingly arcane, constrained way of representing the dynamic, heterogeneous range of modern data and computation patterns.

    Furthermore, as workflows have become more complex, it has become even more important to focus on the developer experience of building, testing, and monitoring them. Faced with an explosion of available tools, it is more important than ever for development teams to seek orchestration tools that will be compatible with any code, tools, or services they may require in the future.

    And finally, this additional complexity means that providing clear and consistent insight into the behavior of the orchestration engine and any decisions it makes is critically important.

    The Prefect orchestration engine represents a unified solution to these three problems.

    The Prefect orchestration engine is capable of governing any code through a well-defined series of state transitions designed to maximize the user's understanding of what happened during execution. It's popular to describe \"workflows as code\" or \"orchestration as code,\" but the Prefect engine represents \"code as workflows\": rather than ask users to change how they work to meet the requirements of the orchestrator, we've defined an orchestrator that adapts to how our users work.

    To achieve this, we've leveraged the familiar tools of native Python: first class functions, type annotations, and async support. Users are free to implement as much \u2014 or as little \u2014 of the Prefect engine as is useful for their objectives.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#if-im-using-prefect-cloud-2-do-i-still-need-to-run-a-prefect-server-locally","title":"If I\u2019m using Prefect Cloud 2, do I still need to run a Prefect server locally?","text":"

    No, Prefect Cloud hosts an instance of the Prefect API for you. In fact, each workspace in Prefect Cloud corresponds directly to a single instance of the Prefect orchestration engine. See the Prefect Cloud Overview for more information.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#features","title":"Features","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#does-prefect-support-mapping","title":"Does Prefect support mapping?","text":"

    Yes! For more information, see the Task.map API reference

    @flow\ndef my_flow():\n\n    # map over a constant\n    for i in range(10):\n        my_mapped_task(i)\n\n    # map over a task's output\n    l = list_task()\n    for i in l.wait().result():\n        my_mapped_task_2(i)\n

    Note that when tasks are called on constant values, they cannot detect their upstream edges automatically. In this example, my_mapped_task_2 does not know that it is downstream from list_task(). Prefect will have convenience functions for detecting these associations, and Prefect's .map() operator will automatically track them.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-i-enforce-ordering-between-tasks-that-dont-share-data","title":"Can I enforce ordering between tasks that don't share data?","text":"

    Yes! For more information, see the Tasks section.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#does-prefect-support-proxies","title":"Does Prefect support proxies?","text":"

    Yes!

    Prefect supports communicating via proxies through the use of environment variables. You can read more about this in the Installation documentation and the article Using Prefect Cloud with proxies.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-i-run-prefect-flows-on-linux","title":"Can I run Prefect flows on Linux?","text":"

    Yes!

    See the Installation documentation and Linux installation notes for details on getting started with Prefect on Linux.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-i-run-prefect-flows-on-windows","title":"Can I run Prefect flows on Windows?","text":"

    Yes!

    See the Installation documentation and Windows installation notes for details on getting started with Prefect on Windows.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#what-external-requirements-does-prefect-have","title":"What external requirements does Prefect have?","text":"

    Prefect does not have any additional requirements besides those installed by pip install --pre prefect. The entire system, including the UI and services, can be run in a single process via prefect server start and does not require Docker.

    Prefect Cloud users do not need to worry about the Prefect database. Prefect Cloud uses PostgreSQL on GCP behind the scenes. To use PostgreSQL with a self-hosted Prefect server, users must provide the connection string for a running database via the PREFECT_API_DATABASE_CONNECTION_URL environment variable.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#what-databases-does-prefect-support","title":"What databases does Prefect support?","text":"

    A self-hosted Prefect server can work with SQLite and PostgreSQL. New Prefect installs default to a SQLite database hosted at ~/.prefect/prefect.db on Mac or Linux machines. SQLite and PostgreSQL are not installed by Prefect.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#how-do-i-choose-between-sqlite-and-postgres","title":"How do I choose between SQLite and Postgres?","text":"

    SQLite generally works well for getting started and exploring Prefect. We have tested it with up to hundreds of thousands of task runs. Many users may be able to stay on SQLite for some time. However, for production uses, Prefect Cloud or self-hosted PostgreSQL is highly recommended. Under write-heavy workloads, SQLite performance can begin to suffer. Users running many flows with high degrees of parallelism or concurrency should use PostgreSQL.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#relationship-with-other-prefect-products","title":"Relationship with other Prefect products","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-a-flow-written-with-prefect-1-be-orchestrated-with-prefect-2-and-vice-versa","title":"Can a flow written with Prefect 1 be orchestrated with Prefect 2 and vice versa?","text":"

    No. Flows written with the Prefect 1 client must be rewritten with the Prefect 2 client. For most flows, this should take just a few minutes. See our migration guide and our Upgrade to Prefect 2 post for more information.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-a-use-prefect-1-and-prefect-2-at-the-same-time-on-my-local-machine","title":"Can a use Prefect 1 and Prefect 2 at the same time on my local machine?","text":"

    Yes. Just use different virtual environments.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"api-ref/","title":"API Reference","text":"

    Prefect auto-generates reference documentation for the following components:

    • Prefect Python SDK: used to build, test, and execute workflows.
    • Prefect REST API: used by both workflow clients as well as the Prefect UI for orchestration and data retrieval
    • Prefect Cloud REST API documentation is available at https://app.prefect.cloud/api/docs.
    • The REST API documentation for a locally hosted open-source Prefect server is available in the Prefect REST API Reference.
    • Prefect Server SDK: used primarily by the server to work with workflow metadata and enforce orchestration logic. This is only used directly by Prefect developers and contributors.

    Self-hosted docs

    When self-hosting, you can access REST API documentation at the /docs endpoint of your PREFECT_API_URL - for example, if you ran prefect server start with no additional configuration you can find this reference at http://localhost:4200/docs.

    ","tags":["API","Prefect API","Prefect SDK","Prefect Cloud","REST API","development","orchestration"]},{"location":"api-ref/rest-api-reference/","title":"Prefect server REST API reference","text":"","tags":["REST API","Prefect server"]},{"location":"api-ref/prefect/agent/","title":"prefect.agent","text":"","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent","title":"prefect.agent","text":"

    The agent is responsible for checking for flow runs that are ready to run and starting their execution.

    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent","title":"PrefectAgent","text":"Source code in prefect/agent.py
    class PrefectAgent:\n    @experimental_parameter(\n        \"work_pool_name\", group=\"work_pools\", when=lambda y: y is not None\n    )\n    def __init__(\n        self,\n        work_queues: List[str] = None,\n        work_queue_prefix: Union[str, List[str]] = None,\n        work_pool_name: str = None,\n        prefetch_seconds: int = None,\n        default_infrastructure: Infrastructure = None,\n        default_infrastructure_document_id: UUID = None,\n        limit: Optional[int] = None,\n    ) -> None:\n        if default_infrastructure and default_infrastructure_document_id:\n            raise ValueError(\n                \"Provide only one of 'default_infrastructure' and\"\n                \" 'default_infrastructure_document_id'.\"\n            )\n\n        self.work_queues: Set[str] = set(work_queues) if work_queues else set()\n        self.work_pool_name = work_pool_name\n        self.prefetch_seconds = prefetch_seconds\n        self.submitting_flow_run_ids = set()\n        self.cancelling_flow_run_ids = set()\n        self.scheduled_task_scopes = set()\n        self.started = False\n        self.logger = get_logger(\"agent\")\n        self.task_group: Optional[anyio.abc.TaskGroup] = None\n        self.limit: Optional[int] = limit\n        self.limiter: Optional[anyio.CapacityLimiter] = None\n        self.client: Optional[PrefectClient] = None\n\n        if isinstance(work_queue_prefix, str):\n            work_queue_prefix = [work_queue_prefix]\n        self.work_queue_prefix = work_queue_prefix\n\n        self._work_queue_cache_expiration: pendulum.DateTime = None\n        self._work_queue_cache: List[WorkQueue] = []\n\n        if default_infrastructure:\n            self.default_infrastructure_document_id = (\n                default_infrastructure._block_document_id\n            )\n            self.default_infrastructure = default_infrastructure\n        elif default_infrastructure_document_id:\n            self.default_infrastructure_document_id = default_infrastructure_document_id\n            self.default_infrastructure = None\n        else:\n            self.default_infrastructure = Process()\n            self.default_infrastructure_document_id = None\n\n    async def update_matched_agent_work_queues(self):\n        if self.work_queue_prefix:\n            if self.work_pool_name:\n                matched_queues = await self.client.read_work_queues(\n                    work_pool_name=self.work_pool_name,\n                    work_queue_filter=WorkQueueFilter(\n                        name=WorkQueueFilterName(startswith_=self.work_queue_prefix)\n                    ),\n                )\n            else:\n                matched_queues = await self.client.match_work_queues(\n                    self.work_queue_prefix, work_pool_name=DEFAULT_AGENT_WORK_POOL_NAME\n                )\n\n            matched_queues = set(q.name for q in matched_queues)\n            if matched_queues != self.work_queues:\n                new_queues = matched_queues - self.work_queues\n                removed_queues = self.work_queues - matched_queues\n                if new_queues:\n                    self.logger.info(\n                        f\"Matched new work queues: {', '.join(new_queues)}\"\n                    )\n                if removed_queues:\n                    self.logger.info(\n                        f\"Work queues no longer matched: {', '.join(removed_queues)}\"\n                    )\n            self.work_queues = matched_queues\n\n    async def get_work_queues(self) -> AsyncIterator[WorkQueue]:\n        \"\"\"\n        Loads the work queue objects corresponding to the agent's target work\n        queues. If any of them don't exist, they are created.\n        \"\"\"\n\n        # if the queue cache has not expired, yield queues from the cache\n        now = pendulum.now(\"UTC\")\n        if (self._work_queue_cache_expiration or now) > now:\n            for queue in self._work_queue_cache:\n                yield queue\n            return\n\n        # otherwise clear the cache, set the expiration for 30 seconds, and\n        # reload the work queues\n        self._work_queue_cache.clear()\n        self._work_queue_cache_expiration = now.add(seconds=30)\n\n        await self.update_matched_agent_work_queues()\n\n        for name in self.work_queues:\n            try:\n                work_queue = await self.client.read_work_queue_by_name(\n                    work_pool_name=self.work_pool_name, name=name\n                )\n            except (ObjectNotFound, Exception):\n                work_queue = None\n\n            # if the work queue wasn't found and the agent is NOT polling\n            # for queues using a regex, try to create it\n            if work_queue is None and not self.work_queue_prefix:\n                try:\n                    work_queue = await self.client.create_work_queue(\n                        work_pool_name=self.work_pool_name, name=name\n                    )\n                except Exception:\n                    # if creating it raises an exception, it was probably just\n                    # created by some other agent; rather than entering a re-read\n                    # loop with new error handling, we log the exception and\n                    # continue.\n                    self.logger.exception(f\"Failed to create work queue {name!r}.\")\n                    continue\n                else:\n                    log_str = f\"Created work queue {name!r}\"\n                    if self.work_pool_name:\n                        log_str = (\n                            f\"Created work queue {name!r} in work pool\"\n                            f\" {self.work_pool_name!r}.\"\n                        )\n                    else:\n                        log_str = f\"Created work queue '{name}'.\"\n                    self.logger.info(log_str)\n\n            if work_queue is None:\n                self.logger.error(\n                    f\"Work queue '{name!r}' with prefix {self.work_queue_prefix} wasn't\"\n                    \" found\"\n                )\n            else:\n                self._work_queue_cache.append(work_queue)\n                yield work_queue\n\n    async def get_and_submit_flow_runs(self) -> List[FlowRun]:\n        \"\"\"\n        The principle method on agents. Queries for scheduled flow runs and submits\n        them for execution in parallel.\n        \"\"\"\n        if not self.started:\n            raise RuntimeError(\n                \"Agent is not started. Use `async with PrefectAgent()...`\"\n            )\n\n        self.logger.debug(\"Checking for scheduled flow runs...\")\n\n        before = pendulum.now(\"utc\").add(\n            seconds=self.prefetch_seconds or PREFECT_AGENT_PREFETCH_SECONDS.value()\n        )\n\n        submittable_runs: List[FlowRun] = []\n\n        if self.work_pool_name:\n            responses = await self.client.get_scheduled_flow_runs_for_work_pool(\n                work_pool_name=self.work_pool_name,\n                work_queue_names=[wq.name async for wq in self.get_work_queues()],\n                scheduled_before=before,\n            )\n            submittable_runs.extend([response.flow_run for response in responses])\n\n        else:\n            # load runs from each work queue\n            async for work_queue in self.get_work_queues():\n                # print a nice message if the work queue is paused\n                if work_queue.is_paused:\n                    self.logger.info(\n                        f\"Work queue {work_queue.name!r} ({work_queue.id}) is paused.\"\n                    )\n\n                else:\n                    try:\n                        queue_runs = await self.client.get_runs_in_work_queue(\n                            id=work_queue.id, limit=10, scheduled_before=before\n                        )\n                        submittable_runs.extend(queue_runs)\n                    except ObjectNotFound:\n                        self.logger.error(\n                            f\"Work queue {work_queue.name!r} ({work_queue.id}) not\"\n                            \" found.\"\n                        )\n                    except Exception as exc:\n                        self.logger.exception(exc)\n\n            submittable_runs.sort(key=lambda run: run.next_scheduled_start_time)\n\n        for flow_run in submittable_runs:\n            # don't resubmit a run\n            if flow_run.id in self.submitting_flow_run_ids:\n                continue\n\n            try:\n                if self.limiter:\n                    self.limiter.acquire_on_behalf_of_nowait(flow_run.id)\n            except anyio.WouldBlock:\n                self.logger.info(\n                    f\"Flow run limit reached; {self.limiter.borrowed_tokens} flow runs\"\n                    \" in progress.\"\n                )\n                break\n            else:\n                self.logger.info(f\"Submitting flow run '{flow_run.id}'\")\n                self.submitting_flow_run_ids.add(flow_run.id)\n                self.task_group.start_soon(\n                    self.submit_run,\n                    flow_run,\n                )\n\n        return list(\n            filter(lambda run: run.id in self.submitting_flow_run_ids, submittable_runs)\n        )\n\n    async def check_for_cancelled_flow_runs(self):\n        if not self.started:\n            raise RuntimeError(\n                \"Agent is not started. Use `async with PrefectAgent()...`\"\n            )\n\n        self.logger.debug(\"Checking for cancelled flow runs...\")\n\n        work_queue_filter = (\n            WorkQueueFilter(name=WorkQueueFilterName(any_=list(self.work_queues)))\n            if self.work_queues\n            else None\n        )\n\n        work_pool_filter = (\n            WorkPoolFilter(name=WorkPoolFilterName(any_=[self.work_pool_name]))\n            if self.work_pool_name\n            else WorkPoolFilter(name=WorkPoolFilterName(any_=[\"default-agent-pool\"]))\n        )\n        named_cancelling_flow_runs = await self.client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLED]),\n                    name=FlowRunFilterStateName(any_=[\"Cancelling\"]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(not_any_=list(self.cancelling_flow_run_ids)),\n            ),\n            work_pool_filter=work_pool_filter,\n            work_queue_filter=work_queue_filter,\n        )\n\n        typed_cancelling_flow_runs = await self.client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLING]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(not_any_=list(self.cancelling_flow_run_ids)),\n            ),\n            work_pool_filter=work_pool_filter,\n            work_queue_filter=work_queue_filter,\n        )\n\n        cancelling_flow_runs = named_cancelling_flow_runs + typed_cancelling_flow_runs\n\n        if cancelling_flow_runs:\n            self.logger.info(\n                f\"Found {len(cancelling_flow_runs)} flow runs awaiting cancellation.\"\n            )\n\n        for flow_run in cancelling_flow_runs:\n            self.cancelling_flow_run_ids.add(flow_run.id)\n            self.task_group.start_soon(self.cancel_run, flow_run)\n\n        return cancelling_flow_runs\n\n    async def cancel_run(self, flow_run: FlowRun) -> None:\n        \"\"\"\n        Cancel a flow run by killing its infrastructure\n        \"\"\"\n        if not flow_run.infrastructure_pid:\n            self.logger.error(\n                f\"Flow run '{flow_run.id}' does not have an infrastructure pid\"\n                \" attached. Cancellation cannot be guaranteed.\"\n            )\n            await self._mark_flow_run_as_cancelled(\n                flow_run,\n                state_updates={\n                    \"message\": (\n                        \"This flow run is missing infrastructure tracking information\"\n                        \" and cancellation cannot be guaranteed.\"\n                    )\n                },\n            )\n            return\n\n        try:\n            infrastructure = await self.get_infrastructure(flow_run)\n            if infrastructure.is_using_a_runner:\n                self.logger.info(\n                    f\"Skipping cancellation because flow run {str(flow_run.id)!r} is\"\n                    \" using enhanced cancellation. A dedicated runner will handle\"\n                    \" cancellation.\"\n                )\n                return\n        except Exception:\n            self.logger.exception(\n                f\"Failed to get infrastructure for flow run '{flow_run.id}'. \"\n                \"Flow run cannot be cancelled.\"\n            )\n            # Note: We leave this flow run in the cancelling set because it cannot be\n            #       cancelled and this will prevent additional attempts.\n            return\n\n        if not hasattr(infrastructure, \"kill\"):\n            self.logger.error(\n                f\"Flow run '{flow_run.id}' infrastructure {infrastructure.type!r} \"\n                \"does not support killing created infrastructure. \"\n                \"Cancellation cannot be guaranteed.\"\n            )\n            return\n\n        self.logger.info(\n            f\"Killing {infrastructure.type} {flow_run.infrastructure_pid} for flow run \"\n            f\"'{flow_run.id}'...\"\n        )\n        try:\n            await infrastructure.kill(flow_run.infrastructure_pid)\n        except InfrastructureNotFound as exc:\n            self.logger.warning(f\"{exc} Marking flow run as cancelled.\")\n            await self._mark_flow_run_as_cancelled(flow_run)\n        except InfrastructureNotAvailable as exc:\n            self.logger.warning(f\"{exc} Flow run cannot be cancelled by this agent.\")\n        except Exception:\n            self.logger.exception(\n                \"Encountered exception while killing infrastructure for flow run \"\n                f\"'{flow_run.id}'. Flow run may not be cancelled.\"\n            )\n            # We will try again on generic exceptions\n            self.cancelling_flow_run_ids.remove(flow_run.id)\n            return\n        else:\n            await self._mark_flow_run_as_cancelled(flow_run)\n            self.logger.info(f\"Cancelled flow run '{flow_run.id}'!\")\n\n    async def _mark_flow_run_as_cancelled(\n        self, flow_run: FlowRun, state_updates: Optional[dict] = None\n    ) -> None:\n        state_updates = state_updates or {}\n        state_updates.setdefault(\"name\", \"Cancelled\")\n        state_updates.setdefault(\"type\", StateType.CANCELLED)\n        state = flow_run.state.copy(update=state_updates)\n\n        await self.client.set_flow_run_state(flow_run.id, state, force=True)\n\n        # Do not remove the flow run from the cancelling set immediately because\n        # the API caches responses for the `read_flow_runs` and we do not want to\n        # duplicate cancellations.\n        await self._schedule_task(\n            60 * 10, self.cancelling_flow_run_ids.remove, flow_run.id\n        )\n\n    async def get_infrastructure(self, flow_run: FlowRun) -> Infrastructure:\n        deployment = await self.client.read_deployment(flow_run.deployment_id)\n\n        flow = await self.client.read_flow(deployment.flow_id)\n\n        # overrides only apply when configuring known infra blocks\n        if not deployment.infrastructure_document_id:\n            if self.default_infrastructure:\n                infra_block = self.default_infrastructure\n            else:\n                infra_document = await self.client.read_block_document(\n                    self.default_infrastructure_document_id\n                )\n                infra_block = Block._from_block_document(infra_document)\n\n            # Add flow run metadata to the infrastructure\n            prepared_infrastructure = infra_block.prepare_for_flow_run(\n                flow_run, deployment=deployment, flow=flow\n            )\n            return prepared_infrastructure\n\n        ## get infra\n        infra_document = await self.client.read_block_document(\n            deployment.infrastructure_document_id\n        )\n\n        # this piece of logic applies any overrides that may have been set on the\n        # deployment; overrides are defined as dot.delimited paths on possibly nested\n        # attributes of the infrastructure block\n        doc_dict = infra_document.dict()\n        infra_dict = doc_dict.get(\"data\", {})\n        for override, value in (deployment.infra_overrides or {}).items():\n            nested_fields = override.split(\".\")\n            data = infra_dict\n            for field in nested_fields[:-1]:\n                data = data[field]\n\n            # once we reach the end, set the value\n            data[nested_fields[-1]] = value\n\n        # reconstruct the infra block\n        doc_dict[\"data\"] = infra_dict\n        infra_document = BlockDocument(**doc_dict)\n        infrastructure_block = Block._from_block_document(infra_document)\n\n        # TODO: Here the agent may update the infrastructure with agent-level settings\n\n        # Add flow run metadata to the infrastructure\n        prepared_infrastructure = infrastructure_block.prepare_for_flow_run(\n            flow_run, deployment=deployment, flow=flow\n        )\n\n        return prepared_infrastructure\n\n    async def submit_run(self, flow_run: FlowRun) -> None:\n        \"\"\"\n        Submit a flow run to the infrastructure\n        \"\"\"\n        ready_to_submit = await self._propose_pending_state(flow_run)\n\n        if ready_to_submit:\n            try:\n                infrastructure = await self.get_infrastructure(flow_run)\n            except Exception as exc:\n                self.logger.exception(\n                    f\"Failed to get infrastructure for flow run '{flow_run.id}'.\"\n                )\n                await self._propose_failed_state(flow_run, exc)\n                if self.limiter:\n                    self.limiter.release_on_behalf_of(flow_run.id)\n            else:\n                # Wait for submission to be completed. Note that the submission function\n                # may continue to run in the background after this exits.\n                readiness_result = await self.task_group.start(\n                    self._submit_run_and_capture_errors, flow_run, infrastructure\n                )\n\n                if readiness_result and not isinstance(readiness_result, Exception):\n                    try:\n                        await self.client.update_flow_run(\n                            flow_run_id=flow_run.id,\n                            infrastructure_pid=str(readiness_result),\n                        )\n                    except Exception:\n                        self.logger.exception(\n                            \"An error occurred while setting the `infrastructure_pid`\"\n                            f\" on flow run {flow_run.id!r}. The flow run will not be\"\n                            \" cancellable.\"\n                        )\n\n                self.logger.info(f\"Completed submission of flow run '{flow_run.id}'\")\n\n        else:\n            # If the run is not ready to submit, release the concurrency slot\n            if self.limiter:\n                self.limiter.release_on_behalf_of(flow_run.id)\n\n        self.submitting_flow_run_ids.remove(flow_run.id)\n\n    async def _submit_run_and_capture_errors(\n        self,\n        flow_run: FlowRun,\n        infrastructure: Infrastructure,\n        task_status: anyio.abc.TaskStatus = None,\n    ) -> Union[InfrastructureResult, Exception]:\n        # Note: There is not a clear way to determine if task_status.started() has been\n        #       called without peeking at the internal `_future`. Ideally we could just\n        #       check if the flow run id has been removed from `submitting_flow_run_ids`\n        #       but it is not so simple to guarantee that this coroutine yields back\n        #       to `submit_run` to execute that line when exceptions are raised during\n        #       submission.\n        try:\n            result = await infrastructure.run(task_status=task_status)\n        except Exception as exc:\n            if not task_status._future.done():\n                # This flow run was being submitted and did not start successfully\n                self.logger.exception(\n                    f\"Failed to submit flow run '{flow_run.id}' to infrastructure.\"\n                )\n                # Mark the task as started to prevent agent crash\n                task_status.started(exc)\n                await self._propose_crashed_state(\n                    flow_run, \"Flow run could not be submitted to infrastructure\"\n                )\n            else:\n                self.logger.exception(\n                    f\"An error occurred while monitoring flow run '{flow_run.id}'. \"\n                    \"The flow run will not be marked as failed, but an issue may have \"\n                    \"occurred.\"\n                )\n            return exc\n        finally:\n            if self.limiter:\n                self.limiter.release_on_behalf_of(flow_run.id)\n\n        if not task_status._future.done():\n            self.logger.error(\n                f\"Infrastructure returned without reporting flow run '{flow_run.id}' \"\n                \"as started or raising an error. This behavior is not expected and \"\n                \"generally indicates improper implementation of infrastructure. The \"\n                \"flow run will not be marked as failed, but an issue may have occurred.\"\n            )\n            # Mark the task as started to prevent agent crash\n            task_status.started()\n\n        if result.status_code != 0:\n            await self._propose_crashed_state(\n                flow_run,\n                (\n                    \"Flow run infrastructure exited with non-zero status code\"\n                    f\" {result.status_code}.\"\n                ),\n            )\n\n        return result\n\n    async def _propose_pending_state(self, flow_run: FlowRun) -> bool:\n        state = flow_run.state\n        try:\n            state = await propose_state(self.client, Pending(), flow_run_id=flow_run.id)\n        except Abort as exc:\n            self.logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}'. \"\n                    f\"Server sent an abort signal: {exc}\"\n                ),\n            )\n            return False\n        except Exception:\n            self.logger.error(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n                exc_info=True,\n            )\n            return False\n\n        if not state.is_pending():\n            self.logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}': \"\n                    f\"Server returned a non-pending state {state.type.value!r}\"\n                ),\n            )\n            return False\n\n        return True\n\n    async def _propose_failed_state(self, flow_run: FlowRun, exc: Exception) -> None:\n        try:\n            await propose_state(\n                self.client,\n                await exception_to_failed_state(message=\"Submission failed.\", exc=exc),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # We've already failed, no need to note the abort but we don't want it to\n            # raise in the agent process\n            pass\n        except Exception:\n            self.logger.error(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n                exc_info=True,\n            )\n\n    async def _propose_crashed_state(self, flow_run: FlowRun, message: str) -> None:\n        try:\n            state = await propose_state(\n                self.client,\n                Crashed(message=message),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # Flow run already marked as failed\n            pass\n        except Exception:\n            self.logger.exception(f\"Failed to update state of flow run '{flow_run.id}'\")\n        else:\n            if state.is_crashed():\n                self.logger.info(\n                    f\"Reported flow run '{flow_run.id}' as crashed: {message}\"\n                )\n\n    async def _schedule_task(self, __in_seconds: int, fn, *args, **kwargs):\n        \"\"\"\n        Schedule a background task to start after some time.\n\n        These tasks will be run immediately when the agent exits instead of waiting.\n\n        The function may be async or sync. Async functions will be awaited.\n        \"\"\"\n\n        async def wrapper(task_status):\n            # If we are shutting down, do not sleep; otherwise sleep until the scheduled\n            # time or shutdown\n            if self.started:\n                with anyio.CancelScope() as scope:\n                    self.scheduled_task_scopes.add(scope)\n                    task_status.started()\n                    await anyio.sleep(__in_seconds)\n\n                self.scheduled_task_scopes.remove(scope)\n            else:\n                task_status.started()\n\n            result = fn(*args, **kwargs)\n            if inspect.iscoroutine(result):\n                await result\n\n        await self.task_group.start(wrapper)\n\n    # Context management ---------------------------------------------------------------\n\n    async def start(self):\n        self.started = True\n        self.task_group = anyio.create_task_group()\n        self.limiter = (\n            anyio.CapacityLimiter(self.limit) if self.limit is not None else None\n        )\n        self.client = get_client()\n        await self.client.__aenter__()\n        await self.task_group.__aenter__()\n\n    async def shutdown(self, *exc_info):\n        self.started = False\n        # We must cancel scheduled task scopes before closing the task group\n        for scope in self.scheduled_task_scopes:\n            scope.cancel()\n        await self.task_group.__aexit__(*exc_info)\n        await self.client.__aexit__(*exc_info)\n        self.task_group = None\n        self.client = None\n        self.submitting_flow_run_ids.clear()\n        self.cancelling_flow_run_ids.clear()\n        self.scheduled_task_scopes.clear()\n        self._work_queue_cache_expiration = None\n        self._work_queue_cache = []\n\n    async def __aenter__(self):\n        await self.start()\n        return self\n\n    async def __aexit__(self, *exc_info):\n        await self.shutdown(*exc_info)\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent.cancel_run","title":"cancel_run async","text":"

    Cancel a flow run by killing its infrastructure

    Source code in prefect/agent.py
    async def cancel_run(self, flow_run: FlowRun) -> None:\n    \"\"\"\n    Cancel a flow run by killing its infrastructure\n    \"\"\"\n    if not flow_run.infrastructure_pid:\n        self.logger.error(\n            f\"Flow run '{flow_run.id}' does not have an infrastructure pid\"\n            \" attached. Cancellation cannot be guaranteed.\"\n        )\n        await self._mark_flow_run_as_cancelled(\n            flow_run,\n            state_updates={\n                \"message\": (\n                    \"This flow run is missing infrastructure tracking information\"\n                    \" and cancellation cannot be guaranteed.\"\n                )\n            },\n        )\n        return\n\n    try:\n        infrastructure = await self.get_infrastructure(flow_run)\n        if infrastructure.is_using_a_runner:\n            self.logger.info(\n                f\"Skipping cancellation because flow run {str(flow_run.id)!r} is\"\n                \" using enhanced cancellation. A dedicated runner will handle\"\n                \" cancellation.\"\n            )\n            return\n    except Exception:\n        self.logger.exception(\n            f\"Failed to get infrastructure for flow run '{flow_run.id}'. \"\n            \"Flow run cannot be cancelled.\"\n        )\n        # Note: We leave this flow run in the cancelling set because it cannot be\n        #       cancelled and this will prevent additional attempts.\n        return\n\n    if not hasattr(infrastructure, \"kill\"):\n        self.logger.error(\n            f\"Flow run '{flow_run.id}' infrastructure {infrastructure.type!r} \"\n            \"does not support killing created infrastructure. \"\n            \"Cancellation cannot be guaranteed.\"\n        )\n        return\n\n    self.logger.info(\n        f\"Killing {infrastructure.type} {flow_run.infrastructure_pid} for flow run \"\n        f\"'{flow_run.id}'...\"\n    )\n    try:\n        await infrastructure.kill(flow_run.infrastructure_pid)\n    except InfrastructureNotFound as exc:\n        self.logger.warning(f\"{exc} Marking flow run as cancelled.\")\n        await self._mark_flow_run_as_cancelled(flow_run)\n    except InfrastructureNotAvailable as exc:\n        self.logger.warning(f\"{exc} Flow run cannot be cancelled by this agent.\")\n    except Exception:\n        self.logger.exception(\n            \"Encountered exception while killing infrastructure for flow run \"\n            f\"'{flow_run.id}'. Flow run may not be cancelled.\"\n        )\n        # We will try again on generic exceptions\n        self.cancelling_flow_run_ids.remove(flow_run.id)\n        return\n    else:\n        await self._mark_flow_run_as_cancelled(flow_run)\n        self.logger.info(f\"Cancelled flow run '{flow_run.id}'!\")\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent.get_and_submit_flow_runs","title":"get_and_submit_flow_runs async","text":"

    The principle method on agents. Queries for scheduled flow runs and submits them for execution in parallel.

    Source code in prefect/agent.py
    async def get_and_submit_flow_runs(self) -> List[FlowRun]:\n    \"\"\"\n    The principle method on agents. Queries for scheduled flow runs and submits\n    them for execution in parallel.\n    \"\"\"\n    if not self.started:\n        raise RuntimeError(\n            \"Agent is not started. Use `async with PrefectAgent()...`\"\n        )\n\n    self.logger.debug(\"Checking for scheduled flow runs...\")\n\n    before = pendulum.now(\"utc\").add(\n        seconds=self.prefetch_seconds or PREFECT_AGENT_PREFETCH_SECONDS.value()\n    )\n\n    submittable_runs: List[FlowRun] = []\n\n    if self.work_pool_name:\n        responses = await self.client.get_scheduled_flow_runs_for_work_pool(\n            work_pool_name=self.work_pool_name,\n            work_queue_names=[wq.name async for wq in self.get_work_queues()],\n            scheduled_before=before,\n        )\n        submittable_runs.extend([response.flow_run for response in responses])\n\n    else:\n        # load runs from each work queue\n        async for work_queue in self.get_work_queues():\n            # print a nice message if the work queue is paused\n            if work_queue.is_paused:\n                self.logger.info(\n                    f\"Work queue {work_queue.name!r} ({work_queue.id}) is paused.\"\n                )\n\n            else:\n                try:\n                    queue_runs = await self.client.get_runs_in_work_queue(\n                        id=work_queue.id, limit=10, scheduled_before=before\n                    )\n                    submittable_runs.extend(queue_runs)\n                except ObjectNotFound:\n                    self.logger.error(\n                        f\"Work queue {work_queue.name!r} ({work_queue.id}) not\"\n                        \" found.\"\n                    )\n                except Exception as exc:\n                    self.logger.exception(exc)\n\n        submittable_runs.sort(key=lambda run: run.next_scheduled_start_time)\n\n    for flow_run in submittable_runs:\n        # don't resubmit a run\n        if flow_run.id in self.submitting_flow_run_ids:\n            continue\n\n        try:\n            if self.limiter:\n                self.limiter.acquire_on_behalf_of_nowait(flow_run.id)\n        except anyio.WouldBlock:\n            self.logger.info(\n                f\"Flow run limit reached; {self.limiter.borrowed_tokens} flow runs\"\n                \" in progress.\"\n            )\n            break\n        else:\n            self.logger.info(f\"Submitting flow run '{flow_run.id}'\")\n            self.submitting_flow_run_ids.add(flow_run.id)\n            self.task_group.start_soon(\n                self.submit_run,\n                flow_run,\n            )\n\n    return list(\n        filter(lambda run: run.id in self.submitting_flow_run_ids, submittable_runs)\n    )\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent.get_work_queues","title":"get_work_queues async","text":"

    Loads the work queue objects corresponding to the agent's target work queues. If any of them don't exist, they are created.

    Source code in prefect/agent.py
    async def get_work_queues(self) -> AsyncIterator[WorkQueue]:\n    \"\"\"\n    Loads the work queue objects corresponding to the agent's target work\n    queues. If any of them don't exist, they are created.\n    \"\"\"\n\n    # if the queue cache has not expired, yield queues from the cache\n    now = pendulum.now(\"UTC\")\n    if (self._work_queue_cache_expiration or now) > now:\n        for queue in self._work_queue_cache:\n            yield queue\n        return\n\n    # otherwise clear the cache, set the expiration for 30 seconds, and\n    # reload the work queues\n    self._work_queue_cache.clear()\n    self._work_queue_cache_expiration = now.add(seconds=30)\n\n    await self.update_matched_agent_work_queues()\n\n    for name in self.work_queues:\n        try:\n            work_queue = await self.client.read_work_queue_by_name(\n                work_pool_name=self.work_pool_name, name=name\n            )\n        except (ObjectNotFound, Exception):\n            work_queue = None\n\n        # if the work queue wasn't found and the agent is NOT polling\n        # for queues using a regex, try to create it\n        if work_queue is None and not self.work_queue_prefix:\n            try:\n                work_queue = await self.client.create_work_queue(\n                    work_pool_name=self.work_pool_name, name=name\n                )\n            except Exception:\n                # if creating it raises an exception, it was probably just\n                # created by some other agent; rather than entering a re-read\n                # loop with new error handling, we log the exception and\n                # continue.\n                self.logger.exception(f\"Failed to create work queue {name!r}.\")\n                continue\n            else:\n                log_str = f\"Created work queue {name!r}\"\n                if self.work_pool_name:\n                    log_str = (\n                        f\"Created work queue {name!r} in work pool\"\n                        f\" {self.work_pool_name!r}.\"\n                    )\n                else:\n                    log_str = f\"Created work queue '{name}'.\"\n                self.logger.info(log_str)\n\n        if work_queue is None:\n            self.logger.error(\n                f\"Work queue '{name!r}' with prefix {self.work_queue_prefix} wasn't\"\n                \" found\"\n            )\n        else:\n            self._work_queue_cache.append(work_queue)\n            yield work_queue\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent.submit_run","title":"submit_run async","text":"

    Submit a flow run to the infrastructure

    Source code in prefect/agent.py
    async def submit_run(self, flow_run: FlowRun) -> None:\n    \"\"\"\n    Submit a flow run to the infrastructure\n    \"\"\"\n    ready_to_submit = await self._propose_pending_state(flow_run)\n\n    if ready_to_submit:\n        try:\n            infrastructure = await self.get_infrastructure(flow_run)\n        except Exception as exc:\n            self.logger.exception(\n                f\"Failed to get infrastructure for flow run '{flow_run.id}'.\"\n            )\n            await self._propose_failed_state(flow_run, exc)\n            if self.limiter:\n                self.limiter.release_on_behalf_of(flow_run.id)\n        else:\n            # Wait for submission to be completed. Note that the submission function\n            # may continue to run in the background after this exits.\n            readiness_result = await self.task_group.start(\n                self._submit_run_and_capture_errors, flow_run, infrastructure\n            )\n\n            if readiness_result and not isinstance(readiness_result, Exception):\n                try:\n                    await self.client.update_flow_run(\n                        flow_run_id=flow_run.id,\n                        infrastructure_pid=str(readiness_result),\n                    )\n                except Exception:\n                    self.logger.exception(\n                        \"An error occurred while setting the `infrastructure_pid`\"\n                        f\" on flow run {flow_run.id!r}. The flow run will not be\"\n                        \" cancellable.\"\n                    )\n\n            self.logger.info(f\"Completed submission of flow run '{flow_run.id}'\")\n\n    else:\n        # If the run is not ready to submit, release the concurrency slot\n        if self.limiter:\n            self.limiter.release_on_behalf_of(flow_run.id)\n\n    self.submitting_flow_run_ids.remove(flow_run.id)\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/artifacts/","title":"prefect.artifacts","text":"","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/artifacts/#prefect.artifacts","title":"prefect.artifacts","text":"

    Interface for creating and reading artifacts.

    ","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/artifacts/#prefect.artifacts.create_link_artifact","title":"create_link_artifact async","text":"

    Create a link artifact.

    Parameters:

    Name Type Description Default link str

    The link to create.

    required link_text Optional[str]

    The link text.

    None key Optional[str]

    A user-provided string identifier. Required for the artifact to show in the Artifacts page in the UI. The key must only contain lowercase letters, numbers, and dashes.

    None description Optional[str]

    A user-specified description of the artifact.

    None

    Returns:

    Type Description UUID

    The table artifact ID.

    Source code in prefect/artifacts.py
    @sync_compatible\nasync def create_link_artifact(\n    link: str,\n    link_text: Optional[str] = None,\n    key: Optional[str] = None,\n    description: Optional[str] = None,\n) -> UUID:\n    \"\"\"\n    Create a link artifact.\n\n    Arguments:\n        link: The link to create.\n        link_text: The link text.\n        key: A user-provided string identifier.\n          Required for the artifact to show in the Artifacts page in the UI.\n          The key must only contain lowercase letters, numbers, and dashes.\n        description: A user-specified description of the artifact.\n\n\n    Returns:\n        The table artifact ID.\n    \"\"\"\n    formatted_link = f\"[{link_text}]({link})\" if link_text else f\"[{link}]({link})\"\n    artifact = await _create_artifact(\n        key=key,\n        type=\"markdown\",\n        description=description,\n        data=formatted_link,\n    )\n\n    return artifact.id\n
    ","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/artifacts/#prefect.artifacts.create_markdown_artifact","title":"create_markdown_artifact async","text":"

    Create a markdown artifact.

    Parameters:

    Name Type Description Default markdown str

    The markdown to create.

    required key Optional[str]

    A user-provided string identifier. Required for the artifact to show in the Artifacts page in the UI. The key must only contain lowercase letters, numbers, and dashes.

    None description Optional[str]

    A user-specified description of the artifact.

    None

    Returns:

    Type Description UUID

    The table artifact ID.

    Source code in prefect/artifacts.py
    @sync_compatible\nasync def create_markdown_artifact(\n    markdown: str,\n    key: Optional[str] = None,\n    description: Optional[str] = None,\n) -> UUID:\n    \"\"\"\n    Create a markdown artifact.\n\n    Arguments:\n        markdown: The markdown to create.\n        key: A user-provided string identifier.\n          Required for the artifact to show in the Artifacts page in the UI.\n          The key must only contain lowercase letters, numbers, and dashes.\n        description: A user-specified description of the artifact.\n\n    Returns:\n        The table artifact ID.\n    \"\"\"\n    artifact = await _create_artifact(\n        key=key,\n        type=\"markdown\",\n        description=description,\n        data=markdown,\n    )\n\n    return artifact.id\n
    ","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/artifacts/#prefect.artifacts.create_table_artifact","title":"create_table_artifact async","text":"

    Create a table artifact.

    Parameters:

    Name Type Description Default table Union[Dict[str, List[Any]], List[Dict[str, Any]], List[List[Any]]]

    The table to create.

    required key Optional[str]

    A user-provided string identifier. Required for the artifact to show in the Artifacts page in the UI. The key must only contain lowercase letters, numbers, and dashes.

    None description Optional[str]

    A user-specified description of the artifact.

    None

    Returns:

    Type Description UUID

    The table artifact ID.

    Source code in prefect/artifacts.py
    @sync_compatible\nasync def create_table_artifact(\n    table: Union[Dict[str, List[Any]], List[Dict[str, Any]], List[List[Any]]],\n    key: Optional[str] = None,\n    description: Optional[str] = None,\n) -> UUID:\n    \"\"\"\n    Create a table artifact.\n\n    Arguments:\n        table: The table to create.\n        key: A user-provided string identifier.\n          Required for the artifact to show in the Artifacts page in the UI.\n          The key must only contain lowercase letters, numbers, and dashes.\n        description: A user-specified description of the artifact.\n\n    Returns:\n        The table artifact ID.\n    \"\"\"\n\n    def _sanitize_nan_values(item):\n        \"\"\"\n        Sanitize NaN values in a given item. The item can be a dict, list or float.\n        \"\"\"\n\n        if isinstance(item, list):\n            return [_sanitize_nan_values(sub_item) for sub_item in item]\n\n        elif isinstance(item, dict):\n            return {k: _sanitize_nan_values(v) for k, v in item.items()}\n\n        elif isinstance(item, float) and math.isnan(item):\n            return None\n\n        else:\n            return item\n\n    sanitized_table = _sanitize_nan_values(table)\n\n    if isinstance(table, dict) and all(isinstance(v, list) for v in table.values()):\n        pass\n    elif isinstance(table, list) and all(isinstance(v, (list, dict)) for v in table):\n        pass\n    else:\n        raise TypeError(INVALID_TABLE_TYPE_ERROR)\n\n    formatted_table = json.dumps(sanitized_table)\n\n    artifact = await _create_artifact(\n        key=key,\n        type=\"table\",\n        description=description,\n        data=formatted_table,\n    )\n\n    return artifact.id\n
    ","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/context/","title":"prefect.context","text":"","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context","title":"prefect.context","text":"

    Async and thread safe models for passing runtime context data.

    These contexts should never be directly mutated by the user.

    For more user-accessible information about the current run, see prefect.runtime.

    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.ContextModel","title":"ContextModel","text":"

    Bases: BaseModel

    A base model for context data that forbids mutation and extra data while providing a context manager

    Source code in prefect/context.py
    class ContextModel(BaseModel):\n    \"\"\"\n    A base model for context data that forbids mutation and extra data while providing\n    a context manager\n    \"\"\"\n\n    # The context variable for storing data must be defined by the child class\n    __var__: ContextVar\n    _token: Token = PrivateAttr(None)\n\n    class Config:\n        allow_mutation = False\n        arbitrary_types_allowed = True\n        extra = \"forbid\"\n\n    def __enter__(self):\n        if self._token is not None:\n            raise RuntimeError(\n                \"Context already entered. Context enter calls cannot be nested.\"\n            )\n        self._token = self.__var__.set(self)\n        return self\n\n    def __exit__(self, *_):\n        if not self._token:\n            raise RuntimeError(\n                \"Asymmetric use of context. Context exit called without an enter.\"\n            )\n        self.__var__.reset(self._token)\n        self._token = None\n\n    @classmethod\n    def get(cls: Type[T]) -> Optional[T]:\n        return cls.__var__.get(None)\n\n    def copy(self, **kwargs):\n        \"\"\"\n        Duplicate the context model, optionally choosing which fields to include, exclude, or change.\n\n        Attributes:\n            include: Fields to include in new model.\n            exclude: Fields to exclude from new model, as with values this takes precedence over include.\n            update: Values to change/add in the new model. Note: the data is not validated before creating\n                the new model - you should trust this data.\n            deep: Set to `True` to make a deep copy of the model.\n\n        Returns:\n            A new model instance.\n        \"\"\"\n        # Remove the token on copy to avoid re-entrance errors\n        new = super().copy(**kwargs)\n        new._token = None\n        return new\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.ContextModel.copy","title":"copy","text":"

    Duplicate the context model, optionally choosing which fields to include, exclude, or change.

    Attributes:

    Name Type Description include

    Fields to include in new model.

    exclude

    Fields to exclude from new model, as with values this takes precedence over include.

    update

    Values to change/add in the new model. Note: the data is not validated before creating the new model - you should trust this data.

    deep

    Set to True to make a deep copy of the model.

    Returns:

    Type Description

    A new model instance.

    Source code in prefect/context.py
    def copy(self, **kwargs):\n    \"\"\"\n    Duplicate the context model, optionally choosing which fields to include, exclude, or change.\n\n    Attributes:\n        include: Fields to include in new model.\n        exclude: Fields to exclude from new model, as with values this takes precedence over include.\n        update: Values to change/add in the new model. Note: the data is not validated before creating\n            the new model - you should trust this data.\n        deep: Set to `True` to make a deep copy of the model.\n\n    Returns:\n        A new model instance.\n    \"\"\"\n    # Remove the token on copy to avoid re-entrance errors\n    new = super().copy(**kwargs)\n    new._token = None\n    return new\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.FlowRunContext","title":"FlowRunContext","text":"

    Bases: RunContext

    The context for a flow run. Data in this context is only available from within a flow run function.

    Attributes:

    Name Type Description flow Optional[Flow]

    The flow instance associated with the run

    flow_run Optional[FlowRun]

    The API metadata for the flow run

    task_runner BaseTaskRunner

    The task runner instance being used for the flow run

    task_run_futures List[PrefectFuture]

    A list of futures for task runs submitted within this flow run

    task_run_states List[State]

    A list of states for task runs created within this flow run

    task_run_results Dict[int, State]

    A mapping of result ids to task run states for this flow run

    flow_run_states List[State]

    A list of states for flow runs created within this flow run

    sync_portal Optional[BlockingPortal]

    A blocking portal for sync task/flow runs in an async flow

    timeout_scope Optional[CancelScope]

    The cancellation scope for flow level timeouts

    Source code in prefect/context.py
    class FlowRunContext(RunContext):\n    \"\"\"\n    The context for a flow run. Data in this context is only available from within a\n    flow run function.\n\n    Attributes:\n        flow: The flow instance associated with the run\n        flow_run: The API metadata for the flow run\n        task_runner: The task runner instance being used for the flow run\n        task_run_futures: A list of futures for task runs submitted within this flow run\n        task_run_states: A list of states for task runs created within this flow run\n        task_run_results: A mapping of result ids to task run states for this flow run\n        flow_run_states: A list of states for flow runs created within this flow run\n        sync_portal: A blocking portal for sync task/flow runs in an async flow\n        timeout_scope: The cancellation scope for flow level timeouts\n    \"\"\"\n\n    flow: Optional[\"Flow\"] = None\n    flow_run: Optional[FlowRun] = None\n    task_runner: BaseTaskRunner\n    log_prints: bool = False\n    parameters: Dict[str, Any]\n\n    # Result handling\n    result_factory: ResultFactory\n\n    # Counter for task calls allowing unique\n    task_run_dynamic_keys: Dict[str, int] = Field(default_factory=dict)\n\n    # Counter for flow pauses\n    observed_flow_pauses: Dict[str, int] = Field(default_factory=dict)\n\n    # Tracking for objects created by this flow run\n    task_run_futures: List[PrefectFuture] = Field(default_factory=list)\n    task_run_states: List[State] = Field(default_factory=list)\n    task_run_results: Dict[int, State] = Field(default_factory=dict)\n    flow_run_states: List[State] = Field(default_factory=list)\n\n    # The synchronous portal is only created for async flows for creating engine calls\n    # from synchronous task and subflow calls\n    sync_portal: Optional[anyio.abc.BlockingPortal] = None\n    timeout_scope: Optional[anyio.abc.CancelScope] = None\n\n    # Task group that can be used for background tasks during the flow run\n    background_tasks: anyio.abc.TaskGroup\n\n    # Events worker to emit events to Prefect Cloud\n    events: Optional[EventsWorker] = None\n\n    __var__ = ContextVar(\"flow_run\")\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.PrefectObjectRegistry","title":"PrefectObjectRegistry","text":"

    Bases: ContextModel

    A context that acts as a registry for all Prefect objects that are registered during load and execution.

    Attributes:

    Name Type Description start_time DateTimeTZ

    The time the object registry was created.

    block_code_execution bool

    If set, flow calls will be ignored.

    capture_failures bool

    If set, failures during init will be silenced and tracked.

    Source code in prefect/context.py
    class PrefectObjectRegistry(ContextModel):\n    \"\"\"\n    A context that acts as a registry for all Prefect objects that are\n    registered during load and execution.\n\n    Attributes:\n        start_time: The time the object registry was created.\n        block_code_execution: If set, flow calls will be ignored.\n        capture_failures: If set, failures during __init__ will be silenced and tracked.\n    \"\"\"\n\n    start_time: DateTimeTZ = Field(default_factory=lambda: pendulum.now(\"UTC\"))\n\n    _instance_registry: Dict[Type[T], List[T]] = PrivateAttr(\n        default_factory=lambda: defaultdict(list)\n    )\n\n    # Failures will be a tuple of (exception, instance, args, kwargs)\n    _instance_init_failures: Dict[Type[T], List[Tuple[Exception, T, Tuple, Dict]]] = (\n        PrivateAttr(default_factory=lambda: defaultdict(list))\n    )\n\n    block_code_execution: bool = False\n    capture_failures: bool = False\n\n    __var__ = ContextVar(\"object_registry\")\n\n    def get_instances(self, type_: Type[T]) -> List[T]:\n        instances = []\n        for registered_type, type_instances in self._instance_registry.items():\n            if type_ in registered_type.mro():\n                instances.extend(type_instances)\n        return instances\n\n    def get_instance_failures(\n        self, type_: Type[T]\n    ) -> List[Tuple[Exception, T, Tuple, Dict]]:\n        failures = []\n        for type__ in type_.mro():\n            failures.extend(self._instance_init_failures[type__])\n        return failures\n\n    def register_instance(self, object):\n        # TODO: Consider using a 'Set' to avoid duplicate entries\n        self._instance_registry[type(object)].append(object)\n\n    def register_init_failure(\n        self, exc: Exception, object: Any, init_args: Tuple, init_kwargs: Dict\n    ):\n        self._instance_init_failures[type(object)].append(\n            (exc, object, init_args, init_kwargs)\n        )\n\n    @classmethod\n    def register_instances(cls, type_: Type):\n        \"\"\"\n        Decorator for a class that adds registration to the `PrefectObjectRegistry`\n        on initialization of instances.\n        \"\"\"\n        __init__ = type_.__init__\n\n        def __register_init__(__self__, *args, **kwargs):\n            registry = cls.get()\n            try:\n                __init__(__self__, *args, **kwargs)\n            except Exception as exc:\n                if not registry or not registry.capture_failures:\n                    raise\n                else:\n                    registry.register_init_failure(exc, __self__, args, kwargs)\n            else:\n                if registry:\n                    registry.register_instance(__self__)\n\n        type_.__init__ = __register_init__\n        return type_\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.PrefectObjectRegistry.register_instances","title":"register_instances classmethod","text":"

    Decorator for a class that adds registration to the PrefectObjectRegistry on initialization of instances.

    Source code in prefect/context.py
    @classmethod\ndef register_instances(cls, type_: Type):\n    \"\"\"\n    Decorator for a class that adds registration to the `PrefectObjectRegistry`\n    on initialization of instances.\n    \"\"\"\n    __init__ = type_.__init__\n\n    def __register_init__(__self__, *args, **kwargs):\n        registry = cls.get()\n        try:\n            __init__(__self__, *args, **kwargs)\n        except Exception as exc:\n            if not registry or not registry.capture_failures:\n                raise\n            else:\n                registry.register_init_failure(exc, __self__, args, kwargs)\n        else:\n            if registry:\n                registry.register_instance(__self__)\n\n    type_.__init__ = __register_init__\n    return type_\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.RunContext","title":"RunContext","text":"

    Bases: ContextModel

    The base context for a flow or task run. Data in this context will always be available when get_run_context is called.

    Attributes:

    Name Type Description start_time DateTimeTZ

    The time the run context was entered

    client PrefectClient

    The Prefect client instance being used for API communication

    Source code in prefect/context.py
    class RunContext(ContextModel):\n    \"\"\"\n    The base context for a flow or task run. Data in this context will always be\n    available when `get_run_context` is called.\n\n    Attributes:\n        start_time: The time the run context was entered\n        client: The Prefect client instance being used for API communication\n    \"\"\"\n\n    start_time: DateTimeTZ = Field(default_factory=lambda: pendulum.now(\"UTC\"))\n    input_keyset: Optional[Dict[str, Dict[str, str]]] = None\n    client: PrefectClient\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.SettingsContext","title":"SettingsContext","text":"

    Bases: ContextModel

    The context for a Prefect settings.

    This allows for safe concurrent access and modification of settings.

    Attributes:

    Name Type Description profile Profile

    The profile that is in use.

    settings Settings

    The complete settings model.

    Source code in prefect/context.py
    class SettingsContext(ContextModel):\n    \"\"\"\n    The context for a Prefect settings.\n\n    This allows for safe concurrent access and modification of settings.\n\n    Attributes:\n        profile: The profile that is in use.\n        settings: The complete settings model.\n    \"\"\"\n\n    profile: Profile\n    settings: Settings\n\n    __var__ = ContextVar(\"settings\")\n\n    def __hash__(self) -> int:\n        return hash(self.settings)\n\n    def __enter__(self):\n        \"\"\"\n        Upon entrance, we ensure the home directory for the profile exists.\n        \"\"\"\n        return_value = super().__enter__()\n\n        try:\n            prefect_home = Path(self.settings.value_of(PREFECT_HOME))\n            prefect_home.mkdir(mode=0o0700, exist_ok=True)\n        except OSError:\n            warnings.warn(\n                (\n                    \"Failed to create the Prefect home directory at \"\n                    f\"{self.settings.value_of(PREFECT_HOME)}\"\n                ),\n                stacklevel=2,\n            )\n\n        return return_value\n\n    @classmethod\n    def get(cls) -> \"SettingsContext\":\n        # Return the global context instead of `None` if no context exists\n        return super().get() or GLOBAL_SETTINGS_CONTEXT\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.TagsContext","title":"TagsContext","text":"

    Bases: ContextModel

    The context for prefect.tags management.

    Attributes:

    Name Type Description current_tags Set[str]

    A set of current tags in the context

    Source code in prefect/context.py
    class TagsContext(ContextModel):\n    \"\"\"\n    The context for `prefect.tags` management.\n\n    Attributes:\n        current_tags: A set of current tags in the context\n    \"\"\"\n\n    current_tags: Set[str] = Field(default_factory=set)\n\n    @classmethod\n    def get(cls) -> \"TagsContext\":\n        # Return an empty `TagsContext` instead of `None` if no context exists\n        return cls.__var__.get(TagsContext())\n\n    __var__ = ContextVar(\"tags\")\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.TaskRunContext","title":"TaskRunContext","text":"

    Bases: RunContext

    The context for a task run. Data in this context is only available from within a task run function.

    Attributes:

    Name Type Description task Task

    The task instance associated with the task run

    task_run TaskRun

    The API metadata for this task run

    Source code in prefect/context.py
    class TaskRunContext(RunContext):\n    \"\"\"\n    The context for a task run. Data in this context is only available from within a\n    task run function.\n\n    Attributes:\n        task: The task instance associated with the task run\n        task_run: The API metadata for this task run\n    \"\"\"\n\n    task: \"Task\"\n    task_run: TaskRun\n    log_prints: bool = False\n    parameters: Dict[str, Any]\n\n    # Result handling\n    result_factory: ResultFactory\n\n    __var__ = ContextVar(\"task_run\")\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.get_run_context","title":"get_run_context","text":"

    Get the current run context from within a task or flow function.

    Returns:

    Type Description Union[FlowRunContext, TaskRunContext]

    A FlowRunContext or TaskRunContext depending on the function type.

    Raises RuntimeError: If called outside of a flow or task run.

    Source code in prefect/context.py
    def get_run_context() -> Union[FlowRunContext, TaskRunContext]:\n    \"\"\"\n    Get the current run context from within a task or flow function.\n\n    Returns:\n        A `FlowRunContext` or `TaskRunContext` depending on the function type.\n\n    Raises\n        RuntimeError: If called outside of a flow or task run.\n    \"\"\"\n    task_run_ctx = TaskRunContext.get()\n    if task_run_ctx:\n        return task_run_ctx\n\n    flow_run_ctx = FlowRunContext.get()\n    if flow_run_ctx:\n        return flow_run_ctx\n\n    raise MissingContextError(\n        \"No run context available. You are not in a flow or task run context.\"\n    )\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.get_settings_context","title":"get_settings_context","text":"

    Get the current settings context which contains profile information and the settings that are being used.

    Generally, the settings that are being used are a combination of values from the profile and environment. See prefect.context.use_profile for more details.

    Source code in prefect/context.py
    def get_settings_context() -> SettingsContext:\n    \"\"\"\n    Get the current settings context which contains profile information and the\n    settings that are being used.\n\n    Generally, the settings that are being used are a combination of values from the\n    profile and environment. See `prefect.context.use_profile` for more details.\n    \"\"\"\n    settings_ctx = SettingsContext.get()\n\n    if not settings_ctx:\n        raise MissingContextError(\"No settings context found.\")\n\n    return settings_ctx\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.registry_from_script","title":"registry_from_script","text":"

    Return a fresh registry with instances populated from execution of a script.

    Source code in prefect/context.py
    def registry_from_script(\n    path: str,\n    block_code_execution: bool = True,\n    capture_failures: bool = True,\n) -> PrefectObjectRegistry:\n    \"\"\"\n    Return a fresh registry with instances populated from execution of a script.\n    \"\"\"\n    with PrefectObjectRegistry(\n        block_code_execution=block_code_execution,\n        capture_failures=capture_failures,\n    ) as registry:\n        load_script_as_module(path)\n\n    return registry\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.root_settings_context","title":"root_settings_context","text":"

    Return the settings context that will exist as the root context for the module.

    The profile to use is determined with the following precedence - Command line via 'prefect --profile ' - Environment variable via 'PREFECT_PROFILE' - Profiles file via the 'active' key Source code in prefect/context.py

    def root_settings_context():\n    \"\"\"\n    Return the settings context that will exist as the root context for the module.\n\n    The profile to use is determined with the following precedence\n    - Command line via 'prefect --profile <name>'\n    - Environment variable via 'PREFECT_PROFILE'\n    - Profiles file via the 'active' key\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    active_name = profiles.active_name\n    profile_source = \"in the profiles file\"\n\n    if \"PREFECT_PROFILE\" in os.environ:\n        active_name = os.environ[\"PREFECT_PROFILE\"]\n        profile_source = \"by environment variable\"\n\n    if (\n        sys.argv[0].endswith(\"/prefect\")\n        and len(sys.argv) >= 3\n        and sys.argv[1] == \"--profile\"\n    ):\n        active_name = sys.argv[2]\n        profile_source = \"by command line argument\"\n\n    if active_name not in profiles.names:\n        print(\n            (\n                f\"WARNING: Active profile {active_name!r} set {profile_source} not \"\n                \"found. The default profile will be used instead. \"\n            ),\n            file=sys.stderr,\n        )\n        active_name = \"default\"\n\n    with use_profile(\n        profiles[active_name],\n        # Override environment variables if the profile was set by the CLI\n        override_environment_variables=profile_source == \"by command line argument\",\n    ) as settings_context:\n        return settings_context\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.tags","title":"tags","text":"

    Context manager to add tags to flow and task run calls.

    Tags are always combined with any existing tags.

    Yields:

    Type Description Set[str]

    The current set of tags

    Examples:

    >>> from prefect import tags, task, flow\n>>> @task\n>>> def my_task():\n>>>     pass\n

    Run a task with tags

    >>> @flow\n>>> def my_flow():\n>>>     with tags(\"a\", \"b\"):\n>>>         my_task()  # has tags: a, b\n

    Run a flow with tags

    >>> @flow\n>>> def my_flow():\n>>>     pass\n>>> with tags(\"a\", \"b\"):\n>>>     my_flow()  # has tags: a, b\n

    Run a task with nested tag contexts

    >>> @flow\n>>> def my_flow():\n>>>     with tags(\"a\", \"b\"):\n>>>         with tags(\"c\", \"d\"):\n>>>             my_task()  # has tags: a, b, c, d\n>>>         my_task()  # has tags: a, b\n

    Inspect the current tags

    >>> @flow\n>>> def my_flow():\n>>>     with tags(\"c\", \"d\"):\n>>>         with tags(\"e\", \"f\") as current_tags:\n>>>              print(current_tags)\n>>> with tags(\"a\", \"b\"):\n>>>     my_flow()\n{\"a\", \"b\", \"c\", \"d\", \"e\", \"f\"}\n
    Source code in prefect/context.py
    @contextmanager\ndef tags(*new_tags: str) -> Set[str]:\n    \"\"\"\n    Context manager to add tags to flow and task run calls.\n\n    Tags are always combined with any existing tags.\n\n    Yields:\n        The current set of tags\n\n    Examples:\n        >>> from prefect import tags, task, flow\n        >>> @task\n        >>> def my_task():\n        >>>     pass\n\n        Run a task with tags\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     with tags(\"a\", \"b\"):\n        >>>         my_task()  # has tags: a, b\n\n        Run a flow with tags\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     pass\n        >>> with tags(\"a\", \"b\"):\n        >>>     my_flow()  # has tags: a, b\n\n        Run a task with nested tag contexts\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     with tags(\"a\", \"b\"):\n        >>>         with tags(\"c\", \"d\"):\n        >>>             my_task()  # has tags: a, b, c, d\n        >>>         my_task()  # has tags: a, b\n\n        Inspect the current tags\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     with tags(\"c\", \"d\"):\n        >>>         with tags(\"e\", \"f\") as current_tags:\n        >>>              print(current_tags)\n        >>> with tags(\"a\", \"b\"):\n        >>>     my_flow()\n        {\"a\", \"b\", \"c\", \"d\", \"e\", \"f\"}\n    \"\"\"\n    current_tags = TagsContext.get().current_tags\n    new_tags = current_tags.union(new_tags)\n    with TagsContext(current_tags=new_tags):\n        yield new_tags\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.use_profile","title":"use_profile","text":"

    Switch to a profile for the duration of this context.

    Profile contexts are confined to an async context in a single thread.

    Parameters:

    Name Type Description Default profile Union[Profile, str]

    The name of the profile to load or an instance of a Profile.

    required override_environment_variable

    If set, variables in the profile will take precedence over current environment variables. By default, environment variables will override profile settings.

    required include_current_context bool

    If set, the new settings will be constructed with the current settings context as a base. If not set, the use_base settings will be loaded from the environment and defaults.

    True

    Yields:

    Type Description

    The created SettingsContext object

    Source code in prefect/context.py
    @contextmanager\ndef use_profile(\n    profile: Union[Profile, str],\n    override_environment_variables: bool = False,\n    include_current_context: bool = True,\n):\n    \"\"\"\n    Switch to a profile for the duration of this context.\n\n    Profile contexts are confined to an async context in a single thread.\n\n    Args:\n        profile: The name of the profile to load or an instance of a Profile.\n        override_environment_variable: If set, variables in the profile will take\n            precedence over current environment variables. By default, environment\n            variables will override profile settings.\n        include_current_context: If set, the new settings will be constructed\n            with the current settings context as a base. If not set, the use_base settings\n            will be loaded from the environment and defaults.\n\n    Yields:\n        The created `SettingsContext` object\n    \"\"\"\n    if isinstance(profile, str):\n        profiles = prefect.settings.load_profiles()\n        profile = profiles[profile]\n\n    if not isinstance(profile, Profile):\n        raise TypeError(\n            f\"Unexpected type {type(profile).__name__!r} for `profile`. \"\n            \"Expected 'str' or 'Profile'.\"\n        )\n\n    # Create a copy of the profiles settings as we will mutate it\n    profile_settings = profile.settings.copy()\n\n    existing_context = SettingsContext.get()\n    if existing_context and include_current_context:\n        settings = existing_context.settings\n    else:\n        settings = prefect.settings.get_settings_from_env()\n\n    if not override_environment_variables:\n        for key in os.environ:\n            if key in prefect.settings.SETTING_VARIABLES:\n                profile_settings.pop(prefect.settings.SETTING_VARIABLES[key], None)\n\n    new_settings = settings.copy_with_update(updates=profile_settings)\n\n    with SettingsContext(profile=profile, settings=new_settings) as ctx:\n        yield ctx\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/engine/","title":"prefect.engine","text":"","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine","title":"prefect.engine","text":"

    Client-side execution and orchestration of flows and tasks.

    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine--engine-process-overview","title":"Engine process overview","text":"","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine--flows","title":"Flows","text":"
    • The flow is called by the user or an existing flow run is executed in a new process.

      See Flow.__call__ and prefect.engine.__main__ (python -m prefect.engine)

    • A synchronous function acts as an entrypoint to the engine. The engine executes on a dedicated \"global loop\" thread. For asynchronous flow calls, we return a coroutine from the entrypoint so the user can enter the engine without blocking their event loop.

      See enter_flow_run_engine_from_flow_call, enter_flow_run_engine_from_subprocess

    • The thread that calls the entrypoint waits until orchestration of the flow run completes. This thread is referred to as the \"user\" thread and is usually the \"main\" thread. The thread is not blocked while waiting \u2014 it allows the engine to send work back to it. This allows us to send calls back to the user thread from the global loop thread.

      See wait_for_call_in_loop_thread and call_soon_in_waiting_thread

    • The asynchronous engine branches depending on if the flow run exists already and if there is a parent flow run in the current context.

      See create_then_begin_flow_run, create_and_begin_subflow_run, and retrieve_flow_then_begin_flow_run

    • The asynchronous engine prepares for execution of the flow run. This includes starting the task runner, preparing context, etc.

      See begin_flow_run

    • The flow run is orchestrated through states, calling the user's function as necessary. Generally the user's function is sent for execution on the user thread. If the flow function cannot be safely executed on the user thread, e.g. it is a synchronous child in an asynchronous parent it will be scheduled on a worker thread instead.

      See orchestrate_flow_run, call_soon_in_waiting_thread, call_soon_in_new_thread

    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine--tasks","title":"Tasks","text":"
    • The task is called or submitted by the user. We require that this is always within a flow.

      See Task.__call__ and Task.submit

    • A synchronous function acts as an entrypoint to the engine. Unlike flow calls, this will not block until completion if submit was used.

      See enter_task_run_engine

    • A future is created for the task call. Creation of the task run and submission to the task runner is scheduled as a background task so submission of many tasks can occur concurrently.

      See create_task_run_future and create_task_run_then_submit

    • The engine branches depending on if a future, state, or result is requested. If a future is requested, it is returned immediately to the user thread. Otherwise, the engine will wait for the task run to complete and return the final state or result.

      See get_task_call_return_value

    • An engine function is submitted to the task runner. The task runner will schedule this function for execution on a worker. When executed, it will prepare for orchestration and wait for completion of the run.

      See create_task_run_then_submit and begin_task_run

    • The task run is orchestrated through states, calling the user's function as necessary. The user's function is always executed in a worker thread for isolation.

      See orchestrate_task_run, call_soon_in_new_thread

      _Ideally, for local and sequential task runners we would send the task run to the user thread as we do for flows. See #9855.

    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.begin_flow_run","title":"begin_flow_run async","text":"

    Begins execution of a flow run; blocks until completion of the flow run

    • Starts a task runner
    • Determines the result storage block to use
    • Orchestrates the flow run (runs the user-function and generates tasks)
    • Waits for tasks to complete / shutsdown the task runner
    • Sets a terminal state for the flow run

    Note that the flow_run contains a parameters attribute which is the serialized parameters sent to the backend while the parameters argument here should be the deserialized and validated dictionary of python objects.

    Returns:

    Type Description State

    The final state of the run

    Source code in prefect/engine.py
    async def begin_flow_run(\n    flow: Flow,\n    flow_run: FlowRun,\n    parameters: Dict[str, Any],\n    client: PrefectClient,\n    user_thread: threading.Thread,\n) -> State:\n    \"\"\"\n    Begins execution of a flow run; blocks until completion of the flow run\n\n    - Starts a task runner\n    - Determines the result storage block to use\n    - Orchestrates the flow run (runs the user-function and generates tasks)\n    - Waits for tasks to complete / shutsdown the task runner\n    - Sets a terminal state for the flow run\n\n    Note that the `flow_run` contains a `parameters` attribute which is the serialized\n    parameters sent to the backend while the `parameters` argument here should be the\n    deserialized and validated dictionary of python objects.\n\n    Returns:\n        The final state of the run\n    \"\"\"\n    logger = flow_run_logger(flow_run, flow)\n\n    log_prints = should_log_prints(flow)\n    flow_run_context = PartialModel(FlowRunContext, log_prints=log_prints)\n\n    async with AsyncExitStack() as stack:\n        await stack.enter_async_context(\n            report_flow_run_crashes(flow_run=flow_run, client=client, flow=flow)\n        )\n\n        # Create a task group for background tasks\n        flow_run_context.background_tasks = await stack.enter_async_context(\n            anyio.create_task_group()\n        )\n\n        # If the flow is async, we need to provide a portal so sync tasks can run\n        flow_run_context.sync_portal = (\n            stack.enter_context(start_blocking_portal()) if flow.isasync else None\n        )\n\n        task_runner = flow.task_runner.duplicate()\n        if task_runner is NotImplemented:\n            # Backwards compatibility; will not support concurrent flow runs\n            task_runner = flow.task_runner\n            logger.warning(\n                f\"Task runner {type(task_runner).__name__!r} does not implement the\"\n                \" `duplicate` method and will fail if used for concurrent execution of\"\n                \" the same flow.\"\n            )\n\n        logger.debug(\n            f\"Starting {type(flow.task_runner).__name__!r}; submitted tasks \"\n            f\"will be run {CONCURRENCY_MESSAGES[flow.task_runner.concurrency_type]}...\"\n        )\n\n        flow_run_context.task_runner = await stack.enter_async_context(\n            task_runner.start()\n        )\n\n        flow_run_context.result_factory = await ResultFactory.from_flow(\n            flow, client=client\n        )\n\n        if log_prints:\n            stack.enter_context(patch_print())\n\n        terminal_or_paused_state = await orchestrate_flow_run(\n            flow,\n            flow_run=flow_run,\n            parameters=parameters,\n            wait_for=None,\n            client=client,\n            partial_flow_run_context=flow_run_context,\n            # Orchestration needs to be interruptible if it has a timeout\n            interruptible=flow.timeout_seconds is not None,\n            user_thread=user_thread,\n        )\n\n    if terminal_or_paused_state.is_paused():\n        timeout = terminal_or_paused_state.state_details.pause_timeout\n        msg = \"Currently paused and suspending execution.\"\n        if timeout:\n            msg += f\" Resume before {timeout.to_rfc3339_string()} to finish execution.\"\n        logger.log(level=logging.INFO, msg=msg)\n        await APILogHandler.aflush()\n\n        return terminal_or_paused_state\n    else:\n        terminal_state = terminal_or_paused_state\n\n    # If debugging, use the more complete `repr` than the usual `str` description\n    display_state = repr(terminal_state) if PREFECT_DEBUG_MODE else str(terminal_state)\n\n    logger.log(\n        level=logging.INFO if terminal_state.is_completed() else logging.ERROR,\n        msg=f\"Finished in state {display_state}\",\n    )\n\n    # When a \"root\" flow run finishes, flush logs so we do not have to rely on handling\n    # during interpreter shutdown\n    await APILogHandler.aflush()\n\n    return terminal_state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.begin_task_map","title":"begin_task_map async","text":"

    Async entrypoint for task mapping

    Source code in prefect/engine.py
    async def begin_task_map(\n    task: Task,\n    flow_run_context: FlowRunContext,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n    task_runner: Optional[BaseTaskRunner],\n) -> List[Union[PrefectFuture, Awaitable[PrefectFuture]]]:\n    \"\"\"Async entrypoint for task mapping\"\"\"\n    # We need to resolve some futures to map over their data, collect the upstream\n    # links beforehand to retain relationship tracking.\n    task_inputs = {\n        k: await collect_task_run_inputs(v, max_depth=0) for k, v in parameters.items()\n    }\n\n    # Resolve the top-level parameters in order to get mappable data of a known length.\n    # Nested parameters will be resolved in each mapped child where their relationships\n    # will also be tracked.\n    parameters = await resolve_inputs(parameters, max_depth=1)\n\n    # Ensure that any parameters in kwargs are expanded before this check\n    parameters = explode_variadic_parameter(task.fn, parameters)\n\n    iterable_parameters = {}\n    static_parameters = {}\n    annotated_parameters = {}\n    for key, val in parameters.items():\n        if isinstance(val, (allow_failure, quote)):\n            # Unwrap annotated parameters to determine if they are iterable\n            annotated_parameters[key] = val\n            val = val.unwrap()\n\n        if isinstance(val, unmapped):\n            static_parameters[key] = val.value\n        elif isiterable(val):\n            iterable_parameters[key] = list(val)\n        else:\n            static_parameters[key] = val\n\n    if not len(iterable_parameters):\n        raise MappingMissingIterable(\n            \"No iterable parameters were received. Parameters for map must \"\n            f\"include at least one iterable. Parameters: {parameters}\"\n        )\n\n    iterable_parameter_lengths = {\n        key: len(val) for key, val in iterable_parameters.items()\n    }\n    lengths = set(iterable_parameter_lengths.values())\n    if len(lengths) > 1:\n        raise MappingLengthMismatch(\n            \"Received iterable parameters with different lengths. Parameters for map\"\n            f\" must all be the same length. Got lengths: {iterable_parameter_lengths}\"\n        )\n\n    map_length = list(lengths)[0]\n\n    task_runs = []\n    for i in range(map_length):\n        call_parameters = {key: value[i] for key, value in iterable_parameters.items()}\n        call_parameters.update({key: value for key, value in static_parameters.items()})\n\n        # Add default values for parameters; these are skipped earlier since they should\n        # not be mapped over\n        for key, value in get_parameter_defaults(task.fn).items():\n            call_parameters.setdefault(key, value)\n\n        # Re-apply annotations to each key again\n        for key, annotation in annotated_parameters.items():\n            call_parameters[key] = annotation.rewrap(call_parameters[key])\n\n        # Collapse any previously exploded kwargs\n        call_parameters = collapse_variadic_parameters(task.fn, call_parameters)\n\n        task_runs.append(\n            partial(\n                get_task_call_return_value,\n                task=task,\n                flow_run_context=flow_run_context,\n                parameters=call_parameters,\n                wait_for=wait_for,\n                return_type=return_type,\n                task_runner=task_runner,\n                extra_task_inputs=task_inputs,\n            )\n        )\n\n    # Maintain the order of the task runs when using the sequential task runner\n    runner = task_runner if task_runner else flow_run_context.task_runner\n    if runner.concurrency_type == TaskConcurrencyType.SEQUENTIAL:\n        return [await task_run() for task_run in task_runs]\n\n    return await gather(*task_runs)\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.begin_task_run","title":"begin_task_run async","text":"

    Entrypoint for task run execution.

    This function is intended for submission to the task runner.

    This method may be called from a worker so we ensure the settings context has been entered. For example, with a runner that is executing tasks in the same event loop, we will likely not enter the context again because the current context already matches:

    main thread: --> Flow called with settings A --> begin_task_run executes same event loop --> Profile A matches and is not entered again

    However, with execution on a remote environment, we are going to need to ensure the settings for the task run are respected by entering the context:

    main thread: --> Flow called with settings A --> begin_task_run is scheduled on a remote worker, settings A is serialized remote worker: --> Remote worker imports Prefect (may not occur) --> Global settings is loaded with default settings --> begin_task_run executes on a different event loop than the flow --> Current settings is not set or does not match, settings A is entered

    Source code in prefect/engine.py
    async def begin_task_run(\n    task: Task,\n    task_run: TaskRun,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    result_factory: ResultFactory,\n    log_prints: bool,\n    settings: prefect.context.SettingsContext,\n):\n    \"\"\"\n    Entrypoint for task run execution.\n\n    This function is intended for submission to the task runner.\n\n    This method may be called from a worker so we ensure the settings context has been\n    entered. For example, with a runner that is executing tasks in the same event loop,\n    we will likely not enter the context again because the current context already\n    matches:\n\n    main thread:\n    --> Flow called with settings A\n    --> `begin_task_run` executes same event loop\n    --> Profile A matches and is not entered again\n\n    However, with execution on a remote environment, we are going to need to ensure the\n    settings for the task run are respected by entering the context:\n\n    main thread:\n    --> Flow called with settings A\n    --> `begin_task_run` is scheduled on a remote worker, settings A is serialized\n    remote worker:\n    --> Remote worker imports Prefect (may not occur)\n    --> Global settings is loaded with default settings\n    --> `begin_task_run` executes on a different event loop than the flow\n    --> Current settings is not set or does not match, settings A is entered\n    \"\"\"\n    maybe_flow_run_context = prefect.context.FlowRunContext.get()\n\n    async with AsyncExitStack() as stack:\n        # The settings context may be null on a remote worker so we use the safe `.get`\n        # method and compare it to the settings required for this task run\n        if prefect.context.SettingsContext.get() != settings:\n            stack.enter_context(settings)\n            setup_logging()\n\n        if maybe_flow_run_context:\n            # Accessible if on a worker that is running in the same thread as the flow\n            client = maybe_flow_run_context.client\n            # Only run the task in an interruptible thread if it in the same thread as\n            # the flow _and_ the flow run has a timeout attached. If the task is on a\n            # worker, the flow run timeout will not be raised in the worker process.\n            interruptible = maybe_flow_run_context.timeout_scope is not None\n        else:\n            # Otherwise, retrieve a new client\n            client = await stack.enter_async_context(get_client())\n            interruptible = False\n            await stack.enter_async_context(anyio.create_task_group())\n\n        await stack.enter_async_context(report_task_run_crashes(task_run, client))\n\n        # TODO: Use the background tasks group to manage logging for this task\n\n        if log_prints:\n            stack.enter_context(patch_print())\n\n        await check_api_reachable(\n            client, f\"Cannot orchestrate task run '{task_run.id}'\"\n        )\n        try:\n            state = await orchestrate_task_run(\n                task=task,\n                task_run=task_run,\n                parameters=parameters,\n                wait_for=wait_for,\n                result_factory=result_factory,\n                log_prints=log_prints,\n                interruptible=interruptible,\n                client=client,\n            )\n\n            if not maybe_flow_run_context:\n                # When a a task run finishes on a remote worker flush logs to prevent\n                # loss if the process exits\n                await APILogHandler.aflush()\n\n        except Abort as abort:\n            # Task run probably already completed, fetch its state\n            task_run = await client.read_task_run(task_run.id)\n\n            if task_run.state.is_final():\n                task_run_logger(task_run).info(\n                    f\"Task run '{task_run.id}' already finished.\"\n                )\n            else:\n                # TODO: This is a concerning case; we should determine when this occurs\n                #       1. This can occur when the flow run is not in a running state\n                task_run_logger(task_run).warning(\n                    f\"Task run '{task_run.id}' received abort during orchestration: \"\n                    f\"{abort} Task run is in {task_run.state.type.value} state.\"\n                )\n            state = task_run.state\n\n        except Pause:\n            # A pause signal here should mean the flow run suspended, so we\n            # should do the same. We'll look up the flow run's pause state to\n            # try and reuse it, so we capture any data like timeouts.\n            flow_run = await client.read_flow_run(task_run.flow_run_id)\n            if flow_run.state and flow_run.state.is_paused():\n                state = flow_run.state\n            else:\n                state = Suspended()\n\n            task_run_logger(task_run).info(\n                \"Task run encountered a pause signal during orchestration.\"\n            )\n\n        return state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.collect_task_run_inputs","title":"collect_task_run_inputs async","text":"

    This function recurses through an expression to generate a set of any discernible task run inputs it finds in the data structure. It produces a set of all inputs found.

    Examples:

    >>> task_inputs = {\n>>>    k: await collect_task_run_inputs(v) for k, v in parameters.items()\n>>> }\n
    Source code in prefect/engine.py
    async def collect_task_run_inputs(expr: Any, max_depth: int = -1) -> Set[TaskRunInput]:\n    \"\"\"\n    This function recurses through an expression to generate a set of any discernible\n    task run inputs it finds in the data structure. It produces a set of all inputs\n    found.\n\n    Examples:\n        >>> task_inputs = {\n        >>>    k: await collect_task_run_inputs(v) for k, v in parameters.items()\n        >>> }\n    \"\"\"\n    # TODO: This function needs to be updated to detect parameters and constants\n\n    inputs = set()\n    futures = set()\n\n    def add_futures_and_states_to_inputs(obj):\n        if isinstance(obj, PrefectFuture):\n            # We need to wait for futures to be submitted before we can get the task\n            # run id but we want to do so asynchronously\n            futures.add(obj)\n        elif is_state(obj):\n            if obj.state_details.task_run_id:\n                inputs.add(TaskRunResult(id=obj.state_details.task_run_id))\n        # Expressions inside quotes should not be traversed\n        elif isinstance(obj, quote):\n            raise StopVisiting\n        else:\n            state = get_state_for_result(obj)\n            if state and state.state_details.task_run_id:\n                inputs.add(TaskRunResult(id=state.state_details.task_run_id))\n\n    visit_collection(\n        expr,\n        visit_fn=add_futures_and_states_to_inputs,\n        return_data=False,\n        max_depth=max_depth,\n    )\n\n    await asyncio.gather(*[future._wait_for_submission() for future in futures])\n    for future in futures:\n        inputs.add(TaskRunResult(id=future.task_run.id))\n\n    return inputs\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.create_and_begin_subflow_run","title":"create_and_begin_subflow_run async","text":"

    Async entrypoint for flows calls within a flow run

    Subflows differ from parent flows in that they - Resolve futures in passed parameters into values - Create a dummy task for representation in the parent flow - Retrieve default result storage from the parent flow rather than the server

    Returns:

    Type Description Any

    The final state of the run

    Source code in prefect/engine.py
    @inject_client\nasync def create_and_begin_subflow_run(\n    flow: Flow,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n    client: PrefectClient,\n    user_thread: threading.Thread,\n) -> Any:\n    \"\"\"\n    Async entrypoint for flows calls within a flow run\n\n    Subflows differ from parent flows in that they\n    - Resolve futures in passed parameters into values\n    - Create a dummy task for representation in the parent flow\n    - Retrieve default result storage from the parent flow rather than the server\n\n    Returns:\n        The final state of the run\n    \"\"\"\n    parent_flow_run_context = FlowRunContext.get()\n    parent_logger = get_run_logger(parent_flow_run_context)\n    log_prints = should_log_prints(flow)\n    terminal_state = None\n\n    parent_logger.debug(f\"Resolving inputs to {flow.name!r}\")\n    task_inputs = {k: await collect_task_run_inputs(v) for k, v in parameters.items()}\n\n    if wait_for:\n        task_inputs[\"wait_for\"] = await collect_task_run_inputs(wait_for)\n\n    rerunning = parent_flow_run_context.flow_run.run_count > 1\n\n    # Generate a task in the parent flow run to represent the result of the subflow run\n    dummy_task = Task(name=flow.name, fn=flow.fn, version=flow.version)\n    parent_task_run = await client.create_task_run(\n        task=dummy_task,\n        flow_run_id=parent_flow_run_context.flow_run.id,\n        dynamic_key=_dynamic_key_for_task_run(parent_flow_run_context, dummy_task),\n        task_inputs=task_inputs,\n        state=Pending(),\n    )\n\n    # Resolve any task futures in the input\n    parameters = await resolve_inputs(parameters)\n\n    if parent_task_run.state.is_final() and not (\n        rerunning and not parent_task_run.state.is_completed()\n    ):\n        # Retrieve the most recent flow run from the database\n        flow_runs = await client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                parent_task_run_id={\"any_\": [parent_task_run.id]}\n            ),\n            sort=FlowRunSort.EXPECTED_START_TIME_ASC,\n        )\n        flow_run = flow_runs[-1]\n\n        # Set up variables required downstream\n        terminal_state = flow_run.state\n        logger = flow_run_logger(flow_run, flow)\n\n    else:\n        flow_run = await client.create_flow_run(\n            flow,\n            parameters=flow.serialize_parameters(parameters),\n            parent_task_run_id=parent_task_run.id,\n            state=parent_task_run.state if not rerunning else Pending(),\n            tags=TagsContext.get().current_tags,\n        )\n\n        parent_logger.info(\n            f\"Created subflow run {flow_run.name!r} for flow {flow.name!r}\"\n        )\n\n        logger = flow_run_logger(flow_run, flow)\n        ui_url = PREFECT_UI_URL.value()\n        if ui_url:\n            logger.info(\n                f\"View at {ui_url}/flow-runs/flow-run/{flow_run.id}\",\n                extra={\"send_to_api\": False},\n            )\n\n        result_factory = await ResultFactory.from_flow(\n            flow, client=parent_flow_run_context.client\n        )\n\n        if flow.should_validate_parameters:\n            try:\n                parameters = flow.validate_parameters(parameters)\n            except Exception:\n                message = \"Validation of flow parameters failed with error:\"\n                logger.exception(message)\n                terminal_state = await propose_state(\n                    client,\n                    state=await exception_to_failed_state(\n                        message=message, result_factory=result_factory\n                    ),\n                    flow_run_id=flow_run.id,\n                )\n\n        if terminal_state is None or not terminal_state.is_final():\n            async with AsyncExitStack() as stack:\n                await stack.enter_async_context(\n                    report_flow_run_crashes(flow_run=flow_run, client=client, flow=flow)\n                )\n\n                task_runner = flow.task_runner.duplicate()\n                if task_runner is NotImplemented:\n                    # Backwards compatibility; will not support concurrent flow runs\n                    task_runner = flow.task_runner\n                    logger.warning(\n                        f\"Task runner {type(task_runner).__name__!r} does not implement\"\n                        \" the `duplicate` method and will fail if used for concurrent\"\n                        \" execution of the same flow.\"\n                    )\n\n                await stack.enter_async_context(task_runner.start())\n\n                if log_prints:\n                    stack.enter_context(patch_print())\n\n                terminal_state = await orchestrate_flow_run(\n                    flow,\n                    flow_run=flow_run,\n                    parameters=parameters,\n                    wait_for=wait_for,\n                    # If the parent flow run has a timeout, then this one needs to be\n                    # interruptible as well\n                    interruptible=parent_flow_run_context.timeout_scope is not None,\n                    client=client,\n                    partial_flow_run_context=PartialModel(\n                        FlowRunContext,\n                        sync_portal=parent_flow_run_context.sync_portal,\n                        task_runner=task_runner,\n                        background_tasks=parent_flow_run_context.background_tasks,\n                        result_factory=result_factory,\n                        log_prints=log_prints,\n                    ),\n                    user_thread=user_thread,\n                )\n\n    # Display the full state (including the result) if debugging\n    display_state = repr(terminal_state) if PREFECT_DEBUG_MODE else str(terminal_state)\n    logger.log(\n        level=logging.INFO if terminal_state.is_completed() else logging.ERROR,\n        msg=f\"Finished in state {display_state}\",\n    )\n\n    # Track the subflow state so the parent flow can use it to determine its final state\n    parent_flow_run_context.flow_run_states.append(terminal_state)\n\n    if return_type == \"state\":\n        return terminal_state\n    elif return_type == \"result\":\n        return await terminal_state.result(fetch=True)\n    else:\n        raise ValueError(f\"Invalid return type for flow engine {return_type!r}.\")\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.create_then_begin_flow_run","title":"create_then_begin_flow_run async","text":"

    Async entrypoint for flow calls

    Creates the flow run in the backend, then enters the main flow run engine.

    Source code in prefect/engine.py
    @inject_client\nasync def create_then_begin_flow_run(\n    flow: Flow,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n    client: PrefectClient,\n    user_thread: threading.Thread,\n) -> Any:\n    \"\"\"\n    Async entrypoint for flow calls\n\n    Creates the flow run in the backend, then enters the main flow run engine.\n    \"\"\"\n    # TODO: Returns a `State` depending on `return_type` and we can add an overload to\n    #       the function signature to clarify this eventually.\n\n    await check_api_reachable(client, \"Cannot create flow run\")\n\n    state = Pending()\n    if flow.should_validate_parameters:\n        try:\n            parameters = flow.validate_parameters(parameters)\n        except Exception:\n            state = await exception_to_failed_state(\n                message=\"Validation of flow parameters failed with error:\"\n            )\n\n    flow_run = await client.create_flow_run(\n        flow,\n        # Send serialized parameters to the backend\n        parameters=flow.serialize_parameters(parameters),\n        state=state,\n        tags=TagsContext.get().current_tags,\n    )\n\n    engine_logger.info(f\"Created flow run {flow_run.name!r} for flow {flow.name!r}\")\n\n    logger = flow_run_logger(flow_run, flow)\n\n    ui_url = PREFECT_UI_URL.value()\n    if ui_url:\n        logger.info(\n            f\"View at {ui_url}/flow-runs/flow-run/{flow_run.id}\",\n            extra={\"send_to_api\": False},\n        )\n\n    if state.is_failed():\n        logger.error(state.message)\n        engine_logger.info(\n            f\"Flow run {flow_run.name!r} received invalid parameters and is marked as\"\n            \" failed.\"\n        )\n    else:\n        state = await begin_flow_run(\n            flow=flow,\n            flow_run=flow_run,\n            parameters=parameters,\n            client=client,\n            user_thread=user_thread,\n        )\n\n    if return_type == \"state\":\n        return state\n    elif return_type == \"result\":\n        return await state.result(fetch=True)\n    else:\n        raise ValueError(f\"Invalid return type for flow engine {return_type!r}.\")\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.enter_flow_run_engine_from_flow_call","title":"enter_flow_run_engine_from_flow_call","text":"

    Sync entrypoint for flow calls.

    This function does the heavy lifting of ensuring we can get into an async context for flow run execution with minimal overhead.

    Source code in prefect/engine.py
    def enter_flow_run_engine_from_flow_call(\n    flow: Flow,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n) -> Union[State, Awaitable[State]]:\n    \"\"\"\n    Sync entrypoint for flow calls.\n\n    This function does the heavy lifting of ensuring we can get into an async context\n    for flow run execution with minimal overhead.\n    \"\"\"\n    setup_logging()\n\n    registry = PrefectObjectRegistry.get()\n    if registry and registry.block_code_execution:\n        engine_logger.warning(\n            f\"Script loading is in progress, flow {flow.name!r} will not be executed.\"\n            \" Consider updating the script to only call the flow if executed\"\n            f' directly:\\n\\n\\tif __name__ == \"__main__\":\\n\\t\\t{flow.fn.__name__}()'\n        )\n        return None\n\n    if TaskRunContext.get():\n        raise RuntimeError(\n            \"Flows cannot be run from within tasks. Did you mean to call this \"\n            \"flow in a flow?\"\n        )\n\n    parent_flow_run_context = FlowRunContext.get()\n    is_subflow_run = parent_flow_run_context is not None\n\n    if wait_for is not None and not is_subflow_run:\n        raise ValueError(\"Only flows run as subflows can wait for dependencies.\")\n\n    begin_run = create_call(\n        create_and_begin_subflow_run if is_subflow_run else create_then_begin_flow_run,\n        flow=flow,\n        parameters=parameters,\n        wait_for=wait_for,\n        return_type=return_type,\n        client=parent_flow_run_context.client if is_subflow_run else None,\n        user_thread=threading.current_thread(),\n    )\n\n    # On completion of root flows, wait for the global thread to ensure that\n    # any work there is complete\n    done_callbacks = (\n        [create_call(wait_for_global_loop_exit)] if not is_subflow_run else None\n    )\n\n    # WARNING: You must define any context managers here to pass to our concurrency\n    # api instead of entering them in here in the engine entrypoint. Otherwise, async\n    # flows will not use the context as this function _exits_ to return an awaitable to\n    # the user. Generally, you should enter contexts _within_ the async `begin_run`\n    # instead but if you need to enter a context from the main thread you'll need to do\n    # it here.\n    contexts = [capture_sigterm()]\n\n    if flow.isasync and (\n        not is_subflow_run or (is_subflow_run and parent_flow_run_context.flow.isasync)\n    ):\n        # return a coro for the user to await if the flow is async\n        # unless it is an async subflow called in a sync flow\n        retval = from_async.wait_for_call_in_loop_thread(\n            begin_run,\n            done_callbacks=done_callbacks,\n            contexts=contexts,\n        )\n\n    else:\n        retval = from_sync.wait_for_call_in_loop_thread(\n            begin_run,\n            done_callbacks=done_callbacks,\n            contexts=contexts,\n        )\n\n    return retval\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.enter_flow_run_engine_from_subprocess","title":"enter_flow_run_engine_from_subprocess","text":"

    Sync entrypoint for flow runs that have been submitted for execution by an agent

    Differs from enter_flow_run_engine_from_flow_call in that we have a flow run id but not a flow object. The flow must be retrieved before execution can begin. Additionally, this assumes that the caller is always in a context without an event loop as this should be called from a fresh process.

    Source code in prefect/engine.py
    def enter_flow_run_engine_from_subprocess(flow_run_id: UUID) -> State:\n    \"\"\"\n    Sync entrypoint for flow runs that have been submitted for execution by an agent\n\n    Differs from `enter_flow_run_engine_from_flow_call` in that we have a flow run id\n    but not a flow object. The flow must be retrieved before execution can begin.\n    Additionally, this assumes that the caller is always in a context without an event\n    loop as this should be called from a fresh process.\n    \"\"\"\n\n    # Ensure collections are imported and have the opportunity to register types before\n    # loading the user code from the deployment\n    prefect.plugins.load_prefect_collections()\n\n    setup_logging()\n\n    state = from_sync.wait_for_call_in_loop_thread(\n        create_call(\n            retrieve_flow_then_begin_flow_run,\n            flow_run_id,\n            user_thread=threading.current_thread(),\n        ),\n        contexts=[capture_sigterm()],\n    )\n\n    APILogHandler.flush()\n    return state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.enter_task_run_engine","title":"enter_task_run_engine","text":"

    Sync entrypoint for task calls

    Source code in prefect/engine.py
    def enter_task_run_engine(\n    task: Task,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n    task_runner: Optional[BaseTaskRunner],\n    mapped: bool,\n) -> Union[PrefectFuture, Awaitable[PrefectFuture]]:\n    \"\"\"\n    Sync entrypoint for task calls\n    \"\"\"\n\n    flow_run_context = FlowRunContext.get()\n    if not flow_run_context:\n        raise RuntimeError(\n            \"Tasks cannot be run outside of a flow. To call the underlying task\"\n            \" function outside of a flow use `task.fn()`.\"\n        )\n\n    if TaskRunContext.get():\n        raise RuntimeError(\n            \"Tasks cannot be run from within tasks. Did you mean to call this \"\n            \"task in a flow?\"\n        )\n\n    if flow_run_context.timeout_scope and flow_run_context.timeout_scope.cancel_called:\n        raise TimeoutError(\"Flow run timed out\")\n\n    begin_run = create_call(\n        begin_task_map if mapped else get_task_call_return_value,\n        task=task,\n        flow_run_context=flow_run_context,\n        parameters=parameters,\n        wait_for=wait_for,\n        return_type=return_type,\n        task_runner=task_runner,\n    )\n\n    if task.isasync and flow_run_context.flow.isasync:\n        # return a coro for the user to await if an async task in an async flow\n        return from_async.wait_for_call_in_loop_thread(begin_run)\n    else:\n        return from_sync.wait_for_call_in_loop_thread(begin_run)\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.get_state_for_result","title":"get_state_for_result","text":"

    Get the state related to a result object.

    link_state_to_result must have been called first.

    Source code in prefect/engine.py
    def get_state_for_result(obj: Any) -> Optional[State]:\n    \"\"\"\n    Get the state related to a result object.\n\n    `link_state_to_result` must have been called first.\n    \"\"\"\n    flow_run_context = FlowRunContext.get()\n    if flow_run_context:\n        return flow_run_context.task_run_results.get(id(obj))\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.link_state_to_result","title":"link_state_to_result","text":"

    Caches a link between a state and a result and its components using the id of the components to map to the state. The cache is persisted to the current flow run context since task relationships are limited to within a flow run.

    This allows dependency tracking to occur when results are passed around. Note: Because id is used, we cannot cache links between singleton objects.

    We only cache the relationship between components 1-layer deep. Example: Given the result [1, [\"a\",\"b\"], (\"c\",)], the following elements will be mapped to the state: - [1, [\"a\",\"b\"], (\"c\",)] - [\"a\",\"b\"] - (\"c\",)

    Note: the int `1` will not be mapped to the state because it is a singleton.\n

    Other Notes: We do not hash the result because: - If changes are made to the object in the flow between task calls, we can still track that they are related. - Hashing can be expensive. - Not all objects are hashable.

    We do not set an attribute, e.g. __prefect_state__, on the result because:

    • Mutating user's objects is dangerous.
    • Unrelated equality comparisons can break unexpectedly.
    • The field can be preserved on copy.
    • We cannot set this attribute on Python built-ins.
    Source code in prefect/engine.py
    def link_state_to_result(state: State, result: Any) -> None:\n    \"\"\"\n    Caches a link between a state and a result and its components using\n    the `id` of the components to map to the state. The cache is persisted to the\n    current flow run context since task relationships are limited to within a flow run.\n\n    This allows dependency tracking to occur when results are passed around.\n    Note: Because `id` is used, we cannot cache links between singleton objects.\n\n    We only cache the relationship between components 1-layer deep.\n    Example:\n        Given the result [1, [\"a\",\"b\"], (\"c\",)], the following elements will be\n        mapped to the state:\n        - [1, [\"a\",\"b\"], (\"c\",)]\n        - [\"a\",\"b\"]\n        - (\"c\",)\n\n        Note: the int `1` will not be mapped to the state because it is a singleton.\n\n    Other Notes:\n    We do not hash the result because:\n    - If changes are made to the object in the flow between task calls, we can still\n      track that they are related.\n    - Hashing can be expensive.\n    - Not all objects are hashable.\n\n    We do not set an attribute, e.g. `__prefect_state__`, on the result because:\n\n    - Mutating user's objects is dangerous.\n    - Unrelated equality comparisons can break unexpectedly.\n    - The field can be preserved on copy.\n    - We cannot set this attribute on Python built-ins.\n    \"\"\"\n\n    flow_run_context = FlowRunContext.get()\n\n    def link_if_trackable(obj: Any) -> None:\n        \"\"\"Track connection between a task run result and its associated state if it has a unique ID.\n\n        We cannot track booleans, Ellipsis, None, NotImplemented, or the integers from -5 to 256\n        because they are singletons.\n\n        This function will mutate the State if the object is an untrackable type by setting the value\n        for `State.state_details.untrackable_result` to `True`.\n\n        \"\"\"\n        if (type(obj) in UNTRACKABLE_TYPES) or (\n            isinstance(obj, int) and (-5 <= obj <= 256)\n        ):\n            state.state_details.untrackable_result = True\n            return\n        flow_run_context.task_run_results[id(obj)] = state\n\n    if flow_run_context:\n        visit_collection(expr=result, visit_fn=link_if_trackable, max_depth=1)\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.orchestrate_flow_run","title":"orchestrate_flow_run async","text":"

    Executes a flow run.

    Note on flow timeouts

    Since async flows are run directly in the main event loop, timeout behavior will match that described by anyio. If the flow is awaiting something, it will immediately return; otherwise, the next time it awaits it will exit. Sync flows are being task runner in a worker thread, which cannot be interrupted. The worker thread will exit at the next task call. The worker thread also has access to the status of the cancellation scope at FlowRunContext.timeout_scope.cancel_called which allows it to raise a TimeoutError to respect the timeout.

    Returns:

    Type Description State

    The final state of the run

    Source code in prefect/engine.py
    async def orchestrate_flow_run(\n    flow: Flow,\n    flow_run: FlowRun,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    interruptible: bool,\n    client: PrefectClient,\n    partial_flow_run_context: PartialModel[FlowRunContext],\n    user_thread: threading.Thread,\n) -> State:\n    \"\"\"\n    Executes a flow run.\n\n    Note on flow timeouts:\n        Since async flows are run directly in the main event loop, timeout behavior will\n        match that described by anyio. If the flow is awaiting something, it will\n        immediately return; otherwise, the next time it awaits it will exit. Sync flows\n        are being task runner in a worker thread, which cannot be interrupted. The worker\n        thread will exit at the next task call. The worker thread also has access to the\n        status of the cancellation scope at `FlowRunContext.timeout_scope.cancel_called`\n        which allows it to raise a `TimeoutError` to respect the timeout.\n\n    Returns:\n        The final state of the run\n    \"\"\"\n\n    logger = flow_run_logger(flow_run, flow)\n\n    flow_run_context = None\n    parent_flow_run_context = FlowRunContext.get()\n\n    try:\n        # Resolve futures in any non-data dependencies to ensure they are ready\n        if wait_for is not None:\n            await resolve_inputs({\"wait_for\": wait_for}, return_data=False)\n    except UpstreamTaskError as upstream_exc:\n        return await propose_state(\n            client,\n            Pending(name=\"NotReady\", message=str(upstream_exc)),\n            flow_run_id=flow_run.id,\n            # if orchestrating a run already in a pending state, force orchestration to\n            # update the state name\n            force=flow_run.state.is_pending(),\n        )\n\n    state = await propose_state(client, Running(), flow_run_id=flow_run.id)\n\n    # flag to ensure we only update the flow run name once\n    run_name_set = False\n\n    while state.is_running():\n        waited_for_task_runs = False\n\n        # Update the flow run to the latest data\n        flow_run = await client.read_flow_run(flow_run.id)\n        try:\n            with partial_flow_run_context.finalize(\n                flow=flow,\n                flow_run=flow_run,\n                client=client,\n                parameters=parameters,\n            ) as flow_run_context:\n                # update flow run name\n                if not run_name_set and flow.flow_run_name:\n                    flow_run_name = _resolve_custom_flow_run_name(\n                        flow=flow, parameters=parameters\n                    )\n\n                    await client.update_flow_run(\n                        flow_run_id=flow_run.id, name=flow_run_name\n                    )\n                    logger.extra[\"flow_run_name\"] = flow_run_name\n                    logger.debug(\n                        f\"Renamed flow run {flow_run.name!r} to {flow_run_name!r}\"\n                    )\n                    flow_run.name = flow_run_name\n                    run_name_set = True\n\n                args, kwargs = parameters_to_args_kwargs(flow.fn, parameters)\n                logger.debug(\n                    f\"Executing flow {flow.name!r} for flow run {flow_run.name!r}...\"\n                )\n\n                if PREFECT_DEBUG_MODE:\n                    logger.debug(f\"Executing {call_repr(flow.fn, *args, **kwargs)}\")\n                else:\n                    logger.debug(\n                        \"Beginning execution...\", extra={\"state_message\": True}\n                    )\n\n                flow_call = create_call(flow.fn, *args, **kwargs)\n\n                # This check for a parent call is needed for cases where the engine\n                # was entered directly during testing\n                parent_call = get_current_call()\n\n                if parent_call and (\n                    not parent_flow_run_context\n                    or (\n                        parent_flow_run_context\n                        and parent_flow_run_context.flow.isasync == flow.isasync\n                    )\n                ):\n                    from_async.call_soon_in_waiting_thread(\n                        flow_call, thread=user_thread, timeout=flow.timeout_seconds\n                    )\n                else:\n                    from_async.call_soon_in_new_thread(\n                        flow_call, timeout=flow.timeout_seconds\n                    )\n\n                result = await flow_call.aresult()\n\n                waited_for_task_runs = await wait_for_task_runs_and_report_crashes(\n                    flow_run_context.task_run_futures, client=client\n                )\n        except PausedRun as exc:\n            # could get raised either via utility or by returning Paused from a task run\n            # if a task run pauses, we set its state as the flow's state\n            # to preserve reschedule and timeout behavior\n            paused_flow_run = await client.read_flow_run(flow_run.id)\n            if paused_flow_run.state.is_running():\n                state = await propose_state(\n                    client,\n                    state=exc.state,\n                    flow_run_id=flow_run.id,\n                )\n\n                return state\n            paused_flow_run_state = paused_flow_run.state\n            return paused_flow_run_state\n        except CancelledError as exc:\n            if not flow_call.timedout():\n                # If the flow call was not cancelled by us; this is a crash\n                raise\n            # Construct a new exception as `TimeoutError`\n            original = exc\n            exc = TimeoutError()\n            exc.__cause__ = original\n            logger.exception(\"Encountered exception during execution:\")\n            terminal_state = await exception_to_failed_state(\n                exc,\n                message=f\"Flow run exceeded timeout of {flow.timeout_seconds} seconds\",\n                result_factory=flow_run_context.result_factory,\n                name=\"TimedOut\",\n            )\n        except Exception:\n            # Generic exception in user code\n            logger.exception(\"Encountered exception during execution:\")\n            terminal_state = await exception_to_failed_state(\n                message=\"Flow run encountered an exception.\",\n                result_factory=flow_run_context.result_factory,\n            )\n        else:\n            if result is None:\n                # All tasks and subflows are reference tasks if there is no return value\n                # If there are no tasks, use `None` instead of an empty iterable\n                result = (\n                    flow_run_context.task_run_futures\n                    + flow_run_context.task_run_states\n                    + flow_run_context.flow_run_states\n                ) or None\n\n            terminal_state = await return_value_to_state(\n                await resolve_futures_to_states(result),\n                result_factory=flow_run_context.result_factory,\n            )\n\n        if not waited_for_task_runs:\n            # An exception occurred that prevented us from waiting for task runs to\n            # complete. Ensure that we wait for them before proposing a final state\n            # for the flow run.\n            await wait_for_task_runs_and_report_crashes(\n                flow_run_context.task_run_futures, client=client\n            )\n\n        # Before setting the flow run state, store state.data using\n        # block storage and send the resulting data document to the Prefect API instead.\n        # This prevents the pickled return value of flow runs\n        # from being sent to the Prefect API and stored in the Prefect database.\n        # state.data is left as is, otherwise we would have to load\n        # the data from block storage again after storing.\n        state = await propose_state(\n            client,\n            state=terminal_state,\n            flow_run_id=flow_run.id,\n        )\n\n        await _run_flow_hooks(flow=flow, flow_run=flow_run, state=state)\n\n        if state.type != terminal_state.type and PREFECT_DEBUG_MODE:\n            logger.debug(\n                (\n                    f\"Received new state {state} when proposing final state\"\n                    f\" {terminal_state}\"\n                ),\n                extra={\"send_to_api\": False},\n            )\n\n        if not state.is_final() and not state.is_paused():\n            logger.info(\n                (\n                    f\"Received non-final state {state.name!r} when proposing final\"\n                    f\" state {terminal_state.name!r} and will attempt to run again...\"\n                ),\n                extra={\"send_to_api\": False},\n            )\n            # Attempt to enter a running state again\n            state = await propose_state(client, Running(), flow_run_id=flow_run.id)\n\n    return state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.orchestrate_task_run","title":"orchestrate_task_run async","text":"

    Execute a task run

    This function should be submitted to an task runner. We must construct the context here instead of receiving it already populated since we may be in a new environment.

    Proposes a RUNNING state, then - if accepted, the task user function will be run - if rejected, the received state will be returned

    When the user function is run, the result will be used to determine a final state - if an exception is encountered, it is trapped and stored in a FAILED state - otherwise, return_value_to_state is used to determine the state

    If the final state is COMPLETED, we generate a cache key as specified by the task

    The final state is then proposed - if accepted, this is the final state and will be returned - if rejected and a new final state is provided, it will be returned - if rejected and a non-final state is provided, we will attempt to enter a RUNNING state again

    Returns:

    Type Description State

    The final state of the run

    Source code in prefect/engine.py
    async def orchestrate_task_run(\n    task: Task,\n    task_run: TaskRun,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    result_factory: ResultFactory,\n    log_prints: bool,\n    interruptible: bool,\n    client: PrefectClient,\n) -> State:\n    \"\"\"\n    Execute a task run\n\n    This function should be submitted to an task runner. We must construct the context\n    here instead of receiving it already populated since we may be in a new environment.\n\n    Proposes a RUNNING state, then\n    - if accepted, the task user function will be run\n    - if rejected, the received state will be returned\n\n    When the user function is run, the result will be used to determine a final state\n    - if an exception is encountered, it is trapped and stored in a FAILED state\n    - otherwise, `return_value_to_state` is used to determine the state\n\n    If the final state is COMPLETED, we generate a cache key as specified by the task\n\n    The final state is then proposed\n    - if accepted, this is the final state and will be returned\n    - if rejected and a new final state is provided, it will be returned\n    - if rejected and a non-final state is provided, we will attempt to enter a RUNNING\n        state again\n\n    Returns:\n        The final state of the run\n    \"\"\"\n    flow_run_context = prefect.context.FlowRunContext.get()\n    if flow_run_context:\n        flow_run = flow_run_context.flow_run\n    else:\n        flow_run = await client.read_flow_run(task_run.flow_run_id)\n    logger = task_run_logger(task_run, task=task, flow_run=flow_run)\n\n    partial_task_run_context = PartialModel(\n        TaskRunContext,\n        task_run=task_run,\n        task=task,\n        client=client,\n        result_factory=result_factory,\n        log_prints=log_prints,\n    )\n    task_introspection_start_time = time.perf_counter()\n    try:\n        # Resolve futures in parameters into data\n        resolved_parameters = await resolve_inputs(parameters)\n        # Resolve futures in any non-data dependencies to ensure they are ready\n        await resolve_inputs({\"wait_for\": wait_for}, return_data=False)\n    except UpstreamTaskError as upstream_exc:\n        return await propose_state(\n            client,\n            Pending(name=\"NotReady\", message=str(upstream_exc)),\n            task_run_id=task_run.id,\n            # if orchestrating a run already in a pending state, force orchestration to\n            # update the state name\n            force=task_run.state.is_pending(),\n        )\n    task_introspection_end_time = time.perf_counter()\n\n    introspection_time = round(\n        task_introspection_end_time - task_introspection_start_time, 3\n    )\n    threshold = PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD.value()\n    if threshold and introspection_time > threshold:\n        logger.warning(\n            f\"Task parameter introspection took {introspection_time} seconds \"\n            f\", exceeding `PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD` of {threshold}. \"\n            \"Try wrapping large task parameters with \"\n            \"`prefect.utilities.annotations.quote` for increased performance, \"\n            \"e.g. `my_task(quote(param))`. To disable this message set \"\n            \"`PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD=0`.\"\n        )\n\n    # Generate the cache key to attach to proposed states\n    # The cache key uses a TaskRunContext that does not include a `timeout_context``\n    cache_key = (\n        task.cache_key_fn(\n            partial_task_run_context.finalize(parameters=resolved_parameters),\n            resolved_parameters,\n        )\n        if task.cache_key_fn\n        else None\n    )\n\n    task_run_context = partial_task_run_context.finalize(parameters=resolved_parameters)\n\n    # Ignore the cached results for a cache key, default = false\n    # Setting on task level overrules the Prefect setting (env var)\n    refresh_cache = (\n        task.refresh_cache\n        if task.refresh_cache is not None\n        else PREFECT_TASKS_REFRESH_CACHE.value()\n    )\n\n    # Emit an event to capture that the task run was in the `PENDING` state.\n    last_event = _emit_task_run_state_change_event(\n        task_run=task_run, initial_state=None, validated_state=task_run.state\n    )\n    last_state = task_run.state\n\n    # Transition from `PENDING` -> `RUNNING`\n    try:\n        state = await propose_state(\n            client,\n            Running(\n                state_details=StateDetails(\n                    cache_key=cache_key, refresh_cache=refresh_cache\n                )\n            ),\n            task_run_id=task_run.id,\n        )\n    except Pause as exc:\n        # We shouldn't get a pause signal without a state, but if this happens,\n        # just use a Paused state to assume an in-process pause.\n        state = exc.state if exc.state else Paused()\n\n        # If a flow submits tasks and then pauses, we may reach this point due\n        # to concurrency timing because the tasks will try to transition after\n        # the flow run has paused. Orchestration will send back a Paused state\n        # for the task runs.\n        if state.state_details.pause_reschedule:\n            # If we're being asked to pause and reschedule, we should exit the\n            # task and expect to be resumed later.\n            raise\n\n    if state.is_paused():\n        BACKOFF_MAX = 10  # Seconds\n        backoff_count = 0\n\n        async def tick():\n            nonlocal backoff_count\n            if backoff_count < BACKOFF_MAX:\n                backoff_count += 1\n            interval = 1 + backoff_count + random.random() * backoff_count\n            await anyio.sleep(interval)\n\n        # Enter a loop to wait for the task run to be resumed, i.e.\n        # become Pending, and then propose a Running state again.\n        while True:\n            await tick()\n\n            # Propose a Running state again. We do this instead of reading the\n            # task run because if the flow run times out, this lets\n            # orchestration fail the task run.\n            try:\n                state = await propose_state(\n                    client,\n                    Running(\n                        state_details=StateDetails(\n                            cache_key=cache_key, refresh_cache=refresh_cache\n                        )\n                    ),\n                    task_run_id=task_run.id,\n                )\n            except Pause as exc:\n                if not exc.state:\n                    continue\n\n                if exc.state.state_details.pause_reschedule:\n                    # If the pause state includes pause_reschedule, we should exit the\n                    # task and expect to be resumed later. We've already checked for this\n                    # above, but we check again here in case the state changed; e.g. the\n                    # flow run suspended.\n                    raise\n                else:\n                    # Propose a Running state again.\n                    continue\n            else:\n                break\n\n    # Emit an event to capture the result of proposing a `RUNNING` state.\n    last_event = _emit_task_run_state_change_event(\n        task_run=task_run,\n        initial_state=last_state,\n        validated_state=state,\n        follows=last_event,\n    )\n    last_state = state\n\n    # flag to ensure we only update the task run name once\n    run_name_set = False\n\n    # Only run the task if we enter a `RUNNING` state\n    while state.is_running():\n        # Retrieve the latest metadata for the task run context\n        task_run = await client.read_task_run(task_run.id)\n\n        with task_run_context.copy(\n            update={\"task_run\": task_run, \"start_time\": pendulum.now(\"UTC\")}\n        ):\n            try:\n                args, kwargs = parameters_to_args_kwargs(task.fn, resolved_parameters)\n                # update task run name\n                if not run_name_set and task.task_run_name:\n                    task_run_name = _resolve_custom_task_run_name(\n                        task=task, parameters=resolved_parameters\n                    )\n                    await client.set_task_run_name(\n                        task_run_id=task_run.id, name=task_run_name\n                    )\n                    logger.extra[\"task_run_name\"] = task_run_name\n                    logger.debug(\n                        f\"Renamed task run {task_run.name!r} to {task_run_name!r}\"\n                    )\n                    task_run.name = task_run_name\n                    run_name_set = True\n\n                if PREFECT_DEBUG_MODE.value():\n                    logger.debug(f\"Executing {call_repr(task.fn, *args, **kwargs)}\")\n                else:\n                    logger.debug(\n                        \"Beginning execution...\", extra={\"state_message\": True}\n                    )\n\n                call = from_async.call_soon_in_new_thread(\n                    create_call(task.fn, *args, **kwargs), timeout=task.timeout_seconds\n                )\n                result = await call.aresult()\n\n            except (CancelledError, asyncio.CancelledError) as exc:\n                if not call.timedout():\n                    # If the task call was not cancelled by us; this is a crash\n                    raise\n                # Construct a new exception as `TimeoutError`\n                original = exc\n                exc = TimeoutError()\n                exc.__cause__ = original\n                logger.exception(\"Encountered exception during execution:\")\n                terminal_state = await exception_to_failed_state(\n                    exc,\n                    message=(\n                        f\"Task run exceeded timeout of {task.timeout_seconds} seconds\"\n                    ),\n                    result_factory=task_run_context.result_factory,\n                    name=\"TimedOut\",\n                )\n            except Exception as exc:\n                logger.exception(\"Encountered exception during execution:\")\n                terminal_state = await exception_to_failed_state(\n                    exc,\n                    message=\"Task run encountered an exception\",\n                    result_factory=task_run_context.result_factory,\n                )\n            else:\n                terminal_state = await return_value_to_state(\n                    result,\n                    result_factory=task_run_context.result_factory,\n                )\n\n                # for COMPLETED tasks, add the cache key and expiration\n                if terminal_state.is_completed():\n                    terminal_state.state_details.cache_expiration = (\n                        (pendulum.now(\"utc\") + task.cache_expiration)\n                        if task.cache_expiration\n                        else None\n                    )\n                    terminal_state.state_details.cache_key = cache_key\n\n            if terminal_state.is_failed():\n                # Defer to user to decide whether failure is retriable\n                terminal_state.state_details.retriable = (\n                    await _check_task_failure_retriable(task, task_run, terminal_state)\n                )\n            state = await propose_state(client, terminal_state, task_run_id=task_run.id)\n\n            last_event = _emit_task_run_state_change_event(\n                task_run=task_run,\n                initial_state=last_state,\n                validated_state=state,\n                follows=last_event,\n            )\n            last_state = state\n\n            await _run_task_hooks(\n                task=task,\n                task_run=task_run,\n                state=state,\n            )\n\n            if state.type != terminal_state.type and PREFECT_DEBUG_MODE:\n                logger.debug(\n                    (\n                        f\"Received new state {state} when proposing final state\"\n                        f\" {terminal_state}\"\n                    ),\n                    extra={\"send_to_api\": False},\n                )\n\n            if not state.is_final() and not state.is_paused():\n                logger.info(\n                    (\n                        f\"Received non-final state {state.name!r} when proposing final\"\n                        f\" state {terminal_state.name!r} and will attempt to run\"\n                        \" again...\"\n                    ),\n                    extra={\"send_to_api\": False},\n                )\n                # Attempt to enter a running state again\n                state = await propose_state(client, Running(), task_run_id=task_run.id)\n                last_event = _emit_task_run_state_change_event(\n                    task_run=task_run,\n                    initial_state=last_state,\n                    validated_state=state,\n                    follows=last_event,\n                )\n                last_state = state\n\n    # If debugging, use the more complete `repr` than the usual `str` description\n    display_state = repr(state) if PREFECT_DEBUG_MODE else str(state)\n\n    logger.log(\n        level=logging.INFO if state.is_completed() else logging.ERROR,\n        msg=f\"Finished in state {display_state}\",\n    )\n\n    return state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.pause_flow_run","title":"pause_flow_run async","text":"

    Pauses the current flow run by blocking execution until resumed.

    When called within a flow run, execution will block and no downstream tasks will run until the flow is resumed. Task runs that have already started will continue running. A timeout parameter can be passed that will fail the flow run if it has not been resumed within the specified time.

    Parameters:

    Name Type Description Default flow_run_id UUID

    a flow run id. If supplied, this function will attempt to pause the specified flow run outside of the flow run process. When paused, the flow run will continue execution until the NEXT task is orchestrated, at which point the flow will exit. Any tasks that have already started will run until completion. When resumed, the flow run will be rescheduled to finish execution. In order pause a flow run in this way, the flow needs to have an associated deployment and results need to be configured with the persist_results option.

    None timeout int

    the number of seconds to wait for the flow to be resumed before failing. Defaults to 1 hour (3600 seconds). If the pause timeout exceeds any configured flow-level timeout, the flow might fail even after resuming.

    3600 poll_interval int

    The number of seconds between checking whether the flow has been resumed. Defaults to 10 seconds.

    10 reschedule bool

    Flag that will reschedule the flow run if resumed. Instead of blocking execution, the flow will gracefully exit (with no result returned) instead. To use this flag, a flow needs to have an associated deployment and results need to be configured with the persist_results option.

    False key str

    An optional key to prevent calling pauses more than once. This defaults to the number of pauses observed by the flow so far, and prevents pauses that use the \"reschedule\" option from running the same pause twice. A custom key can be supplied for custom pausing behavior.

    None wait_for_input Optional[Type[T]]

    a subclass of RunInput. If provided when the flow pauses, the flow will wait for the input to be provided before resuming. If the flow is resumed without providing the input, the flow will fail. If the flow is resumed with the input, the flow will resume and the input will be loaded and returned from this function.

    None
    @task\ndef task_one():\n    for i in range(3):\n        sleep(1)\n\n@flow\ndef my_flow():\n    terminal_state = task_one.submit(return_state=True)\n    if terminal_state.type == StateType.COMPLETED:\n        print(\"Task one succeeded! Pausing flow run..\")\n        pause_flow_run(timeout=2)\n    else:\n        print(\"Task one failed. Skipping pause flow run..\")\n
    Source code in prefect/engine.py
    @sync_compatible\n@deprecated_parameter(\n    \"flow_run_id\", start_date=\"Dec 2023\", help=\"Use `suspend_flow_run` instead.\"\n)\n@deprecated_parameter(\n    \"reschedule\",\n    start_date=\"Dec 2023\",\n    when=lambda p: p is True,\n    help=\"Use `suspend_flow_run` instead.\",\n)\n@experimental_parameter(\n    \"wait_for_input\", group=\"flow_run_input\", when=lambda y: y is not None\n)\nasync def pause_flow_run(\n    wait_for_input: Optional[Type[T]] = None,\n    flow_run_id: UUID = None,\n    timeout: int = 3600,\n    poll_interval: int = 10,\n    reschedule: bool = False,\n    key: str = None,\n):\n    \"\"\"\n    Pauses the current flow run by blocking execution until resumed.\n\n    When called within a flow run, execution will block and no downstream tasks will\n    run until the flow is resumed. Task runs that have already started will continue\n    running. A timeout parameter can be passed that will fail the flow run if it has not\n    been resumed within the specified time.\n\n    Args:\n        flow_run_id: a flow run id. If supplied, this function will attempt to pause\n            the specified flow run outside of the flow run process. When paused, the\n            flow run will continue execution until the NEXT task is orchestrated, at\n            which point the flow will exit. Any tasks that have already started will\n            run until completion. When resumed, the flow run will be rescheduled to\n            finish execution. In order pause a flow run in this way, the flow needs to\n            have an associated deployment and results need to be configured with the\n            `persist_results` option.\n        timeout: the number of seconds to wait for the flow to be resumed before\n            failing. Defaults to 1 hour (3600 seconds). If the pause timeout exceeds\n            any configured flow-level timeout, the flow might fail even after resuming.\n        poll_interval: The number of seconds between checking whether the flow has been\n            resumed. Defaults to 10 seconds.\n        reschedule: Flag that will reschedule the flow run if resumed. Instead of\n            blocking execution, the flow will gracefully exit (with no result returned)\n            instead. To use this flag, a flow needs to have an associated deployment and\n            results need to be configured with the `persist_results` option.\n        key: An optional key to prevent calling pauses more than once. This defaults to\n            the number of pauses observed by the flow so far, and prevents pauses that\n            use the \"reschedule\" option from running the same pause twice. A custom key\n            can be supplied for custom pausing behavior.\n        wait_for_input: a subclass of `RunInput`. If provided when the flow pauses, the\n            flow will wait for the input to be provided before resuming. If the flow is\n            resumed without providing the input, the flow will fail. If the flow is\n            resumed with the input, the flow will resume and the input will be loaded\n            and returned from this function.\n\n    Example:\n    ```python\n    @task\n    def task_one():\n        for i in range(3):\n            sleep(1)\n\n    @flow\n    def my_flow():\n        terminal_state = task_one.submit(return_state=True)\n        if terminal_state.type == StateType.COMPLETED:\n            print(\"Task one succeeded! Pausing flow run..\")\n            pause_flow_run(timeout=2)\n        else:\n            print(\"Task one failed. Skipping pause flow run..\")\n    ```\n\n    \"\"\"\n    if flow_run_id:\n        if wait_for_input is not None:\n            raise RuntimeError(\"Cannot wait for input when pausing out of process.\")\n\n        return await _out_of_process_pause(\n            flow_run_id=flow_run_id,\n            timeout=timeout,\n            reschedule=reschedule,\n            key=key,\n        )\n    else:\n        return await _in_process_pause(\n            timeout=timeout,\n            poll_interval=poll_interval,\n            reschedule=reschedule,\n            key=key,\n            wait_for_input=wait_for_input,\n        )\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.propose_state","title":"propose_state async","text":"

    Propose a new state for a flow run or task run, invoking Prefect orchestration logic.

    If the proposed state is accepted, the provided state will be augmented with details and returned.

    If the proposed state is rejected, a new state returned by the Prefect API will be returned.

    If the proposed state results in a WAIT instruction from the Prefect API, the function will sleep and attempt to propose the state again.

    If the proposed state results in an ABORT instruction from the Prefect API, an error will be raised.

    Parameters:

    Name Type Description Default state State

    a new state for the task or flow run

    required task_run_id UUID

    an optional task run id, used when proposing task run states

    None flow_run_id UUID

    an optional flow run id, used when proposing flow run states

    None

    Returns:

    Type Description State

    a State model representation of the flow or task run state

    Raises:

    Type Description ValueError

    if neither task_run_id or flow_run_id is provided

    Abort

    if an ABORT instruction is received from the Prefect API

    Source code in prefect/engine.py
    async def propose_state(\n    client: PrefectClient,\n    state: State,\n    force: bool = False,\n    task_run_id: UUID = None,\n    flow_run_id: UUID = None,\n) -> State:\n    \"\"\"\n    Propose a new state for a flow run or task run, invoking Prefect orchestration logic.\n\n    If the proposed state is accepted, the provided `state` will be augmented with\n     details and returned.\n\n    If the proposed state is rejected, a new state returned by the Prefect API will be\n    returned.\n\n    If the proposed state results in a WAIT instruction from the Prefect API, the\n    function will sleep and attempt to propose the state again.\n\n    If the proposed state results in an ABORT instruction from the Prefect API, an\n    error will be raised.\n\n    Args:\n        state: a new state for the task or flow run\n        task_run_id: an optional task run id, used when proposing task run states\n        flow_run_id: an optional flow run id, used when proposing flow run states\n\n    Returns:\n        a [State model][prefect.client.schemas.objects.State] representation of the\n            flow or task run state\n\n    Raises:\n        ValueError: if neither task_run_id or flow_run_id is provided\n        prefect.exceptions.Abort: if an ABORT instruction is received from\n            the Prefect API\n    \"\"\"\n\n    # Determine if working with a task run or flow run\n    if not task_run_id and not flow_run_id:\n        raise ValueError(\"You must provide either a `task_run_id` or `flow_run_id`\")\n\n    # Handle task and sub-flow tracing\n    if state.is_final():\n        if isinstance(state.data, BaseResult) and state.data.has_cached_object():\n            # Avoid fetching the result unless it is cached, otherwise we defeat\n            # the purpose of disabling `cache_result_in_memory`\n            result = await state.result(raise_on_failure=False, fetch=True)\n        else:\n            result = state.data\n\n        link_state_to_result(state, result)\n\n    # Handle repeated WAITs in a loop instead of recursively, to avoid\n    # reaching max recursion depth in extreme cases.\n    async def set_state_and_handle_waits(set_state_func) -> OrchestrationResult:\n        response = await set_state_func()\n        while response.status == SetStateStatus.WAIT:\n            engine_logger.debug(\n                f\"Received wait instruction for {response.details.delay_seconds}s: \"\n                f\"{response.details.reason}\"\n            )\n            await anyio.sleep(response.details.delay_seconds)\n            response = await set_state_func()\n        return response\n\n    # Attempt to set the state\n    if task_run_id:\n        set_state = partial(client.set_task_run_state, task_run_id, state, force=force)\n        response = await set_state_and_handle_waits(set_state)\n    elif flow_run_id:\n        set_state = partial(client.set_flow_run_state, flow_run_id, state, force=force)\n        response = await set_state_and_handle_waits(set_state)\n    else:\n        raise ValueError(\n            \"Neither flow run id or task run id were provided. At least one must \"\n            \"be given.\"\n        )\n\n    # Parse the response to return the new state\n    if response.status == SetStateStatus.ACCEPT:\n        # Update the state with the details if provided\n        state.id = response.state.id\n        state.timestamp = response.state.timestamp\n        if response.state.state_details:\n            state.state_details = response.state.state_details\n        return state\n\n    elif response.status == SetStateStatus.ABORT:\n        raise prefect.exceptions.Abort(response.details.reason)\n\n    elif response.status == SetStateStatus.REJECT:\n        if response.state.is_paused():\n            raise Pause(response.details.reason, state=response.state)\n        return response.state\n\n    else:\n        raise ValueError(\n            f\"Received unexpected `SetStateStatus` from server: {response.status!r}\"\n        )\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.report_flow_run_crashes","title":"report_flow_run_crashes async","text":"

    Detect flow run crashes during this context and update the run to a proper final state.

    This context must reraise the exception to properly exit the run.

    Source code in prefect/engine.py
    @asynccontextmanager\nasync def report_flow_run_crashes(flow_run: FlowRun, client: PrefectClient, flow: Flow):\n    \"\"\"\n    Detect flow run crashes during this context and update the run to a proper final\n    state.\n\n    This context _must_ reraise the exception to properly exit the run.\n    \"\"\"\n\n    try:\n        yield\n    except (Abort, Pause):\n        # Do not capture internal signals as crashes\n        raise\n    except BaseException as exc:\n        state = await exception_to_crashed_state(exc)\n        logger = flow_run_logger(flow_run)\n        with anyio.CancelScope(shield=True):\n            logger.error(f\"Crash detected! {state.message}\")\n            logger.debug(\"Crash details:\", exc_info=exc)\n            flow_run_state = await propose_state(client, state, flow_run_id=flow_run.id)\n            engine_logger.debug(\n                f\"Reported crashed flow run {flow_run.name!r} successfully!\"\n            )\n\n            # Only `on_crashed` and `on_cancellation` flow run state change hooks can be called here.\n            # We call the hooks after the state change proposal to `CRASHED` is validated\n            # or rejected (if it is in a `CANCELLING` state).\n            await _run_flow_hooks(\n                flow=flow,\n                flow_run=flow_run,\n                state=flow_run_state,\n            )\n\n        # Reraise the exception\n        raise\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.report_task_run_crashes","title":"report_task_run_crashes async","text":"

    Detect task run crashes during this context and update the run to a proper final state.

    This context must reraise the exception to properly exit the run.

    Source code in prefect/engine.py
    @asynccontextmanager\nasync def report_task_run_crashes(task_run: TaskRun, client: PrefectClient):\n    \"\"\"\n    Detect task run crashes during this context and update the run to a proper final\n    state.\n\n    This context _must_ reraise the exception to properly exit the run.\n    \"\"\"\n    try:\n        yield\n    except (Abort, Pause):\n        # Do not capture internal signals as crashes\n        raise\n    except BaseException as exc:\n        state = await exception_to_crashed_state(exc)\n        logger = task_run_logger(task_run)\n        with anyio.CancelScope(shield=True):\n            logger.error(f\"Crash detected! {state.message}\")\n            logger.debug(\"Crash details:\", exc_info=exc)\n            await client.set_task_run_state(\n                state=state,\n                task_run_id=task_run.id,\n                force=True,\n            )\n            engine_logger.debug(\n                f\"Reported crashed task run {task_run.name!r} successfully!\"\n            )\n\n        # Reraise the exception\n        raise\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.resolve_inputs","title":"resolve_inputs async","text":"

    Resolve any Quote, PrefectFuture, or State types nested in parameters into data.

    Returns:

    Type Description Dict[str, Any]

    A copy of the parameters with resolved data

    Raises:

    Type Description UpstreamTaskError

    If any of the upstream states are not COMPLETED

    Source code in prefect/engine.py
    async def resolve_inputs(\n    parameters: Dict[str, Any], return_data: bool = True, max_depth: int = -1\n) -> Dict[str, Any]:\n    \"\"\"\n    Resolve any `Quote`, `PrefectFuture`, or `State` types nested in parameters into\n    data.\n\n    Returns:\n        A copy of the parameters with resolved data\n\n    Raises:\n        UpstreamTaskError: If any of the upstream states are not `COMPLETED`\n    \"\"\"\n\n    futures = set()\n    states = set()\n    result_by_state = {}\n\n    if not parameters:\n        return {}\n\n    def collect_futures_and_states(expr, context):\n        # Expressions inside quotes should not be traversed\n        if isinstance(context.get(\"annotation\"), quote):\n            raise StopVisiting()\n\n        if isinstance(expr, PrefectFuture):\n            futures.add(expr)\n        if is_state(expr):\n            states.add(expr)\n\n        return expr\n\n    visit_collection(\n        parameters,\n        visit_fn=collect_futures_and_states,\n        return_data=False,\n        max_depth=max_depth,\n        context={},\n    )\n\n    # Wait for all futures so we do not block when we retrieve the state in `resolve_input`\n    states.update(await asyncio.gather(*[future._wait() for future in futures]))\n\n    # Only retrieve the result if requested as it may be expensive\n    if return_data:\n        finished_states = [state for state in states if state.is_final()]\n\n        state_results = await asyncio.gather(\n            *[\n                state.result(raise_on_failure=False, fetch=True)\n                for state in finished_states\n            ]\n        )\n\n        for state, result in zip(finished_states, state_results):\n            result_by_state[state] = result\n\n    def resolve_input(expr, context):\n        state = None\n\n        # Expressions inside quotes should not be modified\n        if isinstance(context.get(\"annotation\"), quote):\n            raise StopVisiting()\n\n        if isinstance(expr, PrefectFuture):\n            state = expr._final_state\n        elif is_state(expr):\n            state = expr\n        else:\n            return expr\n\n        # Do not allow uncompleted upstreams except failures when `allow_failure` has\n        # been used\n        if not state.is_completed() and not (\n            # TODO: Note that the contextual annotation here is only at the current level\n            #       if `allow_failure` is used then another annotation is used, this will\n            #       incorrectly evaluate to false \u2014 to resolve this, we must track all\n            #       annotations wrapping the current expression but this is not yet\n            #       implemented.\n            isinstance(context.get(\"annotation\"), allow_failure)\n            and state.is_failed()\n        ):\n            raise UpstreamTaskError(\n                f\"Upstream task run '{state.state_details.task_run_id}' did not reach a\"\n                \" 'COMPLETED' state.\"\n            )\n\n        return result_by_state.get(state)\n\n    resolved_parameters = {}\n    for parameter, value in parameters.items():\n        try:\n            resolved_parameters[parameter] = visit_collection(\n                value,\n                visit_fn=resolve_input,\n                return_data=return_data,\n                # we're manually going 1 layer deeper here\n                max_depth=max_depth - 1,\n                remove_annotations=True,\n                context={},\n            )\n        except UpstreamTaskError:\n            raise\n        except Exception as exc:\n            raise PrefectException(\n                f\"Failed to resolve inputs in parameter {parameter!r}. If your\"\n                \" parameter type is not supported, consider using the `quote`\"\n                \" annotation to skip resolution of inputs.\"\n            ) from exc\n\n    return resolved_parameters\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.resume_flow_run","title":"resume_flow_run async","text":"

    Resumes a paused flow.

    Parameters:

    Name Type Description Default flow_run_id

    the flow_run_id to resume

    required run_input Optional[Dict]

    a dictionary of inputs to provide to the flow run.

    None Source code in prefect/engine.py
    @sync_compatible\nasync def resume_flow_run(flow_run_id, run_input: Optional[Dict] = None):\n    \"\"\"\n    Resumes a paused flow.\n\n    Args:\n        flow_run_id: the flow_run_id to resume\n        run_input: a dictionary of inputs to provide to the flow run.\n    \"\"\"\n    client = get_client()\n    flow_run = await client.read_flow_run(flow_run_id)\n\n    if not flow_run.state.is_paused():\n        raise NotPausedError(\"Cannot resume a run that isn't paused!\")\n\n    response = await client.resume_flow_run(flow_run_id, run_input=run_input)\n\n    if response.status == SetStateStatus.REJECT:\n        if response.state.type == StateType.FAILED:\n            raise FlowPauseTimeout(\"Flow run can no longer be resumed.\")\n        else:\n            raise RuntimeError(f\"Cannot resume this run: {response.details.reason}\")\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.retrieve_flow_then_begin_flow_run","title":"retrieve_flow_then_begin_flow_run async","text":"

    Async entrypoint for flow runs that have been submitted for execution by an agent

    • Retrieves the deployment information
    • Loads the flow object using deployment information
    • Updates the flow run version
    Source code in prefect/engine.py
    @inject_client\nasync def retrieve_flow_then_begin_flow_run(\n    flow_run_id: UUID,\n    client: PrefectClient,\n    user_thread: threading.Thread,\n) -> State:\n    \"\"\"\n    Async entrypoint for flow runs that have been submitted for execution by an agent\n\n    - Retrieves the deployment information\n    - Loads the flow object using deployment information\n    - Updates the flow run version\n    \"\"\"\n    flow_run = await client.read_flow_run(flow_run_id)\n    try:\n        flow = await load_flow_from_flow_run(flow_run, client=client)\n    except Exception:\n        message = \"Flow could not be retrieved from deployment.\"\n        flow_run_logger(flow_run).exception(message)\n        state = await exception_to_failed_state(message=message)\n        await client.set_flow_run_state(\n            state=state, flow_run_id=flow_run_id, force=True\n        )\n        return state\n\n    # Update the flow run policy defaults to match settings on the flow\n    # Note: Mutating the flow run object prevents us from performing another read\n    #       operation if these properties are used by the client downstream\n    if flow_run.empirical_policy.retry_delay is None:\n        flow_run.empirical_policy.retry_delay = flow.retry_delay_seconds\n\n    if flow_run.empirical_policy.retries is None:\n        flow_run.empirical_policy.retries = flow.retries\n\n    await client.update_flow_run(\n        flow_run_id=flow_run_id,\n        flow_version=flow.version,\n        empirical_policy=flow_run.empirical_policy,\n    )\n\n    if flow.should_validate_parameters:\n        failed_state = None\n        try:\n            parameters = flow.validate_parameters(flow_run.parameters)\n        except Exception:\n            message = \"Validation of flow parameters failed with error: \"\n            flow_run_logger(flow_run).exception(message)\n            failed_state = await exception_to_failed_state(message=message)\n\n        if failed_state is not None:\n            await propose_state(\n                client,\n                state=failed_state,\n                flow_run_id=flow_run_id,\n            )\n            return failed_state\n    else:\n        parameters = flow_run.parameters\n\n    # Ensure default values are populated\n    parameters = {**get_parameter_defaults(flow.fn), **parameters}\n\n    return await begin_flow_run(\n        flow=flow,\n        flow_run=flow_run,\n        parameters=parameters,\n        client=client,\n        user_thread=user_thread,\n    )\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.suspend_flow_run","title":"suspend_flow_run async","text":"

    Suspends a flow run by stopping code execution until resumed.

    When suspended, the flow run will continue execution until the NEXT task is orchestrated, at which point the flow will exit. Any tasks that have already started will run until completion. When resumed, the flow run will be rescheduled to finish execution. In order suspend a flow run in this way, the flow needs to have an associated deployment and results need to be configured with the persist_results option.

    Parameters:

    Name Type Description Default flow_run_id Optional[UUID]

    a flow run id. If supplied, this function will attempt to suspend the specified flow run. If not supplied will attempt to suspend the current flow run.

    None timeout Optional[int]

    the number of seconds to wait for the flow to be resumed before failing. Defaults to 1 hour (3600 seconds). If the pause timeout exceeds any configured flow-level timeout, the flow might fail even after resuming.

    3600 key Optional[str]

    An optional key to prevent calling suspend more than once. This defaults to a random string and prevents suspends from running the same suspend twice. A custom key can be supplied for custom suspending behavior.

    None wait_for_input Optional[Type[T]]

    a subclass of RunInput. If provided when the flow suspends, the flow will wait for the input to be provided before resuming. If the flow is resumed without providing the input, the flow will fail. If the flow is resumed with the input, the flow will resume and the input will be loaded and returned from this function.

    None Source code in prefect/engine.py
    @sync_compatible\n@inject_client\n@experimental_parameter(\n    \"wait_for_input\", group=\"flow_run_input\", when=lambda y: y is not None\n)\nasync def suspend_flow_run(\n    wait_for_input: Optional[Type[T]] = None,\n    flow_run_id: Optional[UUID] = None,\n    timeout: Optional[int] = 3600,\n    key: Optional[str] = None,\n    client: PrefectClient = None,\n):\n    \"\"\"\n    Suspends a flow run by stopping code execution until resumed.\n\n    When suspended, the flow run will continue execution until the NEXT task is\n    orchestrated, at which point the flow will exit. Any tasks that have\n    already started will run until completion. When resumed, the flow run will\n    be rescheduled to finish execution. In order suspend a flow run in this\n    way, the flow needs to have an associated deployment and results need to be\n    configured with the `persist_results` option.\n\n    Args:\n        flow_run_id: a flow run id. If supplied, this function will attempt to\n            suspend the specified flow run. If not supplied will attempt to\n            suspend the current flow run.\n        timeout: the number of seconds to wait for the flow to be resumed before\n            failing. Defaults to 1 hour (3600 seconds). If the pause timeout\n            exceeds any configured flow-level timeout, the flow might fail even\n            after resuming.\n        key: An optional key to prevent calling suspend more than once. This\n            defaults to a random string and prevents suspends from running the\n            same suspend twice. A custom key can be supplied for custom\n            suspending behavior.\n        wait_for_input: a subclass of `RunInput`. If provided when the flow\n            suspends, the flow will wait for the input to be provided before\n            resuming. If the flow is resumed without providing the input, the\n            flow will fail. If the flow is resumed with the input, the flow\n            will resume and the input will be loaded and returned from this\n            function.\n    \"\"\"\n    context = FlowRunContext.get()\n\n    if flow_run_id is None:\n        if TaskRunContext.get():\n            raise RuntimeError(\"Cannot suspend task runs.\")\n\n        if context is None or context.flow_run is None:\n            raise RuntimeError(\n                \"Flow runs can only be suspended from within a flow run.\"\n            )\n\n        logger = get_run_logger(context=context)\n        logger.info(\n            \"Suspending flow run, execution will be rescheduled when this flow run is\"\n            \" resumed.\"\n        )\n        flow_run_id = context.flow_run.id\n        suspending_current_flow_run = True\n        pause_counter = _observed_flow_pauses(context)\n        pause_key = key or str(pause_counter)\n    else:\n        # Since we're suspending another flow run we need to generate a pause\n        # key that won't conflict with whatever suspends/pauses that flow may\n        # have. Since this method won't be called during that flow run it's\n        # okay that this is non-deterministic.\n        suspending_current_flow_run = False\n        pause_key = key or str(uuid4())\n\n    proposed_state = Suspended(timeout_seconds=timeout, pause_key=pause_key)\n\n    if wait_for_input:\n        run_input_keyset = keyset_from_paused_state(proposed_state)\n        proposed_state.state_details.run_input_keyset = run_input_keyset\n\n    try:\n        state = await propose_state(\n            client=client,\n            state=proposed_state,\n            flow_run_id=flow_run_id,\n        )\n    except Abort as exc:\n        # Aborted requests mean the suspension is not allowed\n        raise RuntimeError(f\"Flow run cannot be suspended: {exc}\")\n\n    if state.is_running():\n        # The orchestrator rejected the suspended state which means that this\n        # suspend has happened before and the flow run has been resumed.\n        if wait_for_input:\n            # The flow run wanted input, so we need to load it and return it\n            # to the user.\n            return await wait_for_input.load(run_input_keyset)\n        return\n\n    if not state.is_paused():\n        # If we receive anything but a PAUSED state, we are unable to continue\n        raise RuntimeError(\n            f\"Flow run cannot be suspended. Received unexpected state from API: {state}\"\n        )\n\n    if wait_for_input:\n        await wait_for_input.save(run_input_keyset)\n\n    if suspending_current_flow_run:\n        # Exit this process so the run can be resubmitted later\n        raise Pause()\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/events/","title":"prefect.events","text":"","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events","title":"prefect.events","text":"","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events.Event","title":"Event","text":"

    Bases: PrefectBaseModel

    The client-side view of an event that has happened to a Resource

    Source code in prefect/events/schemas.py
    class Event(PrefectBaseModel):\n    \"\"\"The client-side view of an event that has happened to a Resource\"\"\"\n\n    occurred: DateTimeTZ = Field(\n        default_factory=pendulum.now,\n        description=\"When the event happened from the sender's perspective\",\n    )\n    event: str = Field(\n        description=\"The name of the event that happened\",\n    )\n    resource: Resource = Field(\n        description=\"The primary Resource this event concerns\",\n    )\n    related: List[RelatedResource] = Field(\n        default_factory=list,\n        description=\"A list of additional Resources involved in this event\",\n    )\n    payload: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"An open-ended set of data describing what happened\",\n    )\n    id: UUID = Field(\n        default_factory=uuid4,\n        description=\"The client-provided identifier of this event\",\n    )\n    follows: Optional[UUID] = Field(\n        None,\n        description=(\n            \"The ID of an event that is known to have occurred prior to this \"\n            \"one. If set, this may be used to establish a more precise \"\n            \"ordering of causally-related events when they occur close enough \"\n            \"together in time that the system may receive them out-of-order.\"\n        ),\n    )\n\n    @property\n    def involved_resources(self) -> Iterable[Resource]:\n        return [self.resource] + list(self.related)\n\n    @validator(\"related\")\n    def enforce_maximum_related_resources(cls, value: List[RelatedResource]):\n        if len(value) > MAXIMUM_RELATED_RESOURCES:\n            raise ValueError(\n                \"The maximum number of related resources \"\n                f\"is {MAXIMUM_RELATED_RESOURCES}\"\n            )\n\n        return value\n
    ","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events.RelatedResource","title":"RelatedResource","text":"

    Bases: Resource

    A Resource with a specific role in an Event

    Source code in prefect/events/schemas.py
    class RelatedResource(Resource):\n    \"\"\"A Resource with a specific role in an Event\"\"\"\n\n    @root_validator(pre=True)\n    def requires_resource_role(cls, values: Dict[str, Any]):\n        labels = values.get(\"__root__\")\n        if not isinstance(labels, dict):\n            return values\n\n        labels = cast(Dict[str, str], labels)\n\n        if \"prefect.resource.role\" not in labels:\n            raise ValueError(\n                \"Related Resources must include the prefect.resource.role label\"\n            )\n        if not labels[\"prefect.resource.role\"]:\n            raise ValueError(\"The prefect.resource.role label must be non-empty\")\n\n        return values\n\n    @property\n    def role(self) -> str:\n        return self[\"prefect.resource.role\"]\n
    ","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events.Resource","title":"Resource","text":"

    Bases: Labelled

    An observable business object of interest to the user

    Source code in prefect/events/schemas.py
    class Resource(Labelled):\n    \"\"\"An observable business object of interest to the user\"\"\"\n\n    @root_validator(pre=True)\n    def enforce_maximum_labels(cls, values: Dict[str, Any]):\n        labels = values.get(\"__root__\")\n        if not isinstance(labels, dict):\n            return values\n\n        if len(labels) > MAXIMUM_LABELS_PER_RESOURCE:\n            raise ValueError(\n                \"The maximum number of labels per resource \"\n                f\"is {MAXIMUM_LABELS_PER_RESOURCE}\"\n            )\n\n        return values\n\n    @root_validator(pre=True)\n    def requires_resource_id(cls, values: Dict[str, Any]):\n        labels = values.get(\"__root__\")\n        if not isinstance(labels, dict):\n            return values\n\n        labels = cast(Dict[str, str], labels)\n\n        if \"prefect.resource.id\" not in labels:\n            raise ValueError(\"Resources must include the prefect.resource.id label\")\n        if not labels[\"prefect.resource.id\"]:\n            raise ValueError(\"The prefect.resource.id label must be non-empty\")\n\n        return values\n\n    @property\n    def id(self) -> str:\n        return self[\"prefect.resource.id\"]\n
    ","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events.emit_event","title":"emit_event","text":"

    Send an event to Prefect Cloud.

    Parameters:

    Name Type Description Default event str

    The name of the event that happened.

    required resource Dict[str, str]

    The primary Resource this event concerns.

    required occurred Optional[DateTimeTZ]

    When the event happened from the sender's perspective. Defaults to the current datetime.

    None related Optional[Union[List[Dict[str, str]], List[RelatedResource]]]

    A list of additional Resources involved in this event.

    None payload Optional[Dict[str, Any]]

    An open-ended set of data describing what happened.

    None id Optional[UUID]

    The sender-provided identifier for this event. Defaults to a random UUID.

    None follows Optional[Event]

    The event that preceded this one. If the preceding event happened more than 5 minutes prior to this event the follows relationship will not be set.

    None

    Returns:

    Type Description Optional[Event]

    The event that was emitted if worker is using a client that emit

    Optional[Event]

    events, otherwise None.

    Source code in prefect/events/utilities.py
    def emit_event(\n    event: str,\n    resource: Dict[str, str],\n    occurred: Optional[DateTimeTZ] = None,\n    related: Optional[Union[List[Dict[str, str]], List[RelatedResource]]] = None,\n    payload: Optional[Dict[str, Any]] = None,\n    id: Optional[UUID] = None,\n    follows: Optional[Event] = None,\n) -> Optional[Event]:\n    \"\"\"\n    Send an event to Prefect Cloud.\n\n    Args:\n        event: The name of the event that happened.\n        resource: The primary Resource this event concerns.\n        occurred: When the event happened from the sender's perspective.\n                  Defaults to the current datetime.\n        related: A list of additional Resources involved in this event.\n        payload: An open-ended set of data describing what happened.\n        id: The sender-provided identifier for this event. Defaults to a random\n            UUID.\n        follows: The event that preceded this one. If the preceding event\n            happened more than 5 minutes prior to this event the follows\n            relationship will not be set.\n\n    Returns:\n        The event that was emitted if worker is using a client that emit\n        events, otherwise None.\n    \"\"\"\n    operational_clients = [AssertingEventsClient, PrefectCloudEventsClient]\n    worker_instance = EventsWorker.instance()\n\n    if worker_instance.client_type not in operational_clients:\n        return None\n\n    event_kwargs = {\n        \"event\": event,\n        \"resource\": resource,\n    }\n\n    if occurred is None:\n        occurred = pendulum.now(\"UTC\")\n    event_kwargs[\"occurred\"] = occurred\n\n    if related is not None:\n        event_kwargs[\"related\"] = related\n\n    if payload is not None:\n        event_kwargs[\"payload\"] = payload\n\n    if id is not None:\n        event_kwargs[\"id\"] = id\n\n    if follows is not None:\n        if -TIGHT_TIMING < (occurred - follows.occurred) < TIGHT_TIMING:\n            event_kwargs[\"follows\"] = follows.id\n\n    event_obj = Event(**event_kwargs)\n    worker_instance.send(event_obj)\n\n    return event_obj\n
    ","tags":["Python API","events"]},{"location":"api-ref/prefect/exceptions/","title":"prefect.exceptions","text":"","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions","title":"prefect.exceptions","text":"

    Prefect-specific exceptions.

    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.Abort","title":"Abort","text":"

    Bases: PrefectSignal

    Raised when the API sends an 'ABORT' instruction during state proposal.

    Indicates that the run should exit immediately.

    Source code in prefect/exceptions.py
    class Abort(PrefectSignal):\n    \"\"\"\n    Raised when the API sends an 'ABORT' instruction during state proposal.\n\n    Indicates that the run should exit immediately.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.BlockMissingCapabilities","title":"BlockMissingCapabilities","text":"

    Bases: PrefectException

    Raised when a block does not have required capabilities for a given operation.

    Source code in prefect/exceptions.py
    class BlockMissingCapabilities(PrefectException):\n    \"\"\"\n    Raised when a block does not have required capabilities for a given operation.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.CancelledRun","title":"CancelledRun","text":"

    Bases: PrefectException

    Raised when the result from a cancelled run is retrieved and an exception is not attached.

    This occurs when a string is attached to the state instead of an exception or if the state's data is null.

    Source code in prefect/exceptions.py
    class CancelledRun(PrefectException):\n    \"\"\"\n    Raised when the result from a cancelled run is retrieved and an exception\n    is not attached.\n\n    This occurs when a string is attached to the state instead of an exception\n    or if the state's data is null.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.CrashedRun","title":"CrashedRun","text":"

    Bases: PrefectException

    Raised when the result from a crashed run is retrieved.

    This occurs when a string is attached to the state instead of an exception or if the state's data is null.

    Source code in prefect/exceptions.py
    class CrashedRun(PrefectException):\n    \"\"\"\n    Raised when the result from a crashed run is retrieved.\n\n    This occurs when a string is attached to the state instead of an exception or if\n    the state's data is null.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ExternalSignal","title":"ExternalSignal","text":"

    Bases: BaseException

    Base type for external signal-like exceptions that should never be caught by users.

    Source code in prefect/exceptions.py
    class ExternalSignal(BaseException):\n    \"\"\"\n    Base type for external signal-like exceptions that should never be caught by users.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.FailedRun","title":"FailedRun","text":"

    Bases: PrefectException

    Raised when the result from a failed run is retrieved and an exception is not attached.

    This occurs when a string is attached to the state instead of an exception or if the state's data is null.

    Source code in prefect/exceptions.py
    class FailedRun(PrefectException):\n    \"\"\"\n    Raised when the result from a failed run is retrieved and an exception is not\n    attached.\n\n    This occurs when a string is attached to the state instead of an exception or if\n    the state's data is null.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.FlowPauseTimeout","title":"FlowPauseTimeout","text":"

    Bases: PrefectException

    Raised when a flow pause times out

    Source code in prefect/exceptions.py
    class FlowPauseTimeout(PrefectException):\n    \"\"\"Raised when a flow pause times out\"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.FlowScriptError","title":"FlowScriptError","text":"

    Bases: PrefectException

    Raised when a script errors during evaluation while attempting to load a flow.

    Source code in prefect/exceptions.py
    class FlowScriptError(PrefectException):\n    \"\"\"\n    Raised when a script errors during evaluation while attempting to load a flow.\n    \"\"\"\n\n    def __init__(\n        self,\n        user_exc: Exception,\n        script_path: str,\n    ) -> None:\n        message = f\"Flow script at {script_path!r} encountered an exception\"\n        super().__init__(message)\n\n        self.user_exc = user_exc\n\n    def rich_user_traceback(self, **kwargs):\n        trace = Traceback.extract(\n            type(self.user_exc),\n            self.user_exc,\n            self.user_exc.__traceback__.tb_next.tb_next.tb_next.tb_next,\n        )\n        return Traceback(trace, **kwargs)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InfrastructureError","title":"InfrastructureError","text":"

    Bases: PrefectException

    A base class for exceptions related to infrastructure blocks

    Source code in prefect/exceptions.py
    class InfrastructureError(PrefectException):\n    \"\"\"\n    A base class for exceptions related to infrastructure blocks\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InfrastructureNotAvailable","title":"InfrastructureNotAvailable","text":"

    Bases: PrefectException

    Raised when infrastructure is not accessible from the current machine. For example, if a process was spawned on another machine it cannot be managed.

    Source code in prefect/exceptions.py
    class InfrastructureNotAvailable(PrefectException):\n    \"\"\"\n    Raised when infrastructure is not accessible from the current machine. For example,\n    if a process was spawned on another machine it cannot be managed.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InfrastructureNotFound","title":"InfrastructureNotFound","text":"

    Bases: PrefectException

    Raised when infrastructure is missing, likely because it has exited or been deleted.

    Source code in prefect/exceptions.py
    class InfrastructureNotFound(PrefectException):\n    \"\"\"\n    Raised when infrastructure is missing, likely because it has exited or been\n    deleted.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InvalidNameError","title":"InvalidNameError","text":"

    Bases: PrefectException, ValueError

    Raised when a name contains characters that are not permitted.

    Source code in prefect/exceptions.py
    class InvalidNameError(PrefectException, ValueError):\n    \"\"\"\n    Raised when a name contains characters that are not permitted.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InvalidRepositoryURLError","title":"InvalidRepositoryURLError","text":"

    Bases: PrefectException

    Raised when an incorrect URL is provided to a GitHub filesystem block.

    Source code in prefect/exceptions.py
    class InvalidRepositoryURLError(PrefectException):\n    \"\"\"Raised when an incorrect URL is provided to a GitHub filesystem block.\"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MappingLengthMismatch","title":"MappingLengthMismatch","text":"

    Bases: PrefectException

    Raised when attempting to call Task.map with arguments of different lengths.

    Source code in prefect/exceptions.py
    class MappingLengthMismatch(PrefectException):\n    \"\"\"\n    Raised when attempting to call Task.map with arguments of different lengths.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MappingMissingIterable","title":"MappingMissingIterable","text":"

    Bases: PrefectException

    Raised when attempting to call Task.map with all static arguments

    Source code in prefect/exceptions.py
    class MappingMissingIterable(PrefectException):\n    \"\"\"\n    Raised when attempting to call Task.map with all static arguments\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MissingContextError","title":"MissingContextError","text":"

    Bases: PrefectException, RuntimeError

    Raised when a method is called that requires a task or flow run context to be active but one cannot be found.

    Source code in prefect/exceptions.py
    class MissingContextError(PrefectException, RuntimeError):\n    \"\"\"\n    Raised when a method is called that requires a task or flow run context to be\n    active but one cannot be found.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MissingFlowError","title":"MissingFlowError","text":"

    Bases: PrefectException

    Raised when a given flow name is not found in the expected script.

    Source code in prefect/exceptions.py
    class MissingFlowError(PrefectException):\n    \"\"\"\n    Raised when a given flow name is not found in the expected script.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MissingProfileError","title":"MissingProfileError","text":"

    Bases: PrefectException, ValueError

    Raised when a profile name does not exist.

    Source code in prefect/exceptions.py
    class MissingProfileError(PrefectException, ValueError):\n    \"\"\"\n    Raised when a profile name does not exist.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MissingResult","title":"MissingResult","text":"

    Bases: PrefectException

    Raised when a result is missing from a state; often when result persistence is disabled and the state is retrieved from the API.

    Source code in prefect/exceptions.py
    class MissingResult(PrefectException):\n    \"\"\"\n    Raised when a result is missing from a state; often when result persistence is\n    disabled and the state is retrieved from the API.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.NotPausedError","title":"NotPausedError","text":"

    Bases: PrefectException

    Raised when attempting to unpause a run that isn't paused.

    Source code in prefect/exceptions.py
    class NotPausedError(PrefectException):\n    \"\"\"Raised when attempting to unpause a run that isn't paused.\"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ObjectAlreadyExists","title":"ObjectAlreadyExists","text":"

    Bases: PrefectException

    Raised when the client receives a 409 (conflict) from the API.

    Source code in prefect/exceptions.py
    class ObjectAlreadyExists(PrefectException):\n    \"\"\"\n    Raised when the client receives a 409 (conflict) from the API.\n    \"\"\"\n\n    def __init__(self, http_exc: Exception, *args, **kwargs):\n        self.http_exc = http_exc\n        super().__init__(*args, **kwargs)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ObjectNotFound","title":"ObjectNotFound","text":"

    Bases: PrefectException

    Raised when the client receives a 404 (not found) from the API.

    Source code in prefect/exceptions.py
    class ObjectNotFound(PrefectException):\n    \"\"\"\n    Raised when the client receives a 404 (not found) from the API.\n    \"\"\"\n\n    def __init__(self, http_exc: Exception, *args, **kwargs):\n        self.http_exc = http_exc\n        super().__init__(*args, **kwargs)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ParameterBindError","title":"ParameterBindError","text":"

    Bases: TypeError, PrefectException

    Raised when args and kwargs cannot be converted to parameters.

    Source code in prefect/exceptions.py
    class ParameterBindError(TypeError, PrefectException):\n    \"\"\"\n    Raised when args and kwargs cannot be converted to parameters.\n    \"\"\"\n\n    def __init__(self, msg: str):\n        super().__init__(msg)\n\n    @classmethod\n    def from_bind_failure(\n        cls, fn: Callable, exc: TypeError, call_args: List, call_kwargs: Dict\n    ) -> Self:\n        fn_signature = str(inspect.signature(fn)).strip(\"()\")\n\n        base = f\"Error binding parameters for function '{fn.__name__}': {exc}\"\n        signature = f\"Function '{fn.__name__}' has signature '{fn_signature}'\"\n        received = f\"received args: {call_args} and kwargs: {list(call_kwargs.keys())}\"\n        msg = f\"{base}.\\n{signature} but {received}.\"\n        return cls(msg)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ParameterTypeError","title":"ParameterTypeError","text":"

    Bases: PrefectException

    Raised when a parameter does not pass Pydantic type validation.

    Source code in prefect/exceptions.py
    class ParameterTypeError(PrefectException):\n    \"\"\"\n    Raised when a parameter does not pass Pydantic type validation.\n    \"\"\"\n\n    def __init__(self, msg: str):\n        super().__init__(msg)\n\n    @classmethod\n    def from_validation_error(cls, exc: pydantic.ValidationError) -> Self:\n        bad_params = [f'{err[\"loc\"][0]}: {err[\"msg\"]}' for err in exc.errors()]\n        msg = \"Flow run received invalid parameters:\\n - \" + \"\\n - \".join(bad_params)\n        return cls(msg)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.Pause","title":"Pause","text":"

    Bases: PrefectSignal

    Raised when a flow run is PAUSED and needs to exit for resubmission.

    Source code in prefect/exceptions.py
    class Pause(PrefectSignal):\n    \"\"\"\n    Raised when a flow run is PAUSED and needs to exit for resubmission.\n    \"\"\"\n\n    def __init__(self, *args, state=None, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.state = state\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PausedRun","title":"PausedRun","text":"

    Bases: PrefectException

    Raised when the result from a paused run is retrieved.

    Source code in prefect/exceptions.py
    class PausedRun(PrefectException):\n    \"\"\"\n    Raised when the result from a paused run is retrieved.\n    \"\"\"\n\n    def __init__(self, *args, state=None, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.state = state\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PrefectException","title":"PrefectException","text":"

    Bases: Exception

    Base exception type for Prefect errors.

    Source code in prefect/exceptions.py
    class PrefectException(Exception):\n    \"\"\"\n    Base exception type for Prefect errors.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PrefectHTTPStatusError","title":"PrefectHTTPStatusError","text":"

    Bases: HTTPStatusError

    Raised when client receives a Response that contains an HTTPStatusError.

    Used to include API error details in the error messages that the client provides users.

    Source code in prefect/exceptions.py
    class PrefectHTTPStatusError(HTTPStatusError):\n    \"\"\"\n    Raised when client receives a `Response` that contains an HTTPStatusError.\n\n    Used to include API error details in the error messages that the client provides users.\n    \"\"\"\n\n    @classmethod\n    def from_httpx_error(cls: Type[Self], httpx_error: HTTPStatusError) -> Self:\n        \"\"\"\n        Generate a `PrefectHTTPStatusError` from an `httpx.HTTPStatusError`.\n        \"\"\"\n        try:\n            details = httpx_error.response.json()\n        except Exception:\n            details = None\n\n        error_message, *more_info = str(httpx_error).split(\"\\n\")\n\n        if details:\n            message_components = [error_message, f\"Response: {details}\", *more_info]\n        else:\n            message_components = [error_message, *more_info]\n\n        new_message = \"\\n\".join(message_components)\n\n        return cls(\n            new_message, request=httpx_error.request, response=httpx_error.response\n        )\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PrefectHTTPStatusError.from_httpx_error","title":"from_httpx_error classmethod","text":"

    Generate a PrefectHTTPStatusError from an httpx.HTTPStatusError.

    Source code in prefect/exceptions.py
    @classmethod\ndef from_httpx_error(cls: Type[Self], httpx_error: HTTPStatusError) -> Self:\n    \"\"\"\n    Generate a `PrefectHTTPStatusError` from an `httpx.HTTPStatusError`.\n    \"\"\"\n    try:\n        details = httpx_error.response.json()\n    except Exception:\n        details = None\n\n    error_message, *more_info = str(httpx_error).split(\"\\n\")\n\n    if details:\n        message_components = [error_message, f\"Response: {details}\", *more_info]\n    else:\n        message_components = [error_message, *more_info]\n\n    new_message = \"\\n\".join(message_components)\n\n    return cls(\n        new_message, request=httpx_error.request, response=httpx_error.response\n    )\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PrefectSignal","title":"PrefectSignal","text":"

    Bases: BaseException

    Base type for signal-like exceptions that should never be caught by users.

    Source code in prefect/exceptions.py
    class PrefectSignal(BaseException):\n    \"\"\"\n    Base type for signal-like exceptions that should never be caught by users.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ProtectedBlockError","title":"ProtectedBlockError","text":"

    Bases: PrefectException

    Raised when an operation is prevented due to block protection.

    Source code in prefect/exceptions.py
    class ProtectedBlockError(PrefectException):\n    \"\"\"\n    Raised when an operation is prevented due to block protection.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ReservedArgumentError","title":"ReservedArgumentError","text":"

    Bases: PrefectException, TypeError

    Raised when a function used with Prefect has an argument with a name that is reserved for a Prefect feature

    Source code in prefect/exceptions.py
    class ReservedArgumentError(PrefectException, TypeError):\n    \"\"\"\n    Raised when a function used with Prefect has an argument with a name that is\n    reserved for a Prefect feature\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ScriptError","title":"ScriptError","text":"

    Bases: PrefectException

    Raised when a script errors during evaluation while attempting to load data

    Source code in prefect/exceptions.py
    class ScriptError(PrefectException):\n    \"\"\"\n    Raised when a script errors during evaluation while attempting to load data\n    \"\"\"\n\n    def __init__(\n        self,\n        user_exc: Exception,\n        path: str,\n    ) -> None:\n        message = f\"Script at {str(path)!r} encountered an exception: {user_exc!r}\"\n        super().__init__(message)\n        self.user_exc = user_exc\n\n        # Strip script run information from the traceback\n        self.user_exc.__traceback__ = _trim_traceback(\n            self.user_exc.__traceback__,\n            remove_modules=[prefect.utilities.importtools],\n        )\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.SignatureMismatchError","title":"SignatureMismatchError","text":"

    Bases: PrefectException, TypeError

    Raised when parameters passed to a function do not match its signature.

    Source code in prefect/exceptions.py
    class SignatureMismatchError(PrefectException, TypeError):\n    \"\"\"Raised when parameters passed to a function do not match its signature.\"\"\"\n\n    def __init__(self, msg: str):\n        super().__init__(msg)\n\n    @classmethod\n    def from_bad_params(cls, expected_params: List[str], provided_params: List[str]):\n        msg = (\n            f\"Function expects parameters {expected_params} but was provided with\"\n            f\" parameters {provided_params}\"\n        )\n        return cls(msg)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.TerminationSignal","title":"TerminationSignal","text":"

    Bases: ExternalSignal

    Raised when a flow run receives a termination signal.

    Source code in prefect/exceptions.py
    class TerminationSignal(ExternalSignal):\n    \"\"\"\n    Raised when a flow run receives a termination signal.\n    \"\"\"\n\n    def __init__(self, signal: int):\n        self.signal = signal\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.UnfinishedRun","title":"UnfinishedRun","text":"

    Bases: PrefectException

    Raised when the result from a run that is not finished is retrieved.

    For example, if a run is in a SCHEDULED, PENDING, CANCELLING, or RUNNING state.

    Source code in prefect/exceptions.py
    class UnfinishedRun(PrefectException):\n    \"\"\"\n    Raised when the result from a run that is not finished is retrieved.\n\n    For example, if a run is in a SCHEDULED, PENDING, CANCELLING, or RUNNING state.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.UnspecifiedFlowError","title":"UnspecifiedFlowError","text":"

    Bases: PrefectException

    Raised when multiple flows are found in the expected script and no name is given.

    Source code in prefect/exceptions.py
    class UnspecifiedFlowError(PrefectException):\n    \"\"\"\n    Raised when multiple flows are found in the expected script and no name is given.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.UpstreamTaskError","title":"UpstreamTaskError","text":"

    Bases: PrefectException

    Raised when a task relies on the result of another task but that task is not 'COMPLETE'

    Source code in prefect/exceptions.py
    class UpstreamTaskError(PrefectException):\n    \"\"\"\n    Raised when a task relies on the result of another task but that task is not\n    'COMPLETE'\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.exception_traceback","title":"exception_traceback","text":"

    Convert an exception to a printable string with a traceback

    Source code in prefect/exceptions.py
    def exception_traceback(exc: Exception) -> str:\n    \"\"\"\n    Convert an exception to a printable string with a traceback\n    \"\"\"\n    tb = traceback.TracebackException.from_exception(exc)\n    return \"\".join(list(tb.format()))\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/filesystems/","title":"prefect.filesystems","text":"","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems","title":"prefect.filesystems","text":"","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.Azure","title":"Azure","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on Azure Datalake and Azure Blob Storage.

    Example

    Load stored Azure config:

    from prefect.filesystems import Azure\n\naz_block = Azure.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class Azure(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on Azure Datalake and Azure Blob Storage.\n\n    Example:\n        Load stored Azure config:\n        ```python\n        from prefect.filesystems import Azure\n\n        az_block = Azure.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Azure\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/54e3fa7e00197a4fbd1d82ed62494cb58d08c96a-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#azure\"\n\n    bucket_path: str = Field(\n        default=...,\n        description=\"An Azure storage bucket path.\",\n        example=\"my-bucket/a-directory-within\",\n    )\n    azure_storage_connection_string: Optional[SecretStr] = Field(\n        default=None,\n        title=\"Azure storage connection string\",\n        description=(\n            \"Equivalent to the AZURE_STORAGE_CONNECTION_STRING environment variable.\"\n        ),\n    )\n    azure_storage_account_name: Optional[SecretStr] = Field(\n        default=None,\n        title=\"Azure storage account name\",\n        description=(\n            \"Equivalent to the AZURE_STORAGE_ACCOUNT_NAME environment variable.\"\n        ),\n    )\n    azure_storage_account_key: Optional[SecretStr] = Field(\n        default=None,\n        title=\"Azure storage account key\",\n        description=\"Equivalent to the AZURE_STORAGE_ACCOUNT_KEY environment variable.\",\n    )\n    azure_storage_tenant_id: Optional[SecretStr] = Field(\n        None,\n        title=\"Azure storage tenant ID\",\n        description=\"Equivalent to the AZURE_TENANT_ID environment variable.\",\n    )\n    azure_storage_client_id: Optional[SecretStr] = Field(\n        None,\n        title=\"Azure storage client ID\",\n        description=\"Equivalent to the AZURE_CLIENT_ID environment variable.\",\n    )\n    azure_storage_client_secret: Optional[SecretStr] = Field(\n        None,\n        title=\"Azure storage client secret\",\n        description=\"Equivalent to the AZURE_CLIENT_SECRET environment variable.\",\n    )\n    azure_storage_anon: bool = Field(\n        default=True,\n        title=\"Azure storage anonymous connection\",\n        description=(\n            \"Set the 'anon' flag for ADLFS. This should be False for systems that\"\n            \" require ADLFS to use DefaultAzureCredentials.\"\n        ),\n    )\n\n    _remote_file_system: RemoteFileSystem = None\n\n    @property\n    def basepath(self) -> str:\n        return f\"az://{self.bucket_path}\"\n\n    @property\n    def filesystem(self) -> RemoteFileSystem:\n        settings = {}\n        if self.azure_storage_connection_string:\n            settings[\"connection_string\"] = (\n                self.azure_storage_connection_string.get_secret_value()\n            )\n        if self.azure_storage_account_name:\n            settings[\"account_name\"] = (\n                self.azure_storage_account_name.get_secret_value()\n            )\n        if self.azure_storage_account_key:\n            settings[\"account_key\"] = self.azure_storage_account_key.get_secret_value()\n        if self.azure_storage_tenant_id:\n            settings[\"tenant_id\"] = self.azure_storage_tenant_id.get_secret_value()\n        if self.azure_storage_client_id:\n            settings[\"client_id\"] = self.azure_storage_client_id.get_secret_value()\n        if self.azure_storage_client_secret:\n            settings[\"client_secret\"] = (\n                self.azure_storage_client_secret.get_secret_value()\n            )\n        settings[\"anon\"] = self.azure_storage_anon\n        self._remote_file_system = RemoteFileSystem(\n            basepath=f\"az://{self.bucket_path}\", settings=settings\n        )\n        return self._remote_file_system\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> bytes:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        return await self.filesystem.get_directory(\n            from_path=from_path, local_path=local_path\n        )\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        return await self.filesystem.put_directory(\n            local_path=local_path, to_path=to_path, ignore_file=ignore_file\n        )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        return await self.filesystem.read_path(path)\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        return await self.filesystem.write_path(path=path, content=content)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.Azure.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory.

    Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> bytes:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    return await self.filesystem.get_directory(\n        from_path=from_path, local_path=local_path\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.Azure.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory.

    Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    return await self.filesystem.put_directory(\n        local_path=local_path, to_path=to_path, ignore_file=ignore_file\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GCS","title":"GCS","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on Google Cloud Storage.

    Example

    Load stored GCS config:

    from prefect.filesystems import GCS\n\ngcs_block = GCS.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class GCS(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on Google Cloud Storage.\n\n    Example:\n        Load stored GCS config:\n        ```python\n        from prefect.filesystems import GCS\n\n        gcs_block = GCS.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/422d13bb838cf247eb2b2cf229ce6a2e717d601b-256x256.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#gcs\"\n\n    bucket_path: str = Field(\n        default=...,\n        description=\"A GCS bucket path.\",\n        example=\"my-bucket/a-directory-within\",\n    )\n    service_account_info: Optional[SecretStr] = Field(\n        default=None,\n        description=\"The contents of a service account keyfile as a JSON string.\",\n    )\n    project: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The project the GCS bucket resides in. If not provided, the project will\"\n            \" be inferred from the credentials or environment.\"\n        ),\n    )\n\n    @property\n    def basepath(self) -> str:\n        return f\"gcs://{self.bucket_path}\"\n\n    @property\n    def filesystem(self) -> RemoteFileSystem:\n        settings = {}\n        if self.service_account_info:\n            try:\n                settings[\"token\"] = json.loads(\n                    self.service_account_info.get_secret_value()\n                )\n            except json.JSONDecodeError:\n                raise ValueError(\n                    \"Unable to load provided service_account_info. Please make sure\"\n                    \" that the provided value is a valid JSON string.\"\n                )\n        remote_file_system = RemoteFileSystem(\n            basepath=f\"gcs://{self.bucket_path}\", settings=settings\n        )\n        return remote_file_system\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> bytes:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        return await self.filesystem.get_directory(\n            from_path=from_path, local_path=local_path\n        )\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        return await self.filesystem.put_directory(\n            local_path=local_path, to_path=to_path, ignore_file=ignore_file\n        )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        return await self.filesystem.read_path(path)\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        return await self.filesystem.write_path(path=path, content=content)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GCS.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory.

    Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> bytes:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    return await self.filesystem.get_directory(\n        from_path=from_path, local_path=local_path\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GCS.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory.

    Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    return await self.filesystem.put_directory(\n        local_path=local_path, to_path=to_path, ignore_file=ignore_file\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GitHub","title":"GitHub","text":"

    Bases: ReadableDeploymentStorage

    Interact with files stored on GitHub repositories.

    Source code in prefect/filesystems.py
    class GitHub(ReadableDeploymentStorage):\n    \"\"\"\n    Interact with files stored on GitHub repositories.\n    \"\"\"\n\n    _block_type_name = \"GitHub\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/41971cfecfea5f79ff334164f06ecb34d1038dd4-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#github\"\n\n    repository: str = Field(\n        default=...,\n        description=(\n            \"The URL of a GitHub repository to read from, in either HTTPS or SSH\"\n            \" format.\"\n        ),\n    )\n    reference: Optional[str] = Field(\n        default=None,\n        description=\"An optional reference to pin to; can be a branch name or tag.\",\n    )\n    access_token: Optional[SecretStr] = Field(\n        name=\"Personal Access Token\",\n        default=None,\n        description=(\n            \"A GitHub Personal Access Token (PAT) with repo scope.\"\n            \" To use a fine-grained PAT, provide '{username}:{PAT}' as the value.\"\n        ),\n    )\n    include_git_objects: bool = Field(\n        default=True,\n        description=(\n            \"Whether to include git objects when copying the repo contents to a\"\n            \" directory.\"\n        ),\n    )\n\n    @validator(\"access_token\")\n    def _ensure_credentials_go_with_https(cls, v: str, values: dict) -> str:\n        \"\"\"Ensure that credentials are not provided with 'SSH' formatted GitHub URLs.\n\n        Note: validates `access_token` specifically so that it only fires when\n        private repositories are used.\n        \"\"\"\n        if v is not None:\n            if urllib.parse.urlparse(values[\"repository\"]).scheme != \"https\":\n                raise InvalidRepositoryURLError(\n                    \"Crendentials can only be used with GitHub repositories \"\n                    \"using the 'HTTPS' format. You must either remove the \"\n                    \"credential if you wish to use the 'SSH' format and are not \"\n                    \"using a private repository, or you must change the repository \"\n                    \"URL to the 'HTTPS' format. \"\n                )\n\n        return v\n\n    def _create_repo_url(self) -> str:\n        \"\"\"Format the URL provided to the `git clone` command.\n\n        For private repos: https://<oauth-key>@github.com/<username>/<repo>.git\n        All other repos should be the same as `self.repository`.\n        \"\"\"\n        url_components = urllib.parse.urlparse(self.repository)\n        if url_components.scheme == \"https\" and self.access_token is not None:\n            updated_components = url_components._replace(\n                netloc=f\"{self.access_token.get_secret_value()}@{url_components.netloc}\"\n            )\n            full_url = urllib.parse.urlunparse(updated_components)\n        else:\n            full_url = self.repository\n\n        return full_url\n\n    @staticmethod\n    def _get_paths(\n        dst_dir: Union[str, None], src_dir: str, sub_directory: str\n    ) -> Tuple[str, str]:\n        \"\"\"Returns the fully formed paths for GitHubRepository contents in the form\n        (content_source, content_destination).\n        \"\"\"\n        if dst_dir is None:\n            content_destination = Path(\".\").absolute()\n        else:\n            content_destination = Path(dst_dir)\n\n        content_source = Path(src_dir)\n\n        if sub_directory:\n            content_destination = content_destination.joinpath(sub_directory)\n            content_source = content_source.joinpath(sub_directory)\n\n        return str(content_source), str(content_destination)\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> None:\n        \"\"\"\n        Clones a GitHub project specified in `from_path` to the provided `local_path`;\n        defaults to cloning the repository reference configured on the Block to the\n        present working directory.\n\n        Args:\n            from_path: If provided, interpreted as a subdirectory of the underlying\n                repository that will be copied to the provided local path.\n            local_path: A local path to clone to; defaults to present working directory.\n        \"\"\"\n        # CONSTRUCT COMMAND\n        cmd = [\"git\", \"clone\", self._create_repo_url()]\n        if self.reference:\n            cmd += [\"-b\", self.reference]\n\n        # Limit git history\n        cmd += [\"--depth\", \"1\"]\n\n        # Clone to a temporary directory and move the subdirectory over\n        with TemporaryDirectory(suffix=\"prefect\") as tmp_dir:\n            cmd.append(tmp_dir)\n\n            err_stream = io.StringIO()\n            out_stream = io.StringIO()\n            process = await run_process(cmd, stream_output=(out_stream, err_stream))\n            if process.returncode != 0:\n                err_stream.seek(0)\n                raise OSError(f\"Failed to pull from remote:\\n {err_stream.read()}\")\n\n            content_source, content_destination = self._get_paths(\n                dst_dir=local_path, src_dir=tmp_dir, sub_directory=from_path\n            )\n\n            ignore_func = None\n            if not self.include_git_objects:\n                ignore_func = ignore_patterns(\".git\")\n\n            copytree(\n                src=content_source,\n                dst=content_destination,\n                dirs_exist_ok=True,\n                ignore=ignore_func,\n            )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GitHub.get_directory","title":"get_directory async","text":"

    Clones a GitHub project specified in from_path to the provided local_path; defaults to cloning the repository reference configured on the Block to the present working directory.

    Parameters:

    Name Type Description Default from_path Optional[str]

    If provided, interpreted as a subdirectory of the underlying repository that will be copied to the provided local path.

    None local_path Optional[str]

    A local path to clone to; defaults to present working directory.

    None Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> None:\n    \"\"\"\n    Clones a GitHub project specified in `from_path` to the provided `local_path`;\n    defaults to cloning the repository reference configured on the Block to the\n    present working directory.\n\n    Args:\n        from_path: If provided, interpreted as a subdirectory of the underlying\n            repository that will be copied to the provided local path.\n        local_path: A local path to clone to; defaults to present working directory.\n    \"\"\"\n    # CONSTRUCT COMMAND\n    cmd = [\"git\", \"clone\", self._create_repo_url()]\n    if self.reference:\n        cmd += [\"-b\", self.reference]\n\n    # Limit git history\n    cmd += [\"--depth\", \"1\"]\n\n    # Clone to a temporary directory and move the subdirectory over\n    with TemporaryDirectory(suffix=\"prefect\") as tmp_dir:\n        cmd.append(tmp_dir)\n\n        err_stream = io.StringIO()\n        out_stream = io.StringIO()\n        process = await run_process(cmd, stream_output=(out_stream, err_stream))\n        if process.returncode != 0:\n            err_stream.seek(0)\n            raise OSError(f\"Failed to pull from remote:\\n {err_stream.read()}\")\n\n        content_source, content_destination = self._get_paths(\n            dst_dir=local_path, src_dir=tmp_dir, sub_directory=from_path\n        )\n\n        ignore_func = None\n        if not self.include_git_objects:\n            ignore_func = ignore_patterns(\".git\")\n\n        copytree(\n            src=content_source,\n            dst=content_destination,\n            dirs_exist_ok=True,\n            ignore=ignore_func,\n        )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.LocalFileSystem","title":"LocalFileSystem","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on a local file system.

    Example

    Load stored local file system config:

    from prefect.filesystems import LocalFileSystem\n\nlocal_file_system_block = LocalFileSystem.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class LocalFileSystem(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on a local file system.\n\n    Example:\n        Load stored local file system config:\n        ```python\n        from prefect.filesystems import LocalFileSystem\n\n        local_file_system_block = LocalFileSystem.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Local File System\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/ad39089fa66d273b943394a68f003f7a19aa850e-48x48.png\"\n    _documentation_url = (\n        \"https://docs.prefect.io/concepts/filesystems/#local-filesystem\"\n    )\n\n    basepath: Optional[str] = Field(\n        default=None, description=\"Default local path for this block to write to.\"\n    )\n\n    @validator(\"basepath\", pre=True)\n    def cast_pathlib(cls, value):\n        if isinstance(value, Path):\n            return str(value)\n        return value\n\n    def _resolve_path(self, path: str) -> Path:\n        # Only resolve the base path at runtime, default to the current directory\n        basepath = (\n            Path(self.basepath).expanduser().resolve()\n            if self.basepath\n            else Path(\".\").resolve()\n        )\n\n        # Determine the path to access relative to the base path, ensuring that paths\n        # outside of the base path are off limits\n        if path is None:\n            return basepath\n\n        path: Path = Path(path).expanduser()\n\n        if not path.is_absolute():\n            path = basepath / path\n        else:\n            path = path.resolve()\n            if basepath not in path.parents and (basepath != path):\n                raise ValueError(\n                    f\"Provided path {path} is outside of the base path {basepath}.\"\n                )\n\n        return path\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: str = None, local_path: str = None\n    ) -> None:\n        \"\"\"\n        Copies a directory from one place to another on the local filesystem.\n\n        Defaults to copying the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        if not from_path:\n            from_path = Path(self.basepath).expanduser().resolve()\n        else:\n            from_path = self._resolve_path(from_path)\n\n        if not local_path:\n            local_path = Path(\".\").resolve()\n        else:\n            local_path = Path(local_path).resolve()\n\n        if from_path == local_path:\n            # If the paths are the same there is no need to copy\n            # and we avoid shutil.copytree raising an error\n            return\n\n        # .prefectignore exists in the original location, not the current location which\n        # is most likely temporary\n        if (from_path / Path(\".prefectignore\")).exists():\n            ignore_func = await self._get_ignore_func(\n                local_path=from_path, ignore_file=from_path / Path(\".prefectignore\")\n            )\n        else:\n            ignore_func = None\n\n        copytree(from_path, local_path, dirs_exist_ok=True, ignore=ignore_func)\n\n    async def _get_ignore_func(self, local_path: str, ignore_file: str):\n        with open(ignore_file, \"r\") as f:\n            ignore_patterns = f.readlines()\n        included_files = filter_files(root=local_path, ignore_patterns=ignore_patterns)\n\n        def ignore_func(directory, files):\n            relative_path = Path(directory).relative_to(local_path)\n\n            files_to_ignore = [\n                f for f in files if str(relative_path / f) not in included_files\n            ]\n            return files_to_ignore\n\n        return ignore_func\n\n    @sync_compatible\n    async def put_directory(\n        self, local_path: str = None, to_path: str = None, ignore_file: str = None\n    ) -> None:\n        \"\"\"\n        Copies a directory from one place to another on the local filesystem.\n\n        Defaults to copying the entire contents of the current working directory to the block's basepath.\n        An `ignore_file` path may be provided that can include gitignore style expressions for filepaths to ignore.\n        \"\"\"\n        destination_path = self._resolve_path(to_path)\n\n        if not local_path:\n            local_path = Path(\".\").absolute()\n\n        if ignore_file:\n            ignore_func = await self._get_ignore_func(\n                local_path=local_path, ignore_file=ignore_file\n            )\n        else:\n            ignore_func = None\n\n        if local_path == destination_path:\n            pass\n        else:\n            copytree(\n                src=local_path,\n                dst=destination_path,\n                ignore=ignore_func,\n                dirs_exist_ok=True,\n            )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        path: Path = self._resolve_path(path)\n\n        # Check if the path exists\n        if not path.exists():\n            raise ValueError(f\"Path {path} does not exist.\")\n\n        # Validate that its a file\n        if not path.is_file():\n            raise ValueError(f\"Path {path} is not a file.\")\n\n        async with await anyio.open_file(str(path), mode=\"rb\") as f:\n            content = await f.read()\n\n        return content\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        path: Path = self._resolve_path(path)\n\n        # Construct the path if it does not exist\n        path.parent.mkdir(exist_ok=True, parents=True)\n\n        # Check if the file already exists\n        if path.exists() and not path.is_file():\n            raise ValueError(f\"Path {path} already exists and is not a file.\")\n\n        async with await anyio.open_file(path, mode=\"wb\") as f:\n            await f.write(content)\n        # Leave path stringify to the OS\n        return str(path)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.LocalFileSystem.get_directory","title":"get_directory async","text":"

    Copies a directory from one place to another on the local filesystem.

    Defaults to copying the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: str = None, local_path: str = None\n) -> None:\n    \"\"\"\n    Copies a directory from one place to another on the local filesystem.\n\n    Defaults to copying the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    if not from_path:\n        from_path = Path(self.basepath).expanduser().resolve()\n    else:\n        from_path = self._resolve_path(from_path)\n\n    if not local_path:\n        local_path = Path(\".\").resolve()\n    else:\n        local_path = Path(local_path).resolve()\n\n    if from_path == local_path:\n        # If the paths are the same there is no need to copy\n        # and we avoid shutil.copytree raising an error\n        return\n\n    # .prefectignore exists in the original location, not the current location which\n    # is most likely temporary\n    if (from_path / Path(\".prefectignore\")).exists():\n        ignore_func = await self._get_ignore_func(\n            local_path=from_path, ignore_file=from_path / Path(\".prefectignore\")\n        )\n    else:\n        ignore_func = None\n\n    copytree(from_path, local_path, dirs_exist_ok=True, ignore=ignore_func)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.LocalFileSystem.put_directory","title":"put_directory async","text":"

    Copies a directory from one place to another on the local filesystem.

    Defaults to copying the entire contents of the current working directory to the block's basepath. An ignore_file path may be provided that can include gitignore style expressions for filepaths to ignore.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self, local_path: str = None, to_path: str = None, ignore_file: str = None\n) -> None:\n    \"\"\"\n    Copies a directory from one place to another on the local filesystem.\n\n    Defaults to copying the entire contents of the current working directory to the block's basepath.\n    An `ignore_file` path may be provided that can include gitignore style expressions for filepaths to ignore.\n    \"\"\"\n    destination_path = self._resolve_path(to_path)\n\n    if not local_path:\n        local_path = Path(\".\").absolute()\n\n    if ignore_file:\n        ignore_func = await self._get_ignore_func(\n            local_path=local_path, ignore_file=ignore_file\n        )\n    else:\n        ignore_func = None\n\n    if local_path == destination_path:\n        pass\n    else:\n        copytree(\n            src=local_path,\n            dst=destination_path,\n            ignore=ignore_func,\n            dirs_exist_ok=True,\n        )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.RemoteFileSystem","title":"RemoteFileSystem","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on a remote file system.

    Supports any remote file system supported by fsspec. The file system is specified using a protocol. For example, \"s3://my-bucket/my-folder/\" will use S3.

    Example

    Load stored remote file system config:

    from prefect.filesystems import RemoteFileSystem\n\nremote_file_system_block = RemoteFileSystem.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class RemoteFileSystem(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on a remote file system.\n\n    Supports any remote file system supported by `fsspec`. The file system is specified\n    using a protocol. For example, \"s3://my-bucket/my-folder/\" will use S3.\n\n    Example:\n        Load stored remote file system config:\n        ```python\n        from prefect.filesystems import RemoteFileSystem\n\n        remote_file_system_block = RemoteFileSystem.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Remote File System\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/e86b41bc0f9c99ba9489abeee83433b43d5c9365-48x48.png\"\n    _documentation_url = (\n        \"https://docs.prefect.io/concepts/filesystems/#remote-file-system\"\n    )\n\n    basepath: str = Field(\n        default=...,\n        description=\"Default path for this block to write to.\",\n        example=\"s3://my-bucket/my-folder/\",\n    )\n    settings: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Additional settings to pass through to fsspec.\",\n    )\n\n    # Cache for the configured fsspec file system used for access\n    _filesystem: fsspec.AbstractFileSystem = None\n\n    @validator(\"basepath\")\n    def check_basepath(cls, value):\n        scheme, netloc, _, _, _ = urllib.parse.urlsplit(value)\n\n        if not scheme:\n            raise ValueError(f\"Base path must start with a scheme. Got {value!r}.\")\n\n        if not netloc:\n            raise ValueError(\n                f\"Base path must include a location after the scheme. Got {value!r}.\"\n            )\n\n        if scheme == \"file\":\n            raise ValueError(\n                \"Base path scheme cannot be 'file'. Use `LocalFileSystem` instead for\"\n                \" local file access.\"\n            )\n\n        return value\n\n    def _resolve_path(self, path: str) -> str:\n        base_scheme, base_netloc, base_urlpath, _, _ = urllib.parse.urlsplit(\n            self.basepath\n        )\n        scheme, netloc, urlpath, _, _ = urllib.parse.urlsplit(path)\n\n        # Confirm that absolute paths are valid\n        if scheme:\n            if scheme != base_scheme:\n                raise ValueError(\n                    f\"Path {path!r} with scheme {scheme!r} must use the same scheme as\"\n                    f\" the base path {base_scheme!r}.\"\n                )\n\n        if netloc:\n            if (netloc != base_netloc) or not urlpath.startswith(base_urlpath):\n                raise ValueError(\n                    f\"Path {path!r} is outside of the base path {self.basepath!r}.\"\n                )\n\n        return f\"{self.basepath.rstrip('/')}/{urlpath.lstrip('/')}\"\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> None:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        if from_path is None:\n            from_path = str(self.basepath)\n        else:\n            from_path = self._resolve_path(from_path)\n\n        if local_path is None:\n            local_path = Path(\".\").absolute()\n\n        # validate that from_path has a trailing slash for proper fsspec behavior across versions\n        if not from_path.endswith(\"/\"):\n            from_path += \"/\"\n\n        return self.filesystem.get(from_path, local_path, recursive=True)\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n        overwrite: bool = True,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        if to_path is None:\n            to_path = str(self.basepath)\n        else:\n            to_path = self._resolve_path(to_path)\n\n        if local_path is None:\n            local_path = \".\"\n\n        included_files = None\n        if ignore_file:\n            with open(ignore_file, \"r\") as f:\n                ignore_patterns = f.readlines()\n\n            included_files = filter_files(\n                local_path, ignore_patterns, include_dirs=True\n            )\n\n        counter = 0\n        for f in Path(local_path).rglob(\"*\"):\n            relative_path = f.relative_to(local_path)\n            if included_files and str(relative_path) not in included_files:\n                continue\n\n            if to_path.endswith(\"/\"):\n                fpath = to_path + relative_path.as_posix()\n            else:\n                fpath = to_path + \"/\" + relative_path.as_posix()\n\n            if f.is_dir():\n                pass\n            else:\n                f = f.as_posix()\n                if overwrite:\n                    self.filesystem.put_file(f, fpath, overwrite=True)\n                else:\n                    self.filesystem.put_file(f, fpath)\n\n                counter += 1\n\n        return counter\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        path = self._resolve_path(path)\n\n        with self.filesystem.open(path, \"rb\") as file:\n            content = await run_sync_in_worker_thread(file.read)\n\n        return content\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        path = self._resolve_path(path)\n        dirpath = path[: path.rindex(\"/\")]\n\n        self.filesystem.makedirs(dirpath, exist_ok=True)\n\n        with self.filesystem.open(path, \"wb\") as file:\n            await run_sync_in_worker_thread(file.write, content)\n        return path\n\n    @property\n    def filesystem(self) -> fsspec.AbstractFileSystem:\n        if not self._filesystem:\n            scheme, _, _, _, _ = urllib.parse.urlsplit(self.basepath)\n\n            try:\n                self._filesystem = fsspec.filesystem(scheme, **self.settings)\n            except ImportError as exc:\n                # The path is a remote file system that uses a lib that is not installed\n                raise RuntimeError(\n                    f\"File system created with scheme {scheme!r} from base path \"\n                    f\"{self.basepath!r} could not be created. \"\n                    \"You are likely missing a Python module required to use the given \"\n                    \"storage protocol.\"\n                ) from exc\n\n        return self._filesystem\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.RemoteFileSystem.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory.

    Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> None:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    if from_path is None:\n        from_path = str(self.basepath)\n    else:\n        from_path = self._resolve_path(from_path)\n\n    if local_path is None:\n        local_path = Path(\".\").absolute()\n\n    # validate that from_path has a trailing slash for proper fsspec behavior across versions\n    if not from_path.endswith(\"/\"):\n        from_path += \"/\"\n\n    return self.filesystem.get(from_path, local_path, recursive=True)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.RemoteFileSystem.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory.

    Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n    overwrite: bool = True,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    if to_path is None:\n        to_path = str(self.basepath)\n    else:\n        to_path = self._resolve_path(to_path)\n\n    if local_path is None:\n        local_path = \".\"\n\n    included_files = None\n    if ignore_file:\n        with open(ignore_file, \"r\") as f:\n            ignore_patterns = f.readlines()\n\n        included_files = filter_files(\n            local_path, ignore_patterns, include_dirs=True\n        )\n\n    counter = 0\n    for f in Path(local_path).rglob(\"*\"):\n        relative_path = f.relative_to(local_path)\n        if included_files and str(relative_path) not in included_files:\n            continue\n\n        if to_path.endswith(\"/\"):\n            fpath = to_path + relative_path.as_posix()\n        else:\n            fpath = to_path + \"/\" + relative_path.as_posix()\n\n        if f.is_dir():\n            pass\n        else:\n            f = f.as_posix()\n            if overwrite:\n                self.filesystem.put_file(f, fpath, overwrite=True)\n            else:\n                self.filesystem.put_file(f, fpath)\n\n            counter += 1\n\n    return counter\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.S3","title":"S3","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on AWS S3.

    Example

    Load stored S3 config:

    from prefect.filesystems import S3\n\ns3_block = S3.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class S3(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on AWS S3.\n\n    Example:\n        Load stored S3 config:\n        ```python\n        from prefect.filesystems import S3\n\n        s3_block = S3.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"S3\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/d74b16fe84ce626345adf235a47008fea2869a60-225x225.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#s3\"\n\n    bucket_path: str = Field(\n        default=...,\n        description=\"An S3 bucket path.\",\n        example=\"my-bucket/a-directory-within\",\n    )\n    aws_access_key_id: Optional[SecretStr] = Field(\n        default=None,\n        title=\"AWS Access Key ID\",\n        description=\"Equivalent to the AWS_ACCESS_KEY_ID environment variable.\",\n        example=\"AKIAIOSFODNN7EXAMPLE\",\n    )\n    aws_secret_access_key: Optional[SecretStr] = Field(\n        default=None,\n        title=\"AWS Secret Access Key\",\n        description=\"Equivalent to the AWS_SECRET_ACCESS_KEY environment variable.\",\n        example=\"wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY\",\n    )\n\n    _remote_file_system: RemoteFileSystem = None\n\n    @property\n    def basepath(self) -> str:\n        return f\"s3://{self.bucket_path}\"\n\n    @property\n    def filesystem(self) -> RemoteFileSystem:\n        settings = {}\n        if self.aws_access_key_id:\n            settings[\"key\"] = self.aws_access_key_id.get_secret_value()\n        if self.aws_secret_access_key:\n            settings[\"secret\"] = self.aws_secret_access_key.get_secret_value()\n        self._remote_file_system = RemoteFileSystem(\n            basepath=f\"s3://{self.bucket_path}\", settings=settings\n        )\n        return self._remote_file_system\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> bytes:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        return await self.filesystem.get_directory(\n            from_path=from_path, local_path=local_path\n        )\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        return await self.filesystem.put_directory(\n            local_path=local_path, to_path=to_path, ignore_file=ignore_file\n        )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        return await self.filesystem.read_path(path)\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        return await self.filesystem.write_path(path=path, content=content)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.S3.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory.

    Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> bytes:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    return await self.filesystem.get_directory(\n        from_path=from_path, local_path=local_path\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.S3.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory.

    Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    return await self.filesystem.put_directory(\n        local_path=local_path, to_path=to_path, ignore_file=ignore_file\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.SMB","title":"SMB","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on a SMB share.

    Example

    Load stored SMB config:

    from prefect.filesystems import SMB\nsmb_block = SMB.load(\"BLOCK_NAME\")\n
    Source code in prefect/filesystems.py
    class SMB(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on a SMB share.\n\n    Example:\n        Load stored SMB config:\n\n        ```python\n        from prefect.filesystems import SMB\n        smb_block = SMB.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"SMB\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/3f624663f7beb97d011d011bffd51ecf6c499efc-195x195.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#smb\"\n\n    share_path: str = Field(\n        default=...,\n        description=\"SMB target (requires <SHARE>, followed by <PATH>).\",\n        example=\"/SHARE/dir/subdir\",\n    )\n    smb_username: Optional[SecretStr] = Field(\n        default=None,\n        title=\"SMB Username\",\n        description=\"Username with access to the target SMB SHARE.\",\n    )\n    smb_password: Optional[SecretStr] = Field(\n        default=None, title=\"SMB Password\", description=\"Password for SMB access.\"\n    )\n    smb_host: str = Field(\n        default=..., tile=\"SMB server/hostname\", description=\"SMB server/hostname.\"\n    )\n    smb_port: Optional[int] = Field(\n        default=None, title=\"SMB port\", description=\"SMB port (default: 445).\"\n    )\n\n    _remote_file_system: RemoteFileSystem = None\n\n    @property\n    def basepath(self) -> str:\n        return f\"smb://{self.smb_host.rstrip('/')}/{self.share_path.lstrip('/')}\"\n\n    @property\n    def filesystem(self) -> RemoteFileSystem:\n        settings = {}\n        if self.smb_username:\n            settings[\"username\"] = self.smb_username.get_secret_value()\n        if self.smb_password:\n            settings[\"password\"] = self.smb_password.get_secret_value()\n        if self.smb_host:\n            settings[\"host\"] = self.smb_host\n        if self.smb_port:\n            settings[\"port\"] = self.smb_port\n        self._remote_file_system = RemoteFileSystem(\n            basepath=f\"smb://{self.smb_host.rstrip('/')}/{self.share_path.lstrip('/')}\",\n            settings=settings,\n        )\n        return self._remote_file_system\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> bytes:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        return await self.filesystem.get_directory(\n            from_path=from_path, local_path=local_path\n        )\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        return await self.filesystem.put_directory(\n            local_path=local_path,\n            to_path=to_path,\n            ignore_file=ignore_file,\n            overwrite=False,\n        )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        return await self.filesystem.read_path(path)\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        return await self.filesystem.write_path(path=path, content=content)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.SMB.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory. Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> bytes:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    return await self.filesystem.get_directory(\n        from_path=from_path, local_path=local_path\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.SMB.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory. Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    return await self.filesystem.put_directory(\n        local_path=local_path,\n        to_path=to_path,\n        ignore_file=ignore_file,\n        overwrite=False,\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/flows/","title":"prefect.flows","text":"","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows","title":"prefect.flows","text":"

    Module containing the base workflow class and decorator - for most use cases, using the @flow decorator is preferred.

    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow","title":"Flow","text":"

    Bases: Generic[P, R]

    A Prefect workflow definition.

    Note

    We recommend using the @flow decorator for most use-cases.

    Wraps a function with an entrypoint to the Prefect engine. To preserve the input and output types, we use the generic type variables P and R for \"Parameters\" and \"Returns\" respectively.

    Parameters:

    Name Type Description Default fn Callable[P, R]

    The function defining the workflow.

    required name Optional[str]

    An optional name for the flow; if not provided, the name will be inferred from the given function.

    None version Optional[str]

    An optional version string for the flow; if not provided, we will attempt to create a version string as a hash of the file containing the wrapped function; if the file cannot be located, the version will be null.

    None flow_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this flow; this name can be provided as a string template with the flow's parameters as variables, or a function that returns a string.

    None task_runner Union[Type[BaseTaskRunner], BaseTaskRunner]

    An optional task runner to use for task execution within the flow; if not provided, a ConcurrentTaskRunner will be used.

    ConcurrentTaskRunner description str

    An optional string description for the flow; if not provided, the description will be pulled from the docstring for the decorated function.

    None timeout_seconds Union[int, float]

    An optional number of seconds indicating a maximum runtime for the flow. If the flow exceeds this runtime, it will be marked as failed. Flow execution may continue until the next task is called.

    None validate_parameters bool

    By default, parameters passed to flows are validated by Pydantic. This will check that input values conform to the annotated types on the function. Where possible, values will be coerced into the correct type; for example, if a parameter is defined as x: int and \"5\" is passed, it will be resolved to 5. If set to False, no validation will be performed on flow parameters.

    True retries Optional[int]

    An optional number of times to retry on flow run failure.

    None retry_delay_seconds Optional[Union[int, float]]

    An optional number of seconds to wait before retrying the flow after failure. This is only applicable if retries is nonzero.

    None persist_result Optional[bool]

    An optional toggle indicating whether the result of this flow should be persisted to result storage. Defaults to None, which indicates that Prefect should choose whether the result should be persisted depending on the features being used.

    None result_storage Optional[ResultStorage]

    An optional block to use to persist the result of this flow. This value will be used as the default for any tasks in this flow. If not provided, the local file system will be used unless called as a subflow, at which point the default will be loaded from the parent flow.

    None result_serializer Optional[ResultSerializer]

    An optional serializer to use to serialize the result of this flow for persistence. This value will be used as the default for any tasks in this flow. If not provided, the value of PREFECT_RESULTS_DEFAULT_SERIALIZER will be used unless called as a subflow, at which point the default will be loaded from the parent flow.

    None on_failure Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of callables to run when the flow enters a failed state.

    None on_completion Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of callables to run when the flow enters a completed state.

    None on_cancellation Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of callables to run when the flow enters a cancelling state.

    None on_crashed Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of callables to run when the flow enters a crashed state.

    None Source code in prefect/flows.py
    @PrefectObjectRegistry.register_instances\nclass Flow(Generic[P, R]):\n    \"\"\"\n    A Prefect workflow definition.\n\n    !!! note\n        We recommend using the [`@flow` decorator][prefect.flows.flow] for most use-cases.\n\n    Wraps a function with an entrypoint to the Prefect engine. To preserve the input\n    and output types, we use the generic type variables `P` and `R` for \"Parameters\" and\n    \"Returns\" respectively.\n\n    Args:\n        fn: The function defining the workflow.\n        name: An optional name for the flow; if not provided, the name will be inferred\n            from the given function.\n        version: An optional version string for the flow; if not provided, we will\n            attempt to create a version string as a hash of the file containing the\n            wrapped function; if the file cannot be located, the version will be null.\n        flow_run_name: An optional name to distinguish runs of this flow; this name can\n            be provided as a string template with the flow's parameters as variables,\n            or a function that returns a string.\n        task_runner: An optional task runner to use for task execution within the flow;\n            if not provided, a `ConcurrentTaskRunner` will be used.\n        description: An optional string description for the flow; if not provided, the\n            description will be pulled from the docstring for the decorated function.\n        timeout_seconds: An optional number of seconds indicating a maximum runtime for\n            the flow. If the flow exceeds this runtime, it will be marked as failed.\n            Flow execution may continue until the next task is called.\n        validate_parameters: By default, parameters passed to flows are validated by\n            Pydantic. This will check that input values conform to the annotated types\n            on the function. Where possible, values will be coerced into the correct\n            type; for example, if a parameter is defined as `x: int` and \"5\" is passed,\n            it will be resolved to `5`. If set to `False`, no validation will be\n            performed on flow parameters.\n        retries: An optional number of times to retry on flow run failure.\n        retry_delay_seconds: An optional number of seconds to wait before retrying the\n            flow after failure. This is only applicable if `retries` is nonzero.\n        persist_result: An optional toggle indicating whether the result of this flow\n            should be persisted to result storage. Defaults to `None`, which indicates\n            that Prefect should choose whether the result should be persisted depending on\n            the features being used.\n        result_storage: An optional block to use to persist the result of this flow.\n            This value will be used as the default for any tasks in this flow.\n            If not provided, the local file system will be used unless called as\n            a subflow, at which point the default will be loaded from the parent flow.\n        result_serializer: An optional serializer to use to serialize the result of this\n            flow for persistence. This value will be used as the default for any tasks\n            in this flow. If not provided, the value of `PREFECT_RESULTS_DEFAULT_SERIALIZER`\n            will be used unless called as a subflow, at which point the default will be\n            loaded from the parent flow.\n        on_failure: An optional list of callables to run when the flow enters a failed state.\n        on_completion: An optional list of callables to run when the flow enters a completed state.\n        on_cancellation: An optional list of callables to run when the flow enters a cancelling state.\n        on_crashed: An optional list of callables to run when the flow enters a crashed state.\n    \"\"\"\n\n    # NOTE: These parameters (types, defaults, and docstrings) should be duplicated\n    #       exactly in the @flow decorator\n    def __init__(\n        self,\n        fn: Callable[P, R],\n        name: Optional[str] = None,\n        version: Optional[str] = None,\n        flow_run_name: Optional[Union[Callable[[], str], str]] = None,\n        retries: Optional[int] = None,\n        retry_delay_seconds: Optional[Union[int, float]] = None,\n        task_runner: Union[Type[BaseTaskRunner], BaseTaskRunner] = ConcurrentTaskRunner,\n        description: str = None,\n        timeout_seconds: Union[int, float] = None,\n        validate_parameters: bool = True,\n        persist_result: Optional[bool] = None,\n        result_storage: Optional[ResultStorage] = None,\n        result_serializer: Optional[ResultSerializer] = None,\n        cache_result_in_memory: bool = True,\n        log_prints: Optional[bool] = None,\n        on_completion: Optional[\n            List[Callable[[FlowSchema, FlowRun, State], None]]\n        ] = None,\n        on_failure: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n        on_cancellation: Optional[\n            List[Callable[[FlowSchema, FlowRun, State], None]]\n        ] = None,\n        on_crashed: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    ):\n        if name is not None and not isinstance(name, str):\n            raise TypeError(\n                \"Expected string for flow parameter 'name'; got {} instead. {}\".format(\n                    type(name).__name__,\n                    (\n                        \"Perhaps you meant to call it? e.g.\"\n                        \" '@flow(name=get_flow_run_name())'\"\n                        if callable(name)\n                        else \"\"\n                    ),\n                )\n            )\n\n        # Validate if hook passed is list and contains callables\n        hook_categories = [on_completion, on_failure, on_cancellation, on_crashed]\n        hook_names = [\"on_completion\", \"on_failure\", \"on_cancellation\", \"on_crashed\"]\n        for hooks, hook_name in zip(hook_categories, hook_names):\n            if hooks is not None:\n                if not hooks:\n                    raise ValueError(f\"Empty list passed for '{hook_name}'\")\n                try:\n                    hooks = list(hooks)\n                except TypeError:\n                    raise TypeError(\n                        f\"Expected iterable for '{hook_name}'; got\"\n                        f\" {type(hooks).__name__} instead. Please provide a list of\"\n                        f\" hooks to '{hook_name}':\\n\\n\"\n                        f\"@flow({hook_name}=[hook1, hook2])\\ndef\"\n                        \" my_flow():\\n\\tpass\"\n                    )\n\n                for hook in hooks:\n                    if not callable(hook):\n                        raise TypeError(\n                            f\"Expected callables in '{hook_name}'; got\"\n                            f\" {type(hook).__name__} instead. Please provide a list of\"\n                            f\" hooks to '{hook_name}':\\n\\n\"\n                            f\"@flow({hook_name}=[hook1, hook2])\\ndef\"\n                            \" my_flow():\\n\\tpass\"\n                        )\n\n        if not callable(fn):\n            raise TypeError(\"'fn' must be callable\")\n\n        # Validate name if given\n        if name:\n            raise_on_name_with_banned_characters(name)\n\n        self.name = name or fn.__name__.replace(\"_\", \"-\")\n\n        if flow_run_name is not None:\n            if not isinstance(flow_run_name, str) and not callable(flow_run_name):\n                raise TypeError(\n                    \"Expected string or callable for 'flow_run_name'; got\"\n                    f\" {type(flow_run_name).__name__} instead.\"\n                )\n        self.flow_run_name = flow_run_name\n\n        task_runner = task_runner or ConcurrentTaskRunner()\n        self.task_runner = (\n            task_runner() if isinstance(task_runner, type) else task_runner\n        )\n\n        self.log_prints = log_prints\n\n        self.description = description or inspect.getdoc(fn)\n        update_wrapper(self, fn)\n        self.fn = fn\n        self.isasync = is_async_fn(self.fn)\n\n        raise_for_reserved_arguments(self.fn, [\"return_state\", \"wait_for\"])\n\n        # Version defaults to a hash of the function's file\n        flow_file = inspect.getsourcefile(self.fn)\n        if not version:\n            try:\n                version = file_hash(flow_file)\n            except (FileNotFoundError, TypeError, OSError):\n                pass  # `getsourcefile` can return null values and \"<stdin>\" for objects in repls\n        self.version = version\n\n        self.timeout_seconds = float(timeout_seconds) if timeout_seconds else None\n\n        # FlowRunPolicy settings\n        # TODO: We can instantiate a `FlowRunPolicy` and add Pydantic bound checks to\n        #       validate that the user passes positive numbers here\n        self.retries = (\n            retries if retries is not None else PREFECT_FLOW_DEFAULT_RETRIES.value()\n        )\n\n        self.retry_delay_seconds = (\n            retry_delay_seconds\n            if retry_delay_seconds is not None\n            else PREFECT_FLOW_DEFAULT_RETRY_DELAY_SECONDS.value()\n        )\n\n        self.parameters = parameter_schema(self.fn)\n        self.should_validate_parameters = validate_parameters\n\n        if self.should_validate_parameters:\n            # Try to create the validated function now so that incompatibility can be\n            # raised at declaration time rather than at runtime\n            # We cannot, however, store the validated function on the flow because it\n            # is not picklable in some environments\n            try:\n                ValidatedFunction(self.fn, config={\"arbitrary_types_allowed\": True})\n            except pydantic.ConfigError as exc:\n                raise ValueError(\n                    \"Flow function is not compatible with `validate_parameters`. \"\n                    \"Disable validation or change the argument names.\"\n                ) from exc\n\n        self.persist_result = persist_result\n        self.result_storage = result_storage\n        self.result_serializer = result_serializer\n        self.cache_result_in_memory = cache_result_in_memory\n\n        # Check for collision in the registry\n        registry = PrefectObjectRegistry.get()\n\n        if registry and any(\n            other\n            for other in registry.get_instances(Flow)\n            if other.name == self.name and id(other.fn) != id(self.fn)\n        ):\n            file = inspect.getsourcefile(self.fn)\n            line_number = inspect.getsourcelines(self.fn)[1]\n            warnings.warn(\n                f\"A flow named {self.name!r} and defined at '{file}:{line_number}' \"\n                \"conflicts with another flow. Consider specifying a unique `name` \"\n                \"parameter in the flow definition:\\n\\n \"\n                \"`@flow(name='my_unique_name', ...)`\"\n            )\n        self.on_completion = on_completion\n        self.on_failure = on_failure\n        self.on_cancellation = on_cancellation\n        self.on_crashed = on_crashed\n\n        # Used for flows loaded from remote storage\n        self._storage: Optional[RunnerStorage] = None\n        self._entrypoint: Optional[str] = None\n\n    def with_options(\n        self,\n        *,\n        name: str = None,\n        version: str = None,\n        retries: int = 0,\n        retry_delay_seconds: Union[int, float] = 0,\n        description: str = None,\n        flow_run_name: Optional[Union[Callable[[], str], str]] = None,\n        task_runner: Union[Type[BaseTaskRunner], BaseTaskRunner] = None,\n        timeout_seconds: Union[int, float] = None,\n        validate_parameters: bool = None,\n        persist_result: Optional[bool] = NotSet,\n        result_storage: Optional[ResultStorage] = NotSet,\n        result_serializer: Optional[ResultSerializer] = NotSet,\n        cache_result_in_memory: bool = None,\n        log_prints: Optional[bool] = NotSet,\n        on_completion: Optional[\n            List[Callable[[FlowSchema, FlowRun, State], None]]\n        ] = None,\n        on_failure: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n        on_cancellation: Optional[\n            List[Callable[[FlowSchema, FlowRun, State], None]]\n        ] = None,\n        on_crashed: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    ) -> Self:\n        \"\"\"\n        Create a new flow from the current object, updating provided options.\n\n        Args:\n            name: A new name for the flow.\n            version: A new version for the flow.\n            description: A new description for the flow.\n            flow_run_name: An optional name to distinguish runs of this flow; this name\n                can be provided as a string template with the flow's parameters as variables,\n                or a function that returns a string.\n            task_runner: A new task runner for the flow.\n            timeout_seconds: A new number of seconds to fail the flow after if still\n                running.\n            validate_parameters: A new value indicating if flow calls should validate\n                given parameters.\n            retries: A new number of times to retry on flow run failure.\n            retry_delay_seconds: A new number of seconds to wait before retrying the\n                flow after failure. This is only applicable if `retries` is nonzero.\n            persist_result: A new option for enabling or disabling result persistence.\n            result_storage: A new storage type to use for results.\n            result_serializer: A new serializer to use for results.\n            cache_result_in_memory: A new value indicating if the flow's result should\n                be cached in memory.\n            on_failure: A new list of callables to run when the flow enters a failed state.\n            on_completion: A new list of callables to run when the flow enters a completed state.\n            on_cancellation: A new list of callables to run when the flow enters a cancelling state.\n            on_crashed: A new list of callables to run when the flow enters a crashed state.\n\n        Returns:\n            A new `Flow` instance.\n\n        Examples:\n\n            Create a new flow from an existing flow and update the name:\n\n            >>> @flow(name=\"My flow\")\n            >>> def my_flow():\n            >>>     return 1\n            >>>\n            >>> new_flow = my_flow.with_options(name=\"My new flow\")\n\n            Create a new flow from an existing flow, update the task runner, and call\n            it without an intermediate variable:\n\n            >>> from prefect.task_runners import SequentialTaskRunner\n            >>>\n            >>> @flow\n            >>> def my_flow(x, y):\n            >>>     return x + y\n            >>>\n            >>> state = my_flow.with_options(task_runner=SequentialTaskRunner)(1, 3)\n            >>> assert state.result() == 4\n\n        \"\"\"\n        new_flow = Flow(\n            fn=self.fn,\n            name=name or self.name,\n            description=description or self.description,\n            flow_run_name=flow_run_name,\n            version=version or self.version,\n            task_runner=task_runner or self.task_runner,\n            retries=retries or self.retries,\n            retry_delay_seconds=retry_delay_seconds or self.retry_delay_seconds,\n            timeout_seconds=(\n                timeout_seconds if timeout_seconds is not None else self.timeout_seconds\n            ),\n            validate_parameters=(\n                validate_parameters\n                if validate_parameters is not None\n                else self.should_validate_parameters\n            ),\n            persist_result=(\n                persist_result if persist_result is not NotSet else self.persist_result\n            ),\n            result_storage=(\n                result_storage if result_storage is not NotSet else self.result_storage\n            ),\n            result_serializer=(\n                result_serializer\n                if result_serializer is not NotSet\n                else self.result_serializer\n            ),\n            cache_result_in_memory=(\n                cache_result_in_memory\n                if cache_result_in_memory is not None\n                else self.cache_result_in_memory\n            ),\n            log_prints=log_prints if log_prints is not NotSet else self.log_prints,\n            on_completion=on_completion or self.on_completion,\n            on_failure=on_failure or self.on_failure,\n            on_cancellation=on_cancellation or self.on_cancellation,\n            on_crashed=on_crashed or self.on_crashed,\n        )\n        new_flow._storage = self._storage\n        new_flow._entrypoint = self._entrypoint\n        return new_flow\n\n    def validate_parameters(self, parameters: Dict[str, Any]) -> Dict[str, Any]:\n        \"\"\"\n        Validate parameters for compatibility with the flow by attempting to cast the inputs to the\n        associated types specified by the function's type annotations.\n\n        Returns:\n            A new dict of parameters that have been cast to the appropriate types\n\n        Raises:\n            ParameterTypeError: if the provided parameters are not valid\n        \"\"\"\n        args, kwargs = parameters_to_args_kwargs(self.fn, parameters)\n\n        if HAS_PYDANTIC_V2:\n            has_v1_models = any(isinstance(o, V1BaseModel) for o in args) or any(\n                isinstance(o, V1BaseModel) for o in kwargs.values()\n            )\n            has_v2_models = any(isinstance(o, V2BaseModel) for o in args) or any(\n                isinstance(o, V2BaseModel) for o in kwargs.values()\n            )\n\n            if has_v1_models and has_v2_models:\n                raise ParameterTypeError(\n                    \"Cannot mix Pydantic v1 and v2 models as arguments to a flow.\"\n                )\n\n            if has_v1_models:\n                validated_fn = V1ValidatedFunction(\n                    self.fn, config={\"arbitrary_types_allowed\": True}\n                )\n            else:\n                validated_fn = V2ValidatedFunction(\n                    self.fn, config={\"arbitrary_types_allowed\": True}\n                )\n\n        else:\n            validated_fn = ValidatedFunction(\n                self.fn, config={\"arbitrary_types_allowed\": True}\n            )\n\n        try:\n            model = validated_fn.init_model_instance(*args, **kwargs)\n        except pydantic.ValidationError as exc:\n            # We capture the pydantic exception and raise our own because the pydantic\n            # exception is not picklable when using a cythonized pydantic installation\n            raise ParameterTypeError.from_validation_error(exc) from None\n        except V2ValidationError as exc:\n            # We capture the pydantic exception and raise our own because the pydantic\n            # exception is not picklable when using a cythonized pydantic installation\n            raise ParameterTypeError.from_validation_error(exc) from None\n\n        # Get the updated parameter dict with cast values from the model\n        cast_parameters = {\n            k: v\n            for k, v in model._iter()\n            if k in model.__fields_set__ or model.__fields__[k].default_factory\n        }\n        return cast_parameters\n\n    def serialize_parameters(self, parameters: Dict[str, Any]) -> Dict[str, Any]:\n        \"\"\"\n        Convert parameters to a serializable form.\n\n        Uses FastAPI's `jsonable_encoder` to convert to JSON compatible objects without\n        converting everything directly to a string. This maintains basic types like\n        integers during API roundtrips.\n        \"\"\"\n        serialized_parameters = {}\n        for key, value in parameters.items():\n            try:\n                serialized_parameters[key] = jsonable_encoder(value)\n            except (TypeError, ValueError):\n                logger.debug(\n                    f\"Parameter {key!r} for flow {self.name!r} is of unserializable \"\n                    f\"type {type(value).__name__!r} and will not be stored \"\n                    \"in the backend.\"\n                )\n                serialized_parameters[key] = f\"<{type(value).__name__}>\"\n        return serialized_parameters\n\n    @sync_compatible\n    async def to_deployment(\n        self,\n        name: str,\n        interval: Optional[Union[int, float, datetime.timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        work_pool_name: Optional[str] = None,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[Dict[str, Any]] = None,\n    ):\n        \"\"\"\n        Creates a runner deployment object for this flow.\n\n        Args:\n            name: The name to give the created deployment.\n            interval: An interval on which to execute the new deployment. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this deployment.\n            rrule: An rrule schedule of when to execute runs of this deployment.\n            timezone: A timezone to use for the schedule. Defaults to UTC.\n            triggers: A list of triggers that will kick off runs of this deployment.\n            schedule: A schedule object defining when to execute runs of this deployment.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            parameters: A dictionary of default parameter values to pass to runs of this deployment.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for the created deployment.\n            work_pool_name: The name of the work pool to use for this deployment.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n\n        Examples:\n            Prepare two deployments and serve them:\n\n            ```python\n            from prefect import flow, serve\n\n            @flow\n            def my_flow(name):\n                print(f\"hello {name}\")\n\n            @flow\n            def my_other_flow(name):\n                print(f\"goodbye {name}\")\n\n            if __name__ == \"__main__\":\n                hello_deploy = my_flow.to_deployment(\"hello\", tags=[\"dev\"])\n                bye_deploy = my_other_flow.to_deployment(\"goodbye\", tags=[\"dev\"])\n                serve(hello_deploy, bye_deploy)\n            ```\n        \"\"\"\n        from prefect.deployments.runner import RunnerDeployment\n\n        if not name.endswith(\".py\"):\n            raise_on_name_with_banned_characters(name)\n        if self._storage and self._entrypoint:\n            return await RunnerDeployment.from_storage(\n                storage=self._storage,\n                entrypoint=self._entrypoint,\n                name=name,\n                interval=interval,\n                cron=cron,\n                rrule=rrule,\n                schedule=schedule,\n                is_schedule_active=is_schedule_active,\n                tags=tags,\n                triggers=triggers,\n                parameters=parameters or {},\n                description=description,\n                version=version,\n                enforce_parameter_schema=enforce_parameter_schema,\n                work_pool_name=work_pool_name,\n                work_queue_name=work_queue_name,\n                job_variables=job_variables,\n            )\n        else:\n            return RunnerDeployment.from_flow(\n                self,\n                name=name,\n                interval=interval,\n                cron=cron,\n                rrule=rrule,\n                schedule=schedule,\n                is_schedule_active=is_schedule_active,\n                tags=tags,\n                triggers=triggers,\n                parameters=parameters or {},\n                description=description,\n                version=version,\n                enforce_parameter_schema=enforce_parameter_schema,\n                work_pool_name=work_pool_name,\n                work_queue_name=work_queue_name,\n                job_variables=job_variables,\n            )\n\n    @sync_compatible\n    async def serve(\n        self,\n        name: str,\n        interval: Optional[Union[int, float, datetime.timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        parameters: Optional[dict] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        pause_on_shutdown: bool = True,\n        print_starting_message: bool = True,\n        webserver: bool = False,\n    ):\n        \"\"\"\n        Creates a deployment for this flow and starts a runner to monitor for scheduled work.\n\n        Args:\n            name: The name to give the created deployment.\n            interval: An interval on which to execute the new deployment. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this deployment.\n            rrule: An rrule schedule of when to execute runs of this deployment.\n            triggers: A list of triggers that will kick off runs of this deployment.\n            schedule: A schedule object defining when to execute runs of this deployment. Used to\n                define additional scheduling options like `timezone`.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            parameters: A dictionary of default parameter values to pass to runs of this deployment.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for the created deployment.\n            pause_on_shutdown: If True, provided schedule will be paused when the serve function is stopped.\n                If False, the schedules will continue running.\n            print_starting_message: Whether or not to print the starting message when flow is served.\n            webserver: Whether or not to start a monitoring webserver for this flow.\n\n        Examples:\n            Serve a flow:\n\n            ```python\n            from prefect import flow\n\n            @flow\n            def my_flow(name):\n                print(f\"hello {name}\")\n\n            if __name__ == \"__main__\":\n                my_flow.serve(\"example-deployment\")\n            ```\n\n            Serve a flow and run it every hour:\n\n            ```python\n            from prefect import flow\n\n            @flow\n            def my_flow(name):\n                print(f\"hello {name}\")\n\n            if __name__ == \"__main__\":\n                my_flow.serve(\"example-deployment\", interval=3600)\n            ```\n        \"\"\"\n        from prefect.runner import Runner\n\n        # Handling for my_flow.serve(__file__)\n        # Will set name to name of file where my_flow.serve() without the extension\n        # Non filepath strings will pass through unchanged\n        name = Path(name).stem\n\n        runner = Runner(name=name, pause_on_shutdown=pause_on_shutdown)\n        deployment_id = await runner.add_flow(\n            self,\n            name=name,\n            triggers=triggers,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            parameters=parameters,\n            description=description,\n            tags=tags,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n        )\n        if print_starting_message:\n            help_message = (\n                f\"[green]Your flow {self.name!r} is being served and polling for\"\n                \" scheduled runs!\\n[/]\\nTo trigger a run for this flow, use the\"\n                \" following command:\\n[blue]\\n\\t$ prefect deployment run\"\n                f\" '{self.name}/{name}'\\n[/]\"\n            )\n            if PREFECT_UI_URL:\n                help_message += (\n                    \"\\nYou can also run your flow via the Prefect UI:\"\n                    f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_id}[/]\\n\"\n                )\n\n            console = Console()\n            console.print(Panel(help_message))\n        await runner.start(webserver=webserver)\n\n    @classmethod\n    @sync_compatible\n    async def from_source(\n        cls: Type[F],\n        source: Union[str, RunnerStorage, ReadableDeploymentStorage],\n        entrypoint: str,\n    ) -> F:\n        \"\"\"\n        Loads a flow from a remote s ource.\n\n        Args:\n            source: Either a URL to a git repository or a storage object.\n            entrypoint:  The path to a file containing a flow and the name of the flow function in\n                the format `./path/to/file.py:flow_func_name`.\n\n        Returns:\n            A new `Flow` instance.\n\n        Examples:\n            Load a flow from a public git repository:\n\n\n            ```python\n            from prefect import flow\n            from prefect.runner.storage import GitRepository\n            from prefect.blocks.system import Secret\n\n            my_flow = flow.from_source(\n                source=\"https://github.com/org/repo.git\",\n                entrypoint=\"flows.py:my_flow\",\n            )\n\n            my_flow()\n            ```\n\n            Load a flow from a private git repository:\n\n            ```python\n            from prefect import flow\n            from prefect.runner.storage import GitRepository\n            from prefect.blocks.system import Secret\n\n            my_flow = flow.from_source(\n                source=GitRepository(\n                    url=\"https://github.com/org/repo.git\",\n                    access_token=Secret.load(\"github-access-token\").get(),\n                ),\n                entrypoint=\"flows.py:my_flow\",\n            )\n\n            my_flow()\n            ```\n        \"\"\"\n        if isinstance(source, str):\n            storage = create_storage_from_url(source)\n        elif isinstance(source, RunnerStorage):\n            storage = source\n        elif hasattr(source, \"get_directory\"):\n            storage = BlockStorageAdapter(source)\n        else:\n            raise TypeError(\n                f\"Unsupported source type {type(source).__name__!r}. Please provide a\"\n                \" URL to remote storage or a storage object.\"\n            )\n        with tempfile.TemporaryDirectory() as tmpdir:\n            storage.set_base_path(Path(tmpdir))\n            await storage.pull_code()\n\n            full_entrypoint = str(storage.destination / entrypoint)\n            flow: \"Flow\" = await from_async.wait_for_call_in_new_thread(\n                create_call(load_flow_from_entrypoint, full_entrypoint)\n            )\n            flow._storage = storage\n            flow._entrypoint = entrypoint\n\n        return flow\n\n    @sync_compatible\n    async def deploy(\n        self,\n        name: str,\n        work_pool_name: Optional[str] = None,\n        image: Optional[Union[str, DeploymentImage]] = None,\n        build: bool = True,\n        push: bool = True,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[dict] = None,\n        interval: Optional[Union[int, float, datetime.timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        parameters: Optional[dict] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        print_next_steps: bool = True,\n    ) -> UUID:\n        \"\"\"\n        Deploys a flow to run on dynamic infrastructure via a work pool.\n\n        By default, calling this method will build a Docker image for the flow, push it to a registry,\n        and create a deployment via the Prefect API that will run the flow on the given schedule.\n\n        If you want to use an existing image, you can pass `build=False` to skip building and pushing\n        an image.\n\n        Args:\n            name: The name to give the created deployment.\n            work_pool_name: The name of the work pool to use for this deployment. Defaults to\n                the value of `PREFECT_DEFAULT_WORK_POOL_NAME`.\n            image: The name of the Docker image to build, including the registry and\n                repository. Pass a DeploymentImage instance to customize the Dockerfile used\n                and build arguments.\n            build: Whether or not to build a new image for the flow. If False, the provided\n                image will be used as-is and pulled at runtime.\n            push: Whether or not to skip pushing the built image to a registry.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n                available settings.\n            interval: An interval on which to execute the new deployment. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this deployment.\n            rrule: An rrule schedule of when to execute runs of this deployment.\n            triggers: A list of triggers that will kick off runs of this deployment.\n            schedule: A schedule object defining when to execute runs of this deployment. Used to\n                define additional scheduling options like `timezone`.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            parameters: A dictionary of default parameter values to pass to runs of this deployment.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for the created deployment.\n            print_next_steps_message: Whether or not to print a message with next steps\n            after deploying the deployments.\n\n        Returns:\n            The ID of the created/updated deployment.\n\n        Examples:\n            Deploy a local flow to a work pool:\n\n            ```python\n            from prefect import flow\n\n            @flow\n            def my_flow(name):\n                print(f\"hello {name}\")\n\n            if __name__ == \"__main__\":\n                my_flow.deploy(\n                    \"example-deployment\",\n                    work_pool_name=\"my-work-pool\",\n                    image=\"my-repository/my-image:dev\",\n                )\n            ```\n\n            Deploy a remotely stored flow to a work pool:\n\n            ```python\n            from prefect import flow\n\n            if __name__ == \"__main__\":\n                flow.from_source(\n                    source=\"https://github.com/org/repo.git\",\n                    entrypoint=\"flows.py:my_flow\",\n                ).deploy(\n                    \"example-deployment\",\n                    work_pool_name=\"my-work-pool\",\n                    image=\"my-repository/my-image:dev\",\n                )\n            ```\n        \"\"\"\n        work_pool_name = work_pool_name or PREFECT_DEFAULT_WORK_POOL_NAME.value()\n\n        try:\n            async with get_client() as client:\n                work_pool = await client.read_work_pool(work_pool_name)\n        except ObjectNotFound as exc:\n            raise ValueError(\n                f\"Could not find work pool {work_pool_name!r}. Please create it before\"\n                \" deploying this flow.\"\n            ) from exc\n\n        deployment = await self.to_deployment(\n            name=name,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            triggers=triggers,\n            parameters=parameters,\n            description=description,\n            tags=tags,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n\n        deployment_ids = await deploy(\n            deployment,\n            work_pool_name=work_pool_name,\n            image=image,\n            build=build,\n            push=push,\n            print_next_steps_message=False,\n        )\n\n        if print_next_steps:\n            console = Console()\n            if not work_pool.is_push_pool and not work_pool.is_managed_pool:\n                console.print(\n                    \"\\nTo execute flow runs from this deployment, start a worker in a\"\n                    \" separate terminal that pulls work from the\"\n                    f\" {work_pool_name!r} work pool:\"\n                )\n                console.print(\n                    f\"\\n\\t$ prefect worker start --pool {work_pool_name!r}\",\n                    style=\"blue\",\n                )\n            console.print(\n                \"\\nTo schedule a run for this deployment, use the following command:\"\n            )\n            console.print(\n                f\"\\n\\t$ prefect deployment run '{self.name}/{name}'\\n\",\n                style=\"blue\",\n            )\n            if PREFECT_UI_URL:\n                console.print(\n                    \"\\nYou can also run your flow via the Prefect UI:\"\n                    f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_ids[0]}[/]\\n\"\n                )\n\n        return deployment_ids[0]\n\n    @overload\n    def __call__(self: \"Flow[P, NoReturn]\", *args: P.args, **kwargs: P.kwargs) -> None:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def __call__(\n        self: \"Flow[P, Coroutine[Any, Any, T]]\", *args: P.args, **kwargs: P.kwargs\n    ) -> Awaitable[T]:\n        ...\n\n    @overload\n    def __call__(\n        self: \"Flow[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> T:\n        ...\n\n    @overload\n    def __call__(\n        self: \"Flow[P, T]\",\n        *args: P.args,\n        return_state: Literal[True],\n        **kwargs: P.kwargs,\n    ) -> State[T]:\n        ...\n\n    def __call__(\n        self,\n        *args: \"P.args\",\n        return_state: bool = False,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: \"P.kwargs\",\n    ):\n        \"\"\"\n        Run the flow and return its result.\n\n\n        Flow parameter values must be serializable by Pydantic.\n\n        If writing an async flow, this call must be awaited.\n\n        This will create a new flow run in the API.\n\n        Args:\n            *args: Arguments to run the flow with.\n            return_state: Return a Prefect State containing the result of the\n                flow run.\n            wait_for: Upstream task futures to wait for before starting the flow if called as a subflow\n            **kwargs: Keyword arguments to run the flow with.\n\n        Returns:\n            If `return_state` is False, returns the result of the flow run.\n            If `return_state` is True, returns the result of the flow run\n                wrapped in a Prefect State which provides error handling.\n\n        Examples:\n\n            Define a flow\n\n            >>> @flow\n            >>> def my_flow(name):\n            >>>     print(f\"hello {name}\")\n            >>>     return f\"goodbye {name}\"\n\n            Run a flow\n\n            >>> my_flow(\"marvin\")\n            hello marvin\n            \"goodbye marvin\"\n\n            Run a flow with additional tags\n\n            >>> from prefect import tags\n            >>> with tags(\"db\", \"blue\"):\n            >>>     my_flow(\"foo\")\n        \"\"\"\n        from prefect.engine import enter_flow_run_engine_from_flow_call\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n\n        return_type = \"state\" if return_state else \"result\"\n\n        task_viz_tracker = get_task_viz_tracker()\n        if task_viz_tracker:\n            # this is a subflow, for now return a single task and do not go further\n            # we can add support for exploring subflows for tasks in the future.\n            return track_viz_task(self.isasync, self.name, parameters)\n\n        return enter_flow_run_engine_from_flow_call(\n            self,\n            parameters,\n            wait_for=wait_for,\n            return_type=return_type,\n        )\n\n    @overload\n    def _run(self: \"Flow[P, NoReturn]\", *args: P.args, **kwargs: P.kwargs) -> State[T]:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def _run(\n        self: \"Flow[P, Coroutine[Any, Any, T]]\", *args: P.args, **kwargs: P.kwargs\n    ) -> Awaitable[T]:\n        ...\n\n    @overload\n    def _run(self: \"Flow[P, T]\", *args: P.args, **kwargs: P.kwargs) -> State[T]:\n        ...\n\n    def _run(\n        self,\n        *args: \"P.args\",\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: \"P.kwargs\",\n    ):\n        \"\"\"\n        Run the flow and return its final state.\n\n        Examples:\n\n            Run a flow and get the returned result\n\n            >>> state = my_flow._run(\"marvin\")\n            >>> state.result()\n           \"goodbye marvin\"\n        \"\"\"\n        from prefect.engine import enter_flow_run_engine_from_flow_call\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n\n        return enter_flow_run_engine_from_flow_call(\n            self,\n            parameters,\n            wait_for=wait_for,\n            return_type=\"state\",\n        )\n\n    @sync_compatible\n    async def visualize(self, *args, **kwargs):\n        \"\"\"\n        Generates a graphviz object representing the current flow. In IPython notebooks,\n        it's rendered inline, otherwise in a new window as a PNG.\n\n        Raises:\n            - ImportError: If `graphviz` isn't installed.\n            - GraphvizExecutableNotFoundError: If the `dot` executable isn't found.\n            - FlowVisualizationError: If the flow can't be visualized for any other reason.\n        \"\"\"\n        if not PREFECT_UNIT_TEST_MODE:\n            warnings.warn(\n                \"`flow.visualize()` will execute code inside of your flow that is not\"\n                \" decorated with `@task` or `@flow`.\"\n            )\n\n        try:\n            with TaskVizTracker() as tracker:\n                if self.isasync:\n                    await self.fn(*args, **kwargs)\n                else:\n                    self.fn(*args, **kwargs)\n\n                graph = build_task_dependencies(tracker)\n\n                visualize_task_dependencies(graph, self.name)\n\n        except GraphvizImportError:\n            raise\n        except GraphvizExecutableNotFoundError:\n            raise\n        except VisualizationUnsupportedError:\n            raise\n        except FlowVisualizationError:\n            raise\n        except Exception as e:\n            msg = (\n                \"It's possible you are trying to visualize a flow that contains \"\n                \"code that directly interacts with the result of a task\"\n                \" inside of the flow. \\nTry passing a `viz_return_value` \"\n                \"to the task decorator, e.g. `@task(viz_return_value=[1, 2, 3]).`\"\n            )\n\n            new_exception = type(e)(str(e) + \"\\n\" + msg)\n            # Copy traceback information from the original exception\n            new_exception.__traceback__ = e.__traceback__\n            raise new_exception\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.deploy","title":"deploy async","text":"

    Deploys a flow to run on dynamic infrastructure via a work pool.

    By default, calling this method will build a Docker image for the flow, push it to a registry, and create a deployment via the Prefect API that will run the flow on the given schedule.

    If you want to use an existing image, you can pass build=False to skip building and pushing an image.

    Parameters:

    Name Type Description Default name str

    The name to give the created deployment.

    required work_pool_name Optional[str]

    The name of the work pool to use for this deployment. Defaults to the value of PREFECT_DEFAULT_WORK_POOL_NAME.

    None image Optional[Union[str, DeploymentImage]]

    The name of the Docker image to build, including the registry and repository. Pass a DeploymentImage instance to customize the Dockerfile used and build arguments.

    None build bool

    Whether or not to build a new image for the flow. If False, the provided image will be used as-is and pulled at runtime.

    True push bool

    Whether or not to skip pushing the built image to a registry.

    True work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[dict]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    None interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the new deployment. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this deployment.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this deployment.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that will kick off runs of this deployment.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object defining when to execute runs of this deployment. Used to define additional scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this deployment.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for the created deployment.

    False print_next_steps_message

    Whether or not to print a message with next steps

    required

    Returns:

    Type Description UUID

    The ID of the created/updated deployment.

    Examples:

    Deploy a local flow to a work pool:

    from prefect import flow\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\nif __name__ == \"__main__\":\n    my_flow.deploy(\n        \"example-deployment\",\n        work_pool_name=\"my-work-pool\",\n        image=\"my-repository/my-image:dev\",\n    )\n

    Deploy a remotely stored flow to a work pool:

    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=\"https://github.com/org/repo.git\",\n        entrypoint=\"flows.py:my_flow\",\n    ).deploy(\n        \"example-deployment\",\n        work_pool_name=\"my-work-pool\",\n        image=\"my-repository/my-image:dev\",\n    )\n
    Source code in prefect/flows.py
    @sync_compatible\nasync def deploy(\n    self,\n    name: str,\n    work_pool_name: Optional[str] = None,\n    image: Optional[Union[str, DeploymentImage]] = None,\n    build: bool = True,\n    push: bool = True,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[dict] = None,\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    parameters: Optional[dict] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    print_next_steps: bool = True,\n) -> UUID:\n    \"\"\"\n    Deploys a flow to run on dynamic infrastructure via a work pool.\n\n    By default, calling this method will build a Docker image for the flow, push it to a registry,\n    and create a deployment via the Prefect API that will run the flow on the given schedule.\n\n    If you want to use an existing image, you can pass `build=False` to skip building and pushing\n    an image.\n\n    Args:\n        name: The name to give the created deployment.\n        work_pool_name: The name of the work pool to use for this deployment. Defaults to\n            the value of `PREFECT_DEFAULT_WORK_POOL_NAME`.\n        image: The name of the Docker image to build, including the registry and\n            repository. Pass a DeploymentImage instance to customize the Dockerfile used\n            and build arguments.\n        build: Whether or not to build a new image for the flow. If False, the provided\n            image will be used as-is and pulled at runtime.\n        push: Whether or not to skip pushing the built image to a registry.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n        interval: An interval on which to execute the new deployment. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this deployment.\n        rrule: An rrule schedule of when to execute runs of this deployment.\n        triggers: A list of triggers that will kick off runs of this deployment.\n        schedule: A schedule object defining when to execute runs of this deployment. Used to\n            define additional scheduling options like `timezone`.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        parameters: A dictionary of default parameter values to pass to runs of this deployment.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for the created deployment.\n        print_next_steps_message: Whether or not to print a message with next steps\n        after deploying the deployments.\n\n    Returns:\n        The ID of the created/updated deployment.\n\n    Examples:\n        Deploy a local flow to a work pool:\n\n        ```python\n        from prefect import flow\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        if __name__ == \"__main__\":\n            my_flow.deploy(\n                \"example-deployment\",\n                work_pool_name=\"my-work-pool\",\n                image=\"my-repository/my-image:dev\",\n            )\n        ```\n\n        Deploy a remotely stored flow to a work pool:\n\n        ```python\n        from prefect import flow\n\n        if __name__ == \"__main__\":\n            flow.from_source(\n                source=\"https://github.com/org/repo.git\",\n                entrypoint=\"flows.py:my_flow\",\n            ).deploy(\n                \"example-deployment\",\n                work_pool_name=\"my-work-pool\",\n                image=\"my-repository/my-image:dev\",\n            )\n        ```\n    \"\"\"\n    work_pool_name = work_pool_name or PREFECT_DEFAULT_WORK_POOL_NAME.value()\n\n    try:\n        async with get_client() as client:\n            work_pool = await client.read_work_pool(work_pool_name)\n    except ObjectNotFound as exc:\n        raise ValueError(\n            f\"Could not find work pool {work_pool_name!r}. Please create it before\"\n            \" deploying this flow.\"\n        ) from exc\n\n    deployment = await self.to_deployment(\n        name=name,\n        interval=interval,\n        cron=cron,\n        rrule=rrule,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        triggers=triggers,\n        parameters=parameters,\n        description=description,\n        tags=tags,\n        version=version,\n        enforce_parameter_schema=enforce_parameter_schema,\n        work_queue_name=work_queue_name,\n        job_variables=job_variables,\n    )\n\n    deployment_ids = await deploy(\n        deployment,\n        work_pool_name=work_pool_name,\n        image=image,\n        build=build,\n        push=push,\n        print_next_steps_message=False,\n    )\n\n    if print_next_steps:\n        console = Console()\n        if not work_pool.is_push_pool and not work_pool.is_managed_pool:\n            console.print(\n                \"\\nTo execute flow runs from this deployment, start a worker in a\"\n                \" separate terminal that pulls work from the\"\n                f\" {work_pool_name!r} work pool:\"\n            )\n            console.print(\n                f\"\\n\\t$ prefect worker start --pool {work_pool_name!r}\",\n                style=\"blue\",\n            )\n        console.print(\n            \"\\nTo schedule a run for this deployment, use the following command:\"\n        )\n        console.print(\n            f\"\\n\\t$ prefect deployment run '{self.name}/{name}'\\n\",\n            style=\"blue\",\n        )\n        if PREFECT_UI_URL:\n            console.print(\n                \"\\nYou can also run your flow via the Prefect UI:\"\n                f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_ids[0]}[/]\\n\"\n            )\n\n    return deployment_ids[0]\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.from_source","title":"from_source async classmethod","text":"

    Loads a flow from a remote s ource.

    Parameters:

    Name Type Description Default source Union[str, RunnerStorage, ReadableDeploymentStorage]

    Either a URL to a git repository or a storage object.

    required entrypoint str

    The path to a file containing a flow and the name of the flow function in the format ./path/to/file.py:flow_func_name.

    required

    Returns:

    Type Description F

    A new Flow instance.

    Examples:

    Load a flow from a public git repository:

    from prefect import flow\nfrom prefect.runner.storage import GitRepository\nfrom prefect.blocks.system import Secret\n\nmy_flow = flow.from_source(\n    source=\"https://github.com/org/repo.git\",\n    entrypoint=\"flows.py:my_flow\",\n)\n\nmy_flow()\n

    Load a flow from a private git repository:

    from prefect import flow\nfrom prefect.runner.storage import GitRepository\nfrom prefect.blocks.system import Secret\n\nmy_flow = flow.from_source(\n    source=GitRepository(\n        url=\"https://github.com/org/repo.git\",\n        access_token=Secret.load(\"github-access-token\").get(),\n    ),\n    entrypoint=\"flows.py:my_flow\",\n)\n\nmy_flow()\n
    Source code in prefect/flows.py
    @classmethod\n@sync_compatible\nasync def from_source(\n    cls: Type[F],\n    source: Union[str, RunnerStorage, ReadableDeploymentStorage],\n    entrypoint: str,\n) -> F:\n    \"\"\"\n    Loads a flow from a remote s ource.\n\n    Args:\n        source: Either a URL to a git repository or a storage object.\n        entrypoint:  The path to a file containing a flow and the name of the flow function in\n            the format `./path/to/file.py:flow_func_name`.\n\n    Returns:\n        A new `Flow` instance.\n\n    Examples:\n        Load a flow from a public git repository:\n\n\n        ```python\n        from prefect import flow\n        from prefect.runner.storage import GitRepository\n        from prefect.blocks.system import Secret\n\n        my_flow = flow.from_source(\n            source=\"https://github.com/org/repo.git\",\n            entrypoint=\"flows.py:my_flow\",\n        )\n\n        my_flow()\n        ```\n\n        Load a flow from a private git repository:\n\n        ```python\n        from prefect import flow\n        from prefect.runner.storage import GitRepository\n        from prefect.blocks.system import Secret\n\n        my_flow = flow.from_source(\n            source=GitRepository(\n                url=\"https://github.com/org/repo.git\",\n                access_token=Secret.load(\"github-access-token\").get(),\n            ),\n            entrypoint=\"flows.py:my_flow\",\n        )\n\n        my_flow()\n        ```\n    \"\"\"\n    if isinstance(source, str):\n        storage = create_storage_from_url(source)\n    elif isinstance(source, RunnerStorage):\n        storage = source\n    elif hasattr(source, \"get_directory\"):\n        storage = BlockStorageAdapter(source)\n    else:\n        raise TypeError(\n            f\"Unsupported source type {type(source).__name__!r}. Please provide a\"\n            \" URL to remote storage or a storage object.\"\n        )\n    with tempfile.TemporaryDirectory() as tmpdir:\n        storage.set_base_path(Path(tmpdir))\n        await storage.pull_code()\n\n        full_entrypoint = str(storage.destination / entrypoint)\n        flow: \"Flow\" = await from_async.wait_for_call_in_new_thread(\n            create_call(load_flow_from_entrypoint, full_entrypoint)\n        )\n        flow._storage = storage\n        flow._entrypoint = entrypoint\n\n    return flow\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.serialize_parameters","title":"serialize_parameters","text":"

    Convert parameters to a serializable form.

    Uses FastAPI's jsonable_encoder to convert to JSON compatible objects without converting everything directly to a string. This maintains basic types like integers during API roundtrips.

    Source code in prefect/flows.py
    def serialize_parameters(self, parameters: Dict[str, Any]) -> Dict[str, Any]:\n    \"\"\"\n    Convert parameters to a serializable form.\n\n    Uses FastAPI's `jsonable_encoder` to convert to JSON compatible objects without\n    converting everything directly to a string. This maintains basic types like\n    integers during API roundtrips.\n    \"\"\"\n    serialized_parameters = {}\n    for key, value in parameters.items():\n        try:\n            serialized_parameters[key] = jsonable_encoder(value)\n        except (TypeError, ValueError):\n            logger.debug(\n                f\"Parameter {key!r} for flow {self.name!r} is of unserializable \"\n                f\"type {type(value).__name__!r} and will not be stored \"\n                \"in the backend.\"\n            )\n            serialized_parameters[key] = f\"<{type(value).__name__}>\"\n    return serialized_parameters\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.serve","title":"serve async","text":"

    Creates a deployment for this flow and starts a runner to monitor for scheduled work.

    Parameters:

    Name Type Description Default name str

    The name to give the created deployment.

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the new deployment. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this deployment.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this deployment.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that will kick off runs of this deployment.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object defining when to execute runs of this deployment. Used to define additional scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this deployment.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for the created deployment.

    False pause_on_shutdown bool

    If True, provided schedule will be paused when the serve function is stopped. If False, the schedules will continue running.

    True print_starting_message bool

    Whether or not to print the starting message when flow is served.

    True webserver bool

    Whether or not to start a monitoring webserver for this flow.

    False

    Examples:

    Serve a flow:

    from prefect import flow\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\nif __name__ == \"__main__\":\n    my_flow.serve(\"example-deployment\")\n

    Serve a flow and run it every hour:

    from prefect import flow\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\nif __name__ == \"__main__\":\n    my_flow.serve(\"example-deployment\", interval=3600)\n
    Source code in prefect/flows.py
    @sync_compatible\nasync def serve(\n    self,\n    name: str,\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    parameters: Optional[dict] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    pause_on_shutdown: bool = True,\n    print_starting_message: bool = True,\n    webserver: bool = False,\n):\n    \"\"\"\n    Creates a deployment for this flow and starts a runner to monitor for scheduled work.\n\n    Args:\n        name: The name to give the created deployment.\n        interval: An interval on which to execute the new deployment. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this deployment.\n        rrule: An rrule schedule of when to execute runs of this deployment.\n        triggers: A list of triggers that will kick off runs of this deployment.\n        schedule: A schedule object defining when to execute runs of this deployment. Used to\n            define additional scheduling options like `timezone`.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        parameters: A dictionary of default parameter values to pass to runs of this deployment.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for the created deployment.\n        pause_on_shutdown: If True, provided schedule will be paused when the serve function is stopped.\n            If False, the schedules will continue running.\n        print_starting_message: Whether or not to print the starting message when flow is served.\n        webserver: Whether or not to start a monitoring webserver for this flow.\n\n    Examples:\n        Serve a flow:\n\n        ```python\n        from prefect import flow\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        if __name__ == \"__main__\":\n            my_flow.serve(\"example-deployment\")\n        ```\n\n        Serve a flow and run it every hour:\n\n        ```python\n        from prefect import flow\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        if __name__ == \"__main__\":\n            my_flow.serve(\"example-deployment\", interval=3600)\n        ```\n    \"\"\"\n    from prefect.runner import Runner\n\n    # Handling for my_flow.serve(__file__)\n    # Will set name to name of file where my_flow.serve() without the extension\n    # Non filepath strings will pass through unchanged\n    name = Path(name).stem\n\n    runner = Runner(name=name, pause_on_shutdown=pause_on_shutdown)\n    deployment_id = await runner.add_flow(\n        self,\n        name=name,\n        triggers=triggers,\n        interval=interval,\n        cron=cron,\n        rrule=rrule,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        parameters=parameters,\n        description=description,\n        tags=tags,\n        version=version,\n        enforce_parameter_schema=enforce_parameter_schema,\n    )\n    if print_starting_message:\n        help_message = (\n            f\"[green]Your flow {self.name!r} is being served and polling for\"\n            \" scheduled runs!\\n[/]\\nTo trigger a run for this flow, use the\"\n            \" following command:\\n[blue]\\n\\t$ prefect deployment run\"\n            f\" '{self.name}/{name}'\\n[/]\"\n        )\n        if PREFECT_UI_URL:\n            help_message += (\n                \"\\nYou can also run your flow via the Prefect UI:\"\n                f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_id}[/]\\n\"\n            )\n\n        console = Console()\n        console.print(Panel(help_message))\n    await runner.start(webserver=webserver)\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.to_deployment","title":"to_deployment async","text":"

    Creates a runner deployment object for this flow.

    Parameters:

    Name Type Description Default name str

    The name to give the created deployment.

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the new deployment. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this deployment.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this deployment.

    None timezone

    A timezone to use for the schedule. Defaults to UTC.

    required triggers Optional[List[DeploymentTrigger]]

    A list of triggers that will kick off runs of this deployment.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object defining when to execute runs of this deployment.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this deployment.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for the created deployment.

    False work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[Dict[str, Any]]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for

    None

    Examples:

    Prepare two deployments and serve them:

    from prefect import flow, serve\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\n@flow\ndef my_other_flow(name):\n    print(f\"goodbye {name}\")\n\nif __name__ == \"__main__\":\n    hello_deploy = my_flow.to_deployment(\"hello\", tags=[\"dev\"])\n    bye_deploy = my_other_flow.to_deployment(\"goodbye\", tags=[\"dev\"])\n    serve(hello_deploy, bye_deploy)\n
    Source code in prefect/flows.py
    @sync_compatible\nasync def to_deployment(\n    self,\n    name: str,\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    work_pool_name: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[Dict[str, Any]] = None,\n):\n    \"\"\"\n    Creates a runner deployment object for this flow.\n\n    Args:\n        name: The name to give the created deployment.\n        interval: An interval on which to execute the new deployment. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this deployment.\n        rrule: An rrule schedule of when to execute runs of this deployment.\n        timezone: A timezone to use for the schedule. Defaults to UTC.\n        triggers: A list of triggers that will kick off runs of this deployment.\n        schedule: A schedule object defining when to execute runs of this deployment.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        parameters: A dictionary of default parameter values to pass to runs of this deployment.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for the created deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n\n    Examples:\n        Prepare two deployments and serve them:\n\n        ```python\n        from prefect import flow, serve\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        @flow\n        def my_other_flow(name):\n            print(f\"goodbye {name}\")\n\n        if __name__ == \"__main__\":\n            hello_deploy = my_flow.to_deployment(\"hello\", tags=[\"dev\"])\n            bye_deploy = my_other_flow.to_deployment(\"goodbye\", tags=[\"dev\"])\n            serve(hello_deploy, bye_deploy)\n        ```\n    \"\"\"\n    from prefect.deployments.runner import RunnerDeployment\n\n    if not name.endswith(\".py\"):\n        raise_on_name_with_banned_characters(name)\n    if self._storage and self._entrypoint:\n        return await RunnerDeployment.from_storage(\n            storage=self._storage,\n            entrypoint=self._entrypoint,\n            name=name,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags,\n            triggers=triggers,\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n    else:\n        return RunnerDeployment.from_flow(\n            self,\n            name=name,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags,\n            triggers=triggers,\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.validate_parameters","title":"validate_parameters","text":"

    Validate parameters for compatibility with the flow by attempting to cast the inputs to the associated types specified by the function's type annotations.

    Returns:

    Type Description Dict[str, Any]

    A new dict of parameters that have been cast to the appropriate types

    Raises:

    Type Description ParameterTypeError

    if the provided parameters are not valid

    Source code in prefect/flows.py
    def validate_parameters(self, parameters: Dict[str, Any]) -> Dict[str, Any]:\n    \"\"\"\n    Validate parameters for compatibility with the flow by attempting to cast the inputs to the\n    associated types specified by the function's type annotations.\n\n    Returns:\n        A new dict of parameters that have been cast to the appropriate types\n\n    Raises:\n        ParameterTypeError: if the provided parameters are not valid\n    \"\"\"\n    args, kwargs = parameters_to_args_kwargs(self.fn, parameters)\n\n    if HAS_PYDANTIC_V2:\n        has_v1_models = any(isinstance(o, V1BaseModel) for o in args) or any(\n            isinstance(o, V1BaseModel) for o in kwargs.values()\n        )\n        has_v2_models = any(isinstance(o, V2BaseModel) for o in args) or any(\n            isinstance(o, V2BaseModel) for o in kwargs.values()\n        )\n\n        if has_v1_models and has_v2_models:\n            raise ParameterTypeError(\n                \"Cannot mix Pydantic v1 and v2 models as arguments to a flow.\"\n            )\n\n        if has_v1_models:\n            validated_fn = V1ValidatedFunction(\n                self.fn, config={\"arbitrary_types_allowed\": True}\n            )\n        else:\n            validated_fn = V2ValidatedFunction(\n                self.fn, config={\"arbitrary_types_allowed\": True}\n            )\n\n    else:\n        validated_fn = ValidatedFunction(\n            self.fn, config={\"arbitrary_types_allowed\": True}\n        )\n\n    try:\n        model = validated_fn.init_model_instance(*args, **kwargs)\n    except pydantic.ValidationError as exc:\n        # We capture the pydantic exception and raise our own because the pydantic\n        # exception is not picklable when using a cythonized pydantic installation\n        raise ParameterTypeError.from_validation_error(exc) from None\n    except V2ValidationError as exc:\n        # We capture the pydantic exception and raise our own because the pydantic\n        # exception is not picklable when using a cythonized pydantic installation\n        raise ParameterTypeError.from_validation_error(exc) from None\n\n    # Get the updated parameter dict with cast values from the model\n    cast_parameters = {\n        k: v\n        for k, v in model._iter()\n        if k in model.__fields_set__ or model.__fields__[k].default_factory\n    }\n    return cast_parameters\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.visualize","title":"visualize async","text":"

    Generates a graphviz object representing the current flow. In IPython notebooks, it's rendered inline, otherwise in a new window as a PNG.

    Raises:

    Type Description -ImportError

    If graphviz isn't installed.

    -GraphvizExecutableNotFoundError

    If the dot executable isn't found.

    -FlowVisualizationError

    If the flow can't be visualized for any other reason.

    Source code in prefect/flows.py
    @sync_compatible\nasync def visualize(self, *args, **kwargs):\n    \"\"\"\n    Generates a graphviz object representing the current flow. In IPython notebooks,\n    it's rendered inline, otherwise in a new window as a PNG.\n\n    Raises:\n        - ImportError: If `graphviz` isn't installed.\n        - GraphvizExecutableNotFoundError: If the `dot` executable isn't found.\n        - FlowVisualizationError: If the flow can't be visualized for any other reason.\n    \"\"\"\n    if not PREFECT_UNIT_TEST_MODE:\n        warnings.warn(\n            \"`flow.visualize()` will execute code inside of your flow that is not\"\n            \" decorated with `@task` or `@flow`.\"\n        )\n\n    try:\n        with TaskVizTracker() as tracker:\n            if self.isasync:\n                await self.fn(*args, **kwargs)\n            else:\n                self.fn(*args, **kwargs)\n\n            graph = build_task_dependencies(tracker)\n\n            visualize_task_dependencies(graph, self.name)\n\n    except GraphvizImportError:\n        raise\n    except GraphvizExecutableNotFoundError:\n        raise\n    except VisualizationUnsupportedError:\n        raise\n    except FlowVisualizationError:\n        raise\n    except Exception as e:\n        msg = (\n            \"It's possible you are trying to visualize a flow that contains \"\n            \"code that directly interacts with the result of a task\"\n            \" inside of the flow. \\nTry passing a `viz_return_value` \"\n            \"to the task decorator, e.g. `@task(viz_return_value=[1, 2, 3]).`\"\n        )\n\n        new_exception = type(e)(str(e) + \"\\n\" + msg)\n        # Copy traceback information from the original exception\n        new_exception.__traceback__ = e.__traceback__\n        raise new_exception\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.with_options","title":"with_options","text":"

    Create a new flow from the current object, updating provided options.

    Parameters:

    Name Type Description Default name str

    A new name for the flow.

    None version str

    A new version for the flow.

    None description str

    A new description for the flow.

    None flow_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this flow; this name can be provided as a string template with the flow's parameters as variables, or a function that returns a string.

    None task_runner Union[Type[BaseTaskRunner], BaseTaskRunner]

    A new task runner for the flow.

    None timeout_seconds Union[int, float]

    A new number of seconds to fail the flow after if still running.

    None validate_parameters bool

    A new value indicating if flow calls should validate given parameters.

    None retries int

    A new number of times to retry on flow run failure.

    0 retry_delay_seconds Union[int, float]

    A new number of seconds to wait before retrying the flow after failure. This is only applicable if retries is nonzero.

    0 persist_result Optional[bool]

    A new option for enabling or disabling result persistence.

    NotSet result_storage Optional[ResultStorage]

    A new storage type to use for results.

    NotSet result_serializer Optional[ResultSerializer]

    A new serializer to use for results.

    NotSet cache_result_in_memory bool

    A new value indicating if the flow's result should be cached in memory.

    None on_failure Optional[List[Callable[[Flow, FlowRun, State], None]]]

    A new list of callables to run when the flow enters a failed state.

    None on_completion Optional[List[Callable[[Flow, FlowRun, State], None]]]

    A new list of callables to run when the flow enters a completed state.

    None on_cancellation Optional[List[Callable[[Flow, FlowRun, State], None]]]

    A new list of callables to run when the flow enters a cancelling state.

    None on_crashed Optional[List[Callable[[Flow, FlowRun, State], None]]]

    A new list of callables to run when the flow enters a crashed state.

    None

    Returns:

    Type Description Self

    A new Flow instance.

    Create a new flow from an existing flow and update the name:\n\n>>> @flow(name=\"My flow\")\n>>> def my_flow():\n>>>     return 1\n>>>\n>>> new_flow = my_flow.with_options(name=\"My new flow\")\n\nCreate a new flow from an existing flow, update the task runner, and call\nit without an intermediate variable:\n\n>>> from prefect.task_runners import SequentialTaskRunner\n>>>\n>>> @flow\n>>> def my_flow(x, y):\n>>>     return x + y\n>>>\n>>> state = my_flow.with_options(task_runner=SequentialTaskRunner)(1, 3)\n>>> assert state.result() == 4\n
    Source code in prefect/flows.py
    def with_options(\n    self,\n    *,\n    name: str = None,\n    version: str = None,\n    retries: int = 0,\n    retry_delay_seconds: Union[int, float] = 0,\n    description: str = None,\n    flow_run_name: Optional[Union[Callable[[], str], str]] = None,\n    task_runner: Union[Type[BaseTaskRunner], BaseTaskRunner] = None,\n    timeout_seconds: Union[int, float] = None,\n    validate_parameters: bool = None,\n    persist_result: Optional[bool] = NotSet,\n    result_storage: Optional[ResultStorage] = NotSet,\n    result_serializer: Optional[ResultSerializer] = NotSet,\n    cache_result_in_memory: bool = None,\n    log_prints: Optional[bool] = NotSet,\n    on_completion: Optional[\n        List[Callable[[FlowSchema, FlowRun, State], None]]\n    ] = None,\n    on_failure: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    on_cancellation: Optional[\n        List[Callable[[FlowSchema, FlowRun, State], None]]\n    ] = None,\n    on_crashed: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n) -> Self:\n    \"\"\"\n    Create a new flow from the current object, updating provided options.\n\n    Args:\n        name: A new name for the flow.\n        version: A new version for the flow.\n        description: A new description for the flow.\n        flow_run_name: An optional name to distinguish runs of this flow; this name\n            can be provided as a string template with the flow's parameters as variables,\n            or a function that returns a string.\n        task_runner: A new task runner for the flow.\n        timeout_seconds: A new number of seconds to fail the flow after if still\n            running.\n        validate_parameters: A new value indicating if flow calls should validate\n            given parameters.\n        retries: A new number of times to retry on flow run failure.\n        retry_delay_seconds: A new number of seconds to wait before retrying the\n            flow after failure. This is only applicable if `retries` is nonzero.\n        persist_result: A new option for enabling or disabling result persistence.\n        result_storage: A new storage type to use for results.\n        result_serializer: A new serializer to use for results.\n        cache_result_in_memory: A new value indicating if the flow's result should\n            be cached in memory.\n        on_failure: A new list of callables to run when the flow enters a failed state.\n        on_completion: A new list of callables to run when the flow enters a completed state.\n        on_cancellation: A new list of callables to run when the flow enters a cancelling state.\n        on_crashed: A new list of callables to run when the flow enters a crashed state.\n\n    Returns:\n        A new `Flow` instance.\n\n    Examples:\n\n        Create a new flow from an existing flow and update the name:\n\n        >>> @flow(name=\"My flow\")\n        >>> def my_flow():\n        >>>     return 1\n        >>>\n        >>> new_flow = my_flow.with_options(name=\"My new flow\")\n\n        Create a new flow from an existing flow, update the task runner, and call\n        it without an intermediate variable:\n\n        >>> from prefect.task_runners import SequentialTaskRunner\n        >>>\n        >>> @flow\n        >>> def my_flow(x, y):\n        >>>     return x + y\n        >>>\n        >>> state = my_flow.with_options(task_runner=SequentialTaskRunner)(1, 3)\n        >>> assert state.result() == 4\n\n    \"\"\"\n    new_flow = Flow(\n        fn=self.fn,\n        name=name or self.name,\n        description=description or self.description,\n        flow_run_name=flow_run_name,\n        version=version or self.version,\n        task_runner=task_runner or self.task_runner,\n        retries=retries or self.retries,\n        retry_delay_seconds=retry_delay_seconds or self.retry_delay_seconds,\n        timeout_seconds=(\n            timeout_seconds if timeout_seconds is not None else self.timeout_seconds\n        ),\n        validate_parameters=(\n            validate_parameters\n            if validate_parameters is not None\n            else self.should_validate_parameters\n        ),\n        persist_result=(\n            persist_result if persist_result is not NotSet else self.persist_result\n        ),\n        result_storage=(\n            result_storage if result_storage is not NotSet else self.result_storage\n        ),\n        result_serializer=(\n            result_serializer\n            if result_serializer is not NotSet\n            else self.result_serializer\n        ),\n        cache_result_in_memory=(\n            cache_result_in_memory\n            if cache_result_in_memory is not None\n            else self.cache_result_in_memory\n        ),\n        log_prints=log_prints if log_prints is not NotSet else self.log_prints,\n        on_completion=on_completion or self.on_completion,\n        on_failure=on_failure or self.on_failure,\n        on_cancellation=on_cancellation or self.on_cancellation,\n        on_crashed=on_crashed or self.on_crashed,\n    )\n    new_flow._storage = self._storage\n    new_flow._entrypoint = self._entrypoint\n    return new_flow\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.flow","title":"flow","text":"

    Decorator to designate a function as a Prefect workflow.

    This decorator may be used for asynchronous or synchronous functions.

    Flow parameters must be serializable by Pydantic.

    Parameters:

    Name Type Description Default name Optional[str]

    An optional name for the flow; if not provided, the name will be inferred from the given function.

    None version Optional[str]

    An optional version string for the flow; if not provided, we will attempt to create a version string as a hash of the file containing the wrapped function; if the file cannot be located, the version will be null.

    None flow_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this flow; this name can be provided as a string template with the flow's parameters as variables, or a function that returns a string.

    None task_runner BaseTaskRunner

    An optional task runner to use for task execution within the flow; if not provided, a ConcurrentTaskRunner will be instantiated.

    ConcurrentTaskRunner description str

    An optional string description for the flow; if not provided, the description will be pulled from the docstring for the decorated function.

    None timeout_seconds Union[int, float]

    An optional number of seconds indicating a maximum runtime for the flow. If the flow exceeds this runtime, it will be marked as failed. Flow execution may continue until the next task is called.

    None validate_parameters bool

    By default, parameters passed to flows are validated by Pydantic. This will check that input values conform to the annotated types on the function. Where possible, values will be coerced into the correct type; for example, if a parameter is defined as x: int and \"5\" is passed, it will be resolved to 5. If set to False, no validation will be performed on flow parameters.

    True retries int

    An optional number of times to retry on flow run failure.

    None retry_delay_seconds Union[int, float]

    An optional number of seconds to wait before retrying the flow after failure. This is only applicable if retries is nonzero.

    None persist_result Optional[bool]

    An optional toggle indicating whether the result of this flow should be persisted to result storage. Defaults to None, which indicates that Prefect should choose whether the result should be persisted depending on the features being used.

    None result_storage Optional[ResultStorage]

    An optional block to use to persist the result of this flow. This value will be used as the default for any tasks in this flow. If not provided, the local file system will be used unless called as a subflow, at which point the default will be loaded from the parent flow.

    None result_serializer Optional[ResultSerializer]

    An optional serializer to use to serialize the result of this flow for persistence. This value will be used as the default for any tasks in this flow. If not provided, the value of PREFECT_RESULTS_DEFAULT_SERIALIZER will be used unless called as a subflow, at which point the default will be loaded from the parent flow.

    None log_prints Optional[bool]

    If set, print statements in the flow will be redirected to the Prefect logger for the flow run. Defaults to None, which indicates that the value from the parent flow should be used. If this is a parent flow, the default is pulled from the PREFECT_LOGGING_LOG_PRINTS setting.

    None on_completion Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of functions to call when the flow run is completed. Each function should accept three arguments: the flow, the flow run, and the final state of the flow run.

    None on_failure Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of functions to call when the flow run fails. Each function should accept three arguments: the flow, the flow run, and the final state of the flow run.

    None on_cancellation Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of functions to call when the flow run is cancelled. These functions will be passed the flow, flow run, and final state.

    None on_crashed Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of functions to call when the flow run crashes. Each function should accept three arguments: the flow, the flow run, and the final state of the flow run.

    None

    Returns:

    Type Description

    A callable Flow object which, when called, will run the flow and return its

    final state.

    Examples:

    Define a simple flow

    >>> from prefect import flow\n>>> @flow\n>>> def add(x, y):\n>>>     return x + y\n

    Define an async flow

    >>> @flow\n>>> async def add(x, y):\n>>>     return x + y\n

    Define a flow with a version and description

    >>> @flow(version=\"first-flow\", description=\"This flow is empty!\")\n>>> def my_flow():\n>>>     pass\n

    Define a flow with a custom name

    >>> @flow(name=\"The Ultimate Flow\")\n>>> def my_flow():\n>>>     pass\n

    Define a flow that submits its tasks to dask

    >>> from prefect_dask.task_runners import DaskTaskRunner\n>>>\n>>> @flow(task_runner=DaskTaskRunner)\n>>> def my_flow():\n>>>     pass\n
    Source code in prefect/flows.py
    def flow(\n    __fn=None,\n    *,\n    name: Optional[str] = None,\n    version: Optional[str] = None,\n    flow_run_name: Optional[Union[Callable[[], str], str]] = None,\n    retries: int = None,\n    retry_delay_seconds: Union[int, float] = None,\n    task_runner: BaseTaskRunner = ConcurrentTaskRunner,\n    description: str = None,\n    timeout_seconds: Union[int, float] = None,\n    validate_parameters: bool = True,\n    persist_result: Optional[bool] = None,\n    result_storage: Optional[ResultStorage] = None,\n    result_serializer: Optional[ResultSerializer] = None,\n    cache_result_in_memory: bool = True,\n    log_prints: Optional[bool] = None,\n    on_completion: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    on_failure: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    on_cancellation: Optional[\n        List[Callable[[FlowSchema, FlowRun, State], None]]\n    ] = None,\n    on_crashed: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n):\n    \"\"\"\n    Decorator to designate a function as a Prefect workflow.\n\n    This decorator may be used for asynchronous or synchronous functions.\n\n    Flow parameters must be serializable by Pydantic.\n\n    Args:\n        name: An optional name for the flow; if not provided, the name will be inferred\n            from the given function.\n        version: An optional version string for the flow; if not provided, we will\n            attempt to create a version string as a hash of the file containing the\n            wrapped function; if the file cannot be located, the version will be null.\n        flow_run_name: An optional name to distinguish runs of this flow; this name can\n            be provided as a string template with the flow's parameters as variables,\n            or a function that returns a string.\n        task_runner: An optional task runner to use for task execution within the flow; if\n            not provided, a `ConcurrentTaskRunner` will be instantiated.\n        description: An optional string description for the flow; if not provided, the\n            description will be pulled from the docstring for the decorated function.\n        timeout_seconds: An optional number of seconds indicating a maximum runtime for\n            the flow. If the flow exceeds this runtime, it will be marked as failed.\n            Flow execution may continue until the next task is called.\n        validate_parameters: By default, parameters passed to flows are validated by\n            Pydantic. This will check that input values conform to the annotated types\n            on the function. Where possible, values will be coerced into the correct\n            type; for example, if a parameter is defined as `x: int` and \"5\" is passed,\n            it will be resolved to `5`. If set to `False`, no validation will be\n            performed on flow parameters.\n        retries: An optional number of times to retry on flow run failure.\n        retry_delay_seconds: An optional number of seconds to wait before retrying the\n            flow after failure. This is only applicable if `retries` is nonzero.\n        persist_result: An optional toggle indicating whether the result of this flow\n            should be persisted to result storage. Defaults to `None`, which indicates\n            that Prefect should choose whether the result should be persisted depending on\n            the features being used.\n        result_storage: An optional block to use to persist the result of this flow.\n            This value will be used as the default for any tasks in this flow.\n            If not provided, the local file system will be used unless called as\n            a subflow, at which point the default will be loaded from the parent flow.\n        result_serializer: An optional serializer to use to serialize the result of this\n            flow for persistence. This value will be used as the default for any tasks\n            in this flow. If not provided, the value of `PREFECT_RESULTS_DEFAULT_SERIALIZER`\n            will be used unless called as a subflow, at which point the default will be\n            loaded from the parent flow.\n        log_prints: If set, `print` statements in the flow will be redirected to the\n            Prefect logger for the flow run. Defaults to `None`, which indicates that\n            the value from the parent flow should be used. If this is a parent flow,\n            the default is pulled from the `PREFECT_LOGGING_LOG_PRINTS` setting.\n        on_completion: An optional list of functions to call when the flow run is\n            completed. Each function should accept three arguments: the flow, the flow\n            run, and the final state of the flow run.\n        on_failure: An optional list of functions to call when the flow run fails. Each\n            function should accept three arguments: the flow, the flow run, and the\n            final state of the flow run.\n        on_cancellation: An optional list of functions to call when the flow run is\n            cancelled. These functions will be passed the flow, flow run, and final state.\n        on_crashed: An optional list of functions to call when the flow run crashes. Each\n            function should accept three arguments: the flow, the flow run, and the\n            final state of the flow run.\n\n    Returns:\n        A callable `Flow` object which, when called, will run the flow and return its\n        final state.\n\n    Examples:\n        Define a simple flow\n\n        >>> from prefect import flow\n        >>> @flow\n        >>> def add(x, y):\n        >>>     return x + y\n\n        Define an async flow\n\n        >>> @flow\n        >>> async def add(x, y):\n        >>>     return x + y\n\n        Define a flow with a version and description\n\n        >>> @flow(version=\"first-flow\", description=\"This flow is empty!\")\n        >>> def my_flow():\n        >>>     pass\n\n        Define a flow with a custom name\n\n        >>> @flow(name=\"The Ultimate Flow\")\n        >>> def my_flow():\n        >>>     pass\n\n        Define a flow that submits its tasks to dask\n\n        >>> from prefect_dask.task_runners import DaskTaskRunner\n        >>>\n        >>> @flow(task_runner=DaskTaskRunner)\n        >>> def my_flow():\n        >>>     pass\n    \"\"\"\n    if __fn:\n        return cast(\n            Flow[P, R],\n            Flow(\n                fn=__fn,\n                name=name,\n                version=version,\n                flow_run_name=flow_run_name,\n                task_runner=task_runner,\n                description=description,\n                timeout_seconds=timeout_seconds,\n                validate_parameters=validate_parameters,\n                retries=retries,\n                retry_delay_seconds=retry_delay_seconds,\n                persist_result=persist_result,\n                result_storage=result_storage,\n                result_serializer=result_serializer,\n                cache_result_in_memory=cache_result_in_memory,\n                log_prints=log_prints,\n                on_completion=on_completion,\n                on_failure=on_failure,\n                on_cancellation=on_cancellation,\n                on_crashed=on_crashed,\n            ),\n        )\n    else:\n        return cast(\n            Callable[[Callable[P, R]], Flow[P, R]],\n            partial(\n                flow,\n                name=name,\n                version=version,\n                flow_run_name=flow_run_name,\n                task_runner=task_runner,\n                description=description,\n                timeout_seconds=timeout_seconds,\n                validate_parameters=validate_parameters,\n                retries=retries,\n                retry_delay_seconds=retry_delay_seconds,\n                persist_result=persist_result,\n                result_storage=result_storage,\n                result_serializer=result_serializer,\n                cache_result_in_memory=cache_result_in_memory,\n                log_prints=log_prints,\n                on_completion=on_completion,\n                on_failure=on_failure,\n                on_cancellation=on_cancellation,\n                on_crashed=on_crashed,\n            ),\n        )\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.load_flow_from_entrypoint","title":"load_flow_from_entrypoint","text":"

    Extract a flow object from a script at an entrypoint by running all of the code in the file.

    Parameters:

    Name Type Description Default entrypoint str

    a string in the format <path_to_script>:<flow_func_name>

    required

    Returns:

    Type Description Flow

    The flow object from the script

    Raises:

    Type Description FlowScriptError

    If an exception is encountered while running the script

    MissingFlowError

    If the flow function specified in the entrypoint does not exist

    Source code in prefect/flows.py
    def load_flow_from_entrypoint(entrypoint: str) -> Flow:\n    \"\"\"\n    Extract a flow object from a script at an entrypoint by running all of the code in the file.\n\n    Args:\n        entrypoint: a string in the format `<path_to_script>:<flow_func_name>`\n\n    Returns:\n        The flow object from the script\n\n    Raises:\n        FlowScriptError: If an exception is encountered while running the script\n        MissingFlowError: If the flow function specified in the entrypoint does not exist\n    \"\"\"\n    with PrefectObjectRegistry(\n        block_code_execution=True,\n        capture_failures=True,\n    ):\n        # split by the last colon once to handle Windows paths with drive letters i.e C:\\path\\to\\file.py:do_stuff\n        path, func_name = entrypoint.rsplit(\":\", maxsplit=1)\n        try:\n            flow = import_object(entrypoint)\n        except AttributeError as exc:\n            raise MissingFlowError(\n                f\"Flow function with name {func_name!r} not found in {path!r}. \"\n            ) from exc\n\n        if not isinstance(flow, Flow):\n            raise MissingFlowError(\n                f\"Function with name {func_name!r} is not a flow. Make sure that it is \"\n                \"decorated with '@flow'.\"\n            )\n\n        return flow\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.load_flow_from_script","title":"load_flow_from_script","text":"

    Extract a flow object from a script by running all of the code in the file.

    If the script has multiple flows in it, a flow name must be provided to specify the flow to return.

    Parameters:

    Name Type Description Default path str

    A path to a Python script containing flows

    required flow_name str

    An optional flow name to look for in the script

    None

    Returns:

    Type Description Flow

    The flow object from the script

    Raises:

    Type Description FlowScriptError

    If an exception is encountered while running the script

    MissingFlowError

    If no flows exist in the iterable

    MissingFlowError

    If a flow name is provided and that flow does not exist

    UnspecifiedFlowError

    If multiple flows exist but no flow name was provided

    Source code in prefect/flows.py
    def load_flow_from_script(path: str, flow_name: str = None) -> Flow:\n    \"\"\"\n    Extract a flow object from a script by running all of the code in the file.\n\n    If the script has multiple flows in it, a flow name must be provided to specify\n    the flow to return.\n\n    Args:\n        path: A path to a Python script containing flows\n        flow_name: An optional flow name to look for in the script\n\n    Returns:\n        The flow object from the script\n\n    Raises:\n        FlowScriptError: If an exception is encountered while running the script\n        MissingFlowError: If no flows exist in the iterable\n        MissingFlowError: If a flow name is provided and that flow does not exist\n        UnspecifiedFlowError: If multiple flows exist but no flow name was provided\n    \"\"\"\n    return select_flow(\n        load_flows_from_script(path),\n        flow_name=flow_name,\n        from_message=f\"in script '{path}'\",\n    )\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.load_flow_from_text","title":"load_flow_from_text","text":"

    Load a flow from a text script.

    The script will be written to a temporary local file path so errors can refer to line numbers and contextual tracebacks can be provided.

    Source code in prefect/flows.py
    def load_flow_from_text(script_contents: AnyStr, flow_name: str):\n    \"\"\"\n    Load a flow from a text script.\n\n    The script will be written to a temporary local file path so errors can refer\n    to line numbers and contextual tracebacks can be provided.\n    \"\"\"\n    with NamedTemporaryFile(\n        mode=\"wt\" if isinstance(script_contents, str) else \"wb\",\n        prefix=f\"flow-script-{flow_name}\",\n        suffix=\".py\",\n        delete=False,\n    ) as tmpfile:\n        tmpfile.write(script_contents)\n        tmpfile.flush()\n    try:\n        flow = load_flow_from_script(tmpfile.name, flow_name=flow_name)\n    finally:\n        # windows compat\n        tmpfile.close()\n        os.remove(tmpfile.name)\n    return flow\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.load_flows_from_script","title":"load_flows_from_script","text":"

    Load all flow objects from the given python script. All of the code in the file will be executed.

    Returns:

    Type Description List[Flow]

    A list of flows

    Raises:

    Type Description FlowScriptError

    If an exception is encountered while running the script

    Source code in prefect/flows.py
    def load_flows_from_script(path: str) -> List[Flow]:\n    \"\"\"\n    Load all flow objects from the given python script. All of the code in the file\n    will be executed.\n\n    Returns:\n        A list of flows\n\n    Raises:\n        FlowScriptError: If an exception is encountered while running the script\n    \"\"\"\n    return registry_from_script(path).get_instances(Flow)\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.select_flow","title":"select_flow","text":"

    Select the only flow in an iterable or a flow specified by name.

    Returns A single flow object

    Raises:

    Type Description MissingFlowError

    If no flows exist in the iterable

    MissingFlowError

    If a flow name is provided and that flow does not exist

    UnspecifiedFlowError

    If multiple flows exist but no flow name was provided

    Source code in prefect/flows.py
    def select_flow(\n    flows: Iterable[Flow], flow_name: str = None, from_message: str = None\n) -> Flow:\n    \"\"\"\n    Select the only flow in an iterable or a flow specified by name.\n\n    Returns\n        A single flow object\n\n    Raises:\n        MissingFlowError: If no flows exist in the iterable\n        MissingFlowError: If a flow name is provided and that flow does not exist\n        UnspecifiedFlowError: If multiple flows exist but no flow name was provided\n    \"\"\"\n    # Convert to flows by name\n    flows = {f.name: f for f in flows}\n\n    # Add a leading space if given, otherwise use an empty string\n    from_message = (\" \" + from_message) if from_message else \"\"\n    if not flows:\n        raise MissingFlowError(f\"No flows found{from_message}.\")\n\n    elif flow_name and flow_name not in flows:\n        raise MissingFlowError(\n            f\"Flow {flow_name!r} not found{from_message}. \"\n            f\"Found the following flows: {listrepr(flows.keys())}. \"\n            \"Check to make sure that your flow function is decorated with `@flow`.\"\n        )\n\n    elif not flow_name and len(flows) > 1:\n        raise UnspecifiedFlowError(\n            (\n                f\"Found {len(flows)} flows{from_message}:\"\n                f\" {listrepr(sorted(flows.keys()))}. Specify a flow name to select a\"\n                \" flow.\"\n            ),\n        )\n\n    if flow_name:\n        return flows[flow_name]\n    else:\n        return list(flows.values())[0]\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/futures/","title":"prefect.futures","text":"","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures","title":"prefect.futures","text":"

    Futures represent the execution of a task and allow retrieval of the task run's state.

    This module contains the definition for futures as well as utilities for resolving futures in nested data structures.

    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.PrefectFuture","title":"PrefectFuture","text":"

    Bases: Generic[R, A]

    Represents the result of a computation happening in a task runner.

    When tasks are called, they are submitted to a task runner which creates a future for access to the state and result of the task.

    Examples:

    Define a task that returns a string

    >>> from prefect import flow, task\n>>> @task\n>>> def my_task() -> str:\n>>>     return \"hello\"\n

    Calls of this task in a flow will return a future

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()  # PrefectFuture[str, Sync] includes result type\n>>>     future.task_run.id  # UUID for the task run\n

    Wait for the task to complete

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     final_state = future.wait()\n

    Wait N seconds for the task to complete

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     final_state = future.wait(0.1)\n>>>     if final_state:\n>>>         ... # Task done\n>>>     else:\n>>>         ... # Task not done yet\n

    Wait for a task to complete and retrieve its result

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     result = future.result()\n>>>     assert result == \"hello\"\n

    Wait N seconds for a task to complete and retrieve its result

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     result = future.result(timeout=5)\n>>>     assert result == \"hello\"\n

    Retrieve the state of a task without waiting for completion

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     state = future.get_state()\n
    Source code in prefect/futures.py
    class PrefectFuture(Generic[R, A]):\n    \"\"\"\n    Represents the result of a computation happening in a task runner.\n\n    When tasks are called, they are submitted to a task runner which creates a future\n    for access to the state and result of the task.\n\n    Examples:\n        Define a task that returns a string\n\n        >>> from prefect import flow, task\n        >>> @task\n        >>> def my_task() -> str:\n        >>>     return \"hello\"\n\n        Calls of this task in a flow will return a future\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()  # PrefectFuture[str, Sync] includes result type\n        >>>     future.task_run.id  # UUID for the task run\n\n        Wait for the task to complete\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     final_state = future.wait()\n\n        Wait N seconds for the task to complete\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     final_state = future.wait(0.1)\n        >>>     if final_state:\n        >>>         ... # Task done\n        >>>     else:\n        >>>         ... # Task not done yet\n\n        Wait for a task to complete and retrieve its result\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     result = future.result()\n        >>>     assert result == \"hello\"\n\n        Wait N seconds for a task to complete and retrieve its result\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     result = future.result(timeout=5)\n        >>>     assert result == \"hello\"\n\n        Retrieve the state of a task without waiting for completion\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     state = future.get_state()\n    \"\"\"\n\n    def __init__(\n        self,\n        name: str,\n        key: UUID,\n        task_runner: \"BaseTaskRunner\",\n        asynchronous: A = True,\n        _final_state: State[R] = None,  # Exposed for testing\n    ) -> None:\n        self.key = key\n        self.name = name\n        self.asynchronous = asynchronous\n        self.task_run = None\n        self._final_state = _final_state\n        self._exception: Optional[Exception] = None\n        self._task_runner = task_runner\n        self._submitted = anyio.Event()\n\n        self._loop = asyncio.get_running_loop()\n\n    @overload\n    def wait(\n        self: \"PrefectFuture[R, Async]\", timeout: None = None\n    ) -> Awaitable[State[R]]:\n        ...\n\n    @overload\n    def wait(self: \"PrefectFuture[R, Sync]\", timeout: None = None) -> State[R]:\n        ...\n\n    @overload\n    def wait(\n        self: \"PrefectFuture[R, Async]\", timeout: float\n    ) -> Awaitable[Optional[State[R]]]:\n        ...\n\n    @overload\n    def wait(self: \"PrefectFuture[R, Sync]\", timeout: float) -> Optional[State[R]]:\n        ...\n\n    def wait(self, timeout=None):\n        \"\"\"\n        Wait for the run to finish and return the final state\n\n        If the timeout is reached before the run reaches a final state,\n        `None` is returned.\n        \"\"\"\n        wait = create_call(self._wait, timeout=timeout)\n        if self.asynchronous:\n            return from_async.call_soon_in_loop_thread(wait).aresult()\n        else:\n            # type checking cannot handle the overloaded timeout passing\n            return from_sync.call_soon_in_loop_thread(wait).result()  # type: ignore\n\n    @overload\n    async def _wait(self, timeout: None = None) -> State[R]:\n        ...\n\n    @overload\n    async def _wait(self, timeout: float) -> Optional[State[R]]:\n        ...\n\n    async def _wait(self, timeout=None):\n        \"\"\"\n        Async implementation for `wait`\n        \"\"\"\n        await self._wait_for_submission()\n\n        if self._final_state:\n            return self._final_state\n\n        self._final_state = await self._task_runner.wait(self.key, timeout)\n        return self._final_state\n\n    @overload\n    def result(\n        self: \"PrefectFuture[R, Sync]\",\n        timeout: float = None,\n        raise_on_failure: bool = True,\n    ) -> R:\n        ...\n\n    @overload\n    def result(\n        self: \"PrefectFuture[R, Sync]\",\n        timeout: float = None,\n        raise_on_failure: bool = False,\n    ) -> Union[R, Exception]:\n        ...\n\n    @overload\n    def result(\n        self: \"PrefectFuture[R, Async]\",\n        timeout: float = None,\n        raise_on_failure: bool = True,\n    ) -> Awaitable[R]:\n        ...\n\n    @overload\n    def result(\n        self: \"PrefectFuture[R, Async]\",\n        timeout: float = None,\n        raise_on_failure: bool = False,\n    ) -> Awaitable[Union[R, Exception]]:\n        ...\n\n    def result(self, timeout: float = None, raise_on_failure: bool = True):\n        \"\"\"\n        Wait for the run to finish and return the final state.\n\n        If the timeout is reached before the run reaches a final state, a `TimeoutError`\n        will be raised.\n\n        If `raise_on_failure` is `True` and the task run failed, the task run's\n        exception will be raised.\n        \"\"\"\n        result = create_call(\n            self._result, timeout=timeout, raise_on_failure=raise_on_failure\n        )\n        if self.asynchronous:\n            return from_async.call_soon_in_loop_thread(result).aresult()\n        else:\n            return from_sync.call_soon_in_loop_thread(result).result()\n\n    async def _result(self, timeout: float = None, raise_on_failure: bool = True):\n        \"\"\"\n        Async implementation of `result`\n        \"\"\"\n        final_state = await self._wait(timeout=timeout)\n        if not final_state:\n            raise TimeoutError(\"Call timed out before task finished.\")\n        return await final_state.result(raise_on_failure=raise_on_failure, fetch=True)\n\n    @overload\n    def get_state(\n        self: \"PrefectFuture[R, Async]\", client: PrefectClient = None\n    ) -> Awaitable[State[R]]:\n        ...\n\n    @overload\n    def get_state(\n        self: \"PrefectFuture[R, Sync]\", client: PrefectClient = None\n    ) -> State[R]:\n        ...\n\n    def get_state(self, client: PrefectClient = None):\n        \"\"\"\n        Get the current state of the task run.\n        \"\"\"\n        if self.asynchronous:\n            return cast(Awaitable[State[R]], self._get_state(client=client))\n        else:\n            return cast(State[R], sync(self._get_state, client=client))\n\n    @inject_client\n    async def _get_state(self, client: PrefectClient = None) -> State[R]:\n        assert client is not None  # always injected\n\n        # We must wait for the task run id to be populated\n        await self._wait_for_submission()\n\n        task_run = await client.read_task_run(self.task_run.id)\n\n        if not task_run:\n            raise RuntimeError(\"Future has no associated task run in the server.\")\n\n        # Update the task run reference\n        self.task_run = task_run\n        return task_run.state\n\n    async def _wait_for_submission(self):\n        await run_coroutine_in_loop_from_async(self._loop, self._submitted.wait())\n\n    def __hash__(self) -> int:\n        return hash(self.key)\n\n    def __repr__(self) -> str:\n        return f\"PrefectFuture({self.name!r})\"\n\n    def __bool__(self) -> bool:\n        warnings.warn(\n            (\n                \"A 'PrefectFuture' from a task call was cast to a boolean; \"\n                \"did you mean to check the result of the task instead? \"\n                \"e.g. `if my_task().result(): ...`\"\n            ),\n            stacklevel=2,\n        )\n        return True\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.PrefectFuture.get_state","title":"get_state","text":"

    Get the current state of the task run.

    Source code in prefect/futures.py
    def get_state(self, client: PrefectClient = None):\n    \"\"\"\n    Get the current state of the task run.\n    \"\"\"\n    if self.asynchronous:\n        return cast(Awaitable[State[R]], self._get_state(client=client))\n    else:\n        return cast(State[R], sync(self._get_state, client=client))\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.PrefectFuture.result","title":"result","text":"

    Wait for the run to finish and return the final state.

    If the timeout is reached before the run reaches a final state, a TimeoutError will be raised.

    If raise_on_failure is True and the task run failed, the task run's exception will be raised.

    Source code in prefect/futures.py
    def result(self, timeout: float = None, raise_on_failure: bool = True):\n    \"\"\"\n    Wait for the run to finish and return the final state.\n\n    If the timeout is reached before the run reaches a final state, a `TimeoutError`\n    will be raised.\n\n    If `raise_on_failure` is `True` and the task run failed, the task run's\n    exception will be raised.\n    \"\"\"\n    result = create_call(\n        self._result, timeout=timeout, raise_on_failure=raise_on_failure\n    )\n    if self.asynchronous:\n        return from_async.call_soon_in_loop_thread(result).aresult()\n    else:\n        return from_sync.call_soon_in_loop_thread(result).result()\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.PrefectFuture.wait","title":"wait","text":"

    Wait for the run to finish and return the final state

    If the timeout is reached before the run reaches a final state, None is returned.

    Source code in prefect/futures.py
    def wait(self, timeout=None):\n    \"\"\"\n    Wait for the run to finish and return the final state\n\n    If the timeout is reached before the run reaches a final state,\n    `None` is returned.\n    \"\"\"\n    wait = create_call(self._wait, timeout=timeout)\n    if self.asynchronous:\n        return from_async.call_soon_in_loop_thread(wait).aresult()\n    else:\n        # type checking cannot handle the overloaded timeout passing\n        return from_sync.call_soon_in_loop_thread(wait).result()  # type: ignore\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.call_repr","title":"call_repr","text":"

    Generate a repr for a function call as \"fn_name(arg_value, kwarg_name=kwarg_value)\"

    Source code in prefect/futures.py
    def call_repr(__fn: Callable, *args: Any, **kwargs: Any) -> str:\n    \"\"\"\n    Generate a repr for a function call as \"fn_name(arg_value, kwarg_name=kwarg_value)\"\n    \"\"\"\n\n    name = __fn.__name__\n\n    # TODO: If this computation is concerningly expensive, we can iterate checking the\n    #       length at each arg or avoid calling `repr` on args with large amounts of\n    #       data\n    call_args = \", \".join(\n        [repr(arg) for arg in args]\n        + [f\"{key}={repr(val)}\" for key, val in kwargs.items()]\n    )\n\n    # Enforce a maximum length\n    if len(call_args) > 100:\n        call_args = call_args[:100] + \"...\"\n\n    return f\"{name}({call_args})\"\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.resolve_futures_to_data","title":"resolve_futures_to_data async","text":"

    Given a Python built-in collection, recursively find PrefectFutures and build a new collection with the same structure with futures resolved to their results. Resolving futures to their results may wait for execution to complete and require communication with the API.

    Unsupported object types will be returned without modification.

    Source code in prefect/futures.py
    async def resolve_futures_to_data(\n    expr: Union[PrefectFuture[R, Any], Any],\n    raise_on_failure: bool = True,\n) -> Union[R, Any]:\n    \"\"\"\n    Given a Python built-in collection, recursively find `PrefectFutures` and build a\n    new collection with the same structure with futures resolved to their results.\n    Resolving futures to their results may wait for execution to complete and require\n    communication with the API.\n\n    Unsupported object types will be returned without modification.\n    \"\"\"\n    futures: Set[PrefectFuture] = set()\n\n    maybe_expr = visit_collection(\n        expr,\n        visit_fn=partial(_collect_futures, futures),\n        return_data=False,\n        context={},\n    )\n    if maybe_expr is not None:\n        expr = maybe_expr\n\n    # Get results\n    results = await asyncio.gather(\n        *[\n            # We must wait for the future in the thread it was created in\n            from_async.call_soon_in_loop_thread(\n                create_call(future._result, raise_on_failure=raise_on_failure)\n            ).aresult()\n            for future in futures\n        ]\n    )\n\n    results_by_future = dict(zip(futures, results))\n\n    def replace_futures_with_results(expr, context):\n        # Expressions inside quotes should not be modified\n        if isinstance(context.get(\"annotation\"), quote):\n            raise StopVisiting()\n\n        if isinstance(expr, PrefectFuture):\n            return results_by_future[expr]\n        else:\n            return expr\n\n    return visit_collection(\n        expr,\n        visit_fn=replace_futures_with_results,\n        return_data=True,\n        context={},\n    )\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.resolve_futures_to_states","title":"resolve_futures_to_states async","text":"

    Given a Python built-in collection, recursively find PrefectFutures and build a new collection with the same structure with futures resolved to their final states. Resolving futures to their final states may wait for execution to complete.

    Unsupported object types will be returned without modification.

    Source code in prefect/futures.py
    async def resolve_futures_to_states(\n    expr: Union[PrefectFuture[R, Any], Any]\n) -> Union[State[R], Any]:\n    \"\"\"\n    Given a Python built-in collection, recursively find `PrefectFutures` and build a\n    new collection with the same structure with futures resolved to their final states.\n    Resolving futures to their final states may wait for execution to complete.\n\n    Unsupported object types will be returned without modification.\n    \"\"\"\n    futures: Set[PrefectFuture] = set()\n\n    visit_collection(\n        expr,\n        visit_fn=partial(_collect_futures, futures),\n        return_data=False,\n        context={},\n    )\n\n    # Get final states for each future\n    states = await asyncio.gather(\n        *[\n            # We must wait for the future in the thread it was created in\n            from_async.call_soon_in_loop_thread(create_call(future._wait)).aresult()\n            for future in futures\n        ]\n    )\n\n    states_by_future = dict(zip(futures, states))\n\n    def replace_futures_with_states(expr, context):\n        # Expressions inside quotes should not be modified\n        if isinstance(context.get(\"annotation\"), quote):\n            raise StopVisiting()\n\n        if isinstance(expr, PrefectFuture):\n            return states_by_future[expr]\n        else:\n            return expr\n\n    return visit_collection(\n        expr,\n        visit_fn=replace_futures_with_states,\n        return_data=True,\n        context={},\n    )\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/infrastructure/","title":"prefect.infrastructure","text":"","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure","title":"prefect.infrastructure","text":"","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.DockerContainer","title":"DockerContainer","text":"

    Bases: Infrastructure

    Runs a command in a container.

    Requires a Docker Engine to be connectable. Docker settings will be retrieved from the environment.

    Click here to see a tutorial.

    Attributes:

    Name Type Description auto_remove bool

    If set, the container will be removed on completion. Otherwise, the container will remain after exit for inspection.

    command bool

    A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this.

    env bool

    Environment variables to set for the container.

    image str

    An optional string specifying the tag of a Docker image to use. Defaults to the Prefect image.

    image_pull_policy Optional[ImagePullPolicy]

    Specifies if the image should be pulled. One of 'ALWAYS', 'NEVER', 'IF_NOT_PRESENT'.

    image_registry Optional[DockerRegistry]

    A DockerRegistry block containing credentials to use if image is stored in a private image registry.

    labels Optional[DockerRegistry]

    An optional dictionary of labels, mapping name to value.

    name Optional[DockerRegistry]

    An optional name for the container.

    network_mode Optional[str]

    Set the network mode for the created container. Defaults to 'host' if a local API url is detected, otherwise the Docker default of 'bridge' is used. If 'networks' is set, this cannot be set.

    networks List[str]

    An optional list of strings specifying Docker networks to connect the container to.

    stream_output bool

    If set, stream output from the container to local standard output.

    volumes List[str]

    An optional list of volume mount strings in the format of \"local_path:container_path\".

    memswap_limit Union[int, str]

    Total memory (memory + swap), -1 to disable swap. Should only be set if mem_limit is also set. If mem_limit is set, this defaults to allowing the container to use as much swap as memory. For example, if mem_limit is 300m and memswap_limit is not set, the container can use 600m in total of memory and swap.

    mem_limit Union[float, str]

    Memory limit of the created container. Accepts float values to enforce a limit in bytes or a string with a unit e.g. 100000b, 1000k, 128m, 1g. If a string is given without a unit, bytes are assumed.

    privileged bool

    Give extended privileges to this container.

    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.DockerContainer--connecting-to-a-locally-hosted-prefect-api","title":"Connecting to a locally hosted Prefect API","text":"

    If using a local API URL on Linux, we will update the network mode default to 'host' to enable connectivity. If using another OS or an alternative network mode is used, we will replace 'localhost' in the API URL with 'host.docker.internal'. Generally, this will enable connectivity, but the API URL can be provided as an environment variable to override inference in more complex use-cases.

    Note, if using 'host.docker.internal' in the API URL on Linux, the API must be bound to 0.0.0.0 or the Docker IP address to allow connectivity. On macOS, this is not necessary and the API is connectable while bound to localhost.

    Source code in prefect/infrastructure/container.py
    class DockerContainer(Infrastructure):\n    \"\"\"\n    Runs a command in a container.\n\n    Requires a Docker Engine to be connectable. Docker settings will be retrieved from\n    the environment.\n\n    Click [here](https://docs.prefect.io/guides/deployment/docker) to see a tutorial.\n\n    Attributes:\n        auto_remove: If set, the container will be removed on completion. Otherwise,\n            the container will remain after exit for inspection.\n        command: A list of strings specifying the command to run in the container to\n            start the flow run. In most cases you should not override this.\n        env: Environment variables to set for the container.\n        image: An optional string specifying the tag of a Docker image to use.\n            Defaults to the Prefect image.\n        image_pull_policy: Specifies if the image should be pulled. One of 'ALWAYS',\n            'NEVER', 'IF_NOT_PRESENT'.\n        image_registry: A `DockerRegistry` block containing credentials to use if `image` is stored in a private\n            image registry.\n        labels: An optional dictionary of labels, mapping name to value.\n        name: An optional name for the container.\n        network_mode: Set the network mode for the created container. Defaults to 'host'\n            if a local API url is detected, otherwise the Docker default of 'bridge' is\n            used. If 'networks' is set, this cannot be set.\n        networks: An optional list of strings specifying Docker networks to connect the\n            container to.\n        stream_output: If set, stream output from the container to local standard output.\n        volumes: An optional list of volume mount strings in the format of\n            \"local_path:container_path\".\n        memswap_limit: Total memory (memory + swap), -1 to disable swap. Should only be\n            set if `mem_limit` is also set. If `mem_limit` is set, this defaults to\n            allowing the container to use as much swap as memory. For example, if\n            `mem_limit` is 300m and `memswap_limit` is not set, the container can use\n            600m in total of memory and swap.\n        mem_limit: Memory limit of the created container. Accepts float values to enforce\n            a limit in bytes or a string with a unit e.g. 100000b, 1000k, 128m, 1g.\n            If a string is given without a unit, bytes are assumed.\n        privileged: Give extended privileges to this container.\n\n    ## Connecting to a locally hosted Prefect API\n\n    If using a local API URL on Linux, we will update the network mode default to 'host'\n    to enable connectivity. If using another OS or an alternative network mode is used,\n    we will replace 'localhost' in the API URL with 'host.docker.internal'. Generally,\n    this will enable connectivity, but the API URL can be provided as an environment\n    variable to override inference in more complex use-cases.\n\n    Note, if using 'host.docker.internal' in the API URL on Linux, the API must be bound\n    to 0.0.0.0 or the Docker IP address to allow connectivity. On macOS, this is not\n    necessary and the API is connectable while bound to localhost.\n    \"\"\"\n\n    type: Literal[\"docker-container\"] = Field(\n        default=\"docker-container\", description=\"The type of infrastructure.\"\n    )\n    image: str = Field(\n        default_factory=get_prefect_image_name,\n        description=\"Tag of a Docker image to use. Defaults to the Prefect image.\",\n    )\n    image_pull_policy: Optional[ImagePullPolicy] = Field(\n        default=None, description=\"Specifies if the image should be pulled.\"\n    )\n    image_registry: Optional[DockerRegistry] = None\n    networks: List[str] = Field(\n        default_factory=list,\n        description=(\n            \"A list of strings specifying Docker networks to connect the container to.\"\n        ),\n    )\n    network_mode: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The network mode for the created container (e.g. host, bridge). If\"\n            \" 'networks' is set, this cannot be set.\"\n        ),\n    )\n    auto_remove: bool = Field(\n        default=False,\n        description=\"If set, the container will be removed on completion.\",\n    )\n    volumes: List[str] = Field(\n        default_factory=list,\n        description=(\n            \"A list of volume mount strings in the format of\"\n            ' \"local_path:container_path\".'\n        ),\n    )\n    stream_output: bool = Field(\n        default=True,\n        description=(\n            \"If set, the output will be streamed from the container to local standard\"\n            \" output.\"\n        ),\n    )\n    memswap_limit: Union[int, str] = Field(\n        default=None,\n        description=(\n            \"Total memory (memory + swap), -1 to disable swap. Should only be \"\n            \"set if `mem_limit` is also set. If `mem_limit` is set, this defaults to\"\n            \"allowing the container to use as much swap as memory. For example, if \"\n            \"`mem_limit` is 300m and `memswap_limit` is not set, the container can use \"\n            \"600m in total of memory and swap.\"\n        ),\n    )\n    mem_limit: Union[float, str] = Field(\n        default=None,\n        description=(\n            \"Memory limit of the created container. Accepts float values to enforce \"\n            \"a limit in bytes or a string with a unit e.g. 100000b, 1000k, 128m, 1g. \"\n            \"If a string is given without a unit, bytes are assumed.\"\n        ),\n    )\n    privileged: bool = Field(\n        default=False,\n        description=\"Give extended privileges to this container.\",\n    )\n\n    _block_type_name = \"Docker Container\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/14a315b79990200db7341e42553e23650b34bb96-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/infrastructure/#prefect.infrastructure.DockerContainer\"\n\n    @validator(\"labels\")\n    def convert_labels_to_docker_format(cls, labels: Dict[str, str]):\n        labels = labels or {}\n        new_labels = {}\n        for name, value in labels.items():\n            if \"/\" in name:\n                namespace, key = name.split(\"/\", maxsplit=1)\n                new_namespace = \".\".join(reversed(namespace.split(\".\")))\n                new_labels[f\"{new_namespace}.{key}\"] = value\n            else:\n                new_labels[name] = value\n        return new_labels\n\n    @validator(\"volumes\")\n    def check_volume_format(cls, volumes):\n        for volume in volumes:\n            if \":\" not in volume:\n                raise ValueError(\n                    \"Invalid volume specification. \"\n                    f\"Expected format 'path:container_path', but got {volume!r}\"\n                )\n\n        return volumes\n\n    @sync_compatible\n    async def run(\n        self,\n        task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> Optional[bool]:\n        if not self.command:\n            raise ValueError(\"Docker container cannot be run with empty command.\")\n\n        # The `docker` library uses requests instead of an async http library so it must\n        # be run in a thread to avoid blocking the event loop.\n        container = await run_sync_in_worker_thread(self._create_and_start_container)\n        container_pid = self._get_infrastructure_pid(container_id=container.id)\n\n        # Mark as started and return the infrastructure id\n        if task_status:\n            task_status.started(container_pid)\n\n        # Monitor the container\n        container = await run_sync_in_worker_thread(\n            self._watch_container_safe, container\n        )\n\n        exit_code = container.attrs[\"State\"].get(\"ExitCode\")\n        return DockerContainerResult(\n            status_code=exit_code if exit_code is not None else -1,\n            identifier=container_pid,\n        )\n\n    async def kill(self, infrastructure_pid: str, grace_seconds: int = 30):\n        docker_client = self._get_client()\n        base_url, container_id = self._parse_infrastructure_pid(infrastructure_pid)\n\n        if docker_client.api.base_url != base_url:\n            raise InfrastructureNotAvailable(\n                \"\".join(\n                    [\n                        (\n                            f\"Unable to stop container {container_id!r}: the current\"\n                            \" Docker API \"\n                        ),\n                        (\n                            f\"URL {docker_client.api.base_url!r} does not match the\"\n                            \" expected \"\n                        ),\n                        f\"API base URL {base_url}.\",\n                    ]\n                )\n            )\n        try:\n            container = docker_client.containers.get(container_id=container_id)\n        except docker.errors.NotFound:\n            raise InfrastructureNotFound(\n                f\"Unable to stop container {container_id!r}: The container was not\"\n                \" found.\"\n            )\n\n        try:\n            container.stop(timeout=grace_seconds)\n        except Exception:\n            raise\n\n    def preview(self):\n        # TODO: build and document a more sophisticated preview\n        docker_client = self._get_client()\n        try:\n            return json.dumps(self._build_container_settings(docker_client))\n        finally:\n            docker_client.close()\n\n    async def generate_work_pool_base_job_template(self):\n        from prefect.workers.utilities import (\n            get_default_base_job_template_for_infrastructure_type,\n        )\n\n        base_job_template = await get_default_base_job_template_for_infrastructure_type(\n            self.get_corresponding_worker_type()\n        )\n        if base_job_template is None:\n            return await super().generate_work_pool_base_job_template()\n        for key, value in self.dict(exclude_unset=True, exclude_defaults=True).items():\n            if key == \"command\":\n                base_job_template[\"variables\"][\"properties\"][\"command\"][\"default\"] = (\n                    shlex.join(value)\n                )\n            elif key == \"image_registry\":\n                self.logger.warning(\n                    \"Image registry blocks are not supported by Docker\"\n                    \" work pools. Please authenticate to your registry using\"\n                    \" the `docker login` command on your worker instances.\"\n                )\n            elif key in [\n                \"type\",\n                \"block_type_slug\",\n                \"_block_document_id\",\n                \"_block_document_name\",\n                \"_is_anonymous\",\n            ]:\n                continue\n            elif key == \"image_pull_policy\":\n                new_value = None\n                if value == ImagePullPolicy.ALWAYS:\n                    new_value = \"Always\"\n                elif value == ImagePullPolicy.NEVER:\n                    new_value = \"Never\"\n                elif value == ImagePullPolicy.IF_NOT_PRESENT:\n                    new_value = \"IfNotPresent\"\n\n                base_job_template[\"variables\"][\"properties\"][key][\"default\"] = new_value\n            elif key in base_job_template[\"variables\"][\"properties\"]:\n                base_job_template[\"variables\"][\"properties\"][key][\"default\"] = value\n            else:\n                self.logger.warning(\n                    f\"Variable {key!r} is not supported by Docker work pools. Skipping.\"\n                )\n\n        return base_job_template\n\n    def get_corresponding_worker_type(self):\n        return \"docker\"\n\n    def _get_infrastructure_pid(self, container_id: str) -> str:\n        \"\"\"Generates a Docker infrastructure_pid string in the form of\n        `<docker_host_base_url>:<container_id>`.\n        \"\"\"\n        docker_client = self._get_client()\n        base_url = docker_client.api.base_url\n        docker_client.close()\n        return f\"{base_url}:{container_id}\"\n\n    def _parse_infrastructure_pid(self, infrastructure_pid: str) -> Tuple[str, str]:\n        \"\"\"Splits a Docker infrastructure_pid into its component parts\"\"\"\n\n        # base_url can contain `:` so we only want the last item of the split\n        base_url, container_id = infrastructure_pid.rsplit(\":\", 1)\n        return base_url, str(container_id)\n\n    def _build_container_settings(\n        self,\n        docker_client: \"DockerClient\",\n    ) -> Dict:\n        network_mode = self._get_network_mode()\n        return dict(\n            image=self.image,\n            network=self.networks[0] if self.networks else None,\n            network_mode=network_mode,\n            command=self.command,\n            environment=self._get_environment_variables(network_mode),\n            auto_remove=self.auto_remove,\n            labels={**CONTAINER_LABELS, **self.labels},\n            extra_hosts=self._get_extra_hosts(docker_client),\n            name=self._get_container_name(),\n            volumes=self.volumes,\n            mem_limit=self.mem_limit,\n            memswap_limit=self.memswap_limit,\n            privileged=self.privileged,\n        )\n\n    def _create_and_start_container(self) -> \"Container\":\n        if self.image_registry:\n            # If an image registry block was supplied, load an authenticated Docker\n            # client from the block. Otherwise, use an unauthenticated client to\n            # pull images from public registries.\n            docker_client = self.image_registry.get_docker_client()\n        else:\n            docker_client = self._get_client()\n        container_settings = self._build_container_settings(docker_client)\n\n        if self._should_pull_image(docker_client):\n            self.logger.info(f\"Pulling image {self.image!r}...\")\n            self._pull_image(docker_client)\n\n        container = self._create_container(docker_client, **container_settings)\n\n        # Add additional networks after the container is created; only one network can\n        # be attached at creation time\n        if len(self.networks) > 1:\n            for network_name in self.networks[1:]:\n                network = docker_client.networks.get(network_name)\n                network.connect(container)\n\n        # Start the container\n        container.start()\n\n        docker_client.close()\n\n        return container\n\n    def _get_image_and_tag(self) -> Tuple[str, Optional[str]]:\n        return parse_image_tag(self.image)\n\n    def _determine_image_pull_policy(self) -> ImagePullPolicy:\n        \"\"\"\n        Determine the appropriate image pull policy.\n\n        1. If they specified an image pull policy, use that.\n\n        2. If they did not specify an image pull policy and gave us\n           the \"latest\" tag, use ImagePullPolicy.always.\n\n        3. If they did not specify an image pull policy and did not\n           specify a tag, use ImagePullPolicy.always.\n\n        4. If they did not specify an image pull policy and gave us\n           a tag other than \"latest\", use ImagePullPolicy.if_not_present.\n\n        This logic matches the behavior of Kubernetes.\n        See:https://kubernetes.io/docs/concepts/containers/images/#imagepullpolicy-defaulting\n        \"\"\"\n        if not self.image_pull_policy:\n            _, tag = self._get_image_and_tag()\n            if tag == \"latest\" or not tag:\n                return ImagePullPolicy.ALWAYS\n            return ImagePullPolicy.IF_NOT_PRESENT\n        return self.image_pull_policy\n\n    def _get_network_mode(self) -> Optional[str]:\n        # User's value takes precedence; this may collide with the incompatible options\n        # mentioned below.\n        if self.network_mode:\n            if sys.platform != \"linux\" and self.network_mode == \"host\":\n                warnings.warn(\n                    f\"{self.network_mode!r} network mode is not supported on platform \"\n                    f\"{sys.platform!r} and may not work as intended.\"\n                )\n            return self.network_mode\n\n        # Network mode is not compatible with networks or ports (we do not support ports\n        # yet though)\n        if self.networks:\n            return None\n\n        # Check for a local API connection\n        api_url = self.env.get(\"PREFECT_API_URL\", PREFECT_API_URL.value())\n\n        if api_url:\n            try:\n                _, netloc, _, _, _, _ = urllib.parse.urlparse(api_url)\n            except Exception as exc:\n                warnings.warn(\n                    f\"Failed to parse host from API URL {api_url!r} with exception: \"\n                    f\"{exc}\\nThe network mode will not be inferred.\"\n                )\n                return None\n\n            host = netloc.split(\":\")[0]\n\n            # If using a locally hosted API, use a host network on linux\n            if sys.platform == \"linux\" and (host == \"127.0.0.1\" or host == \"localhost\"):\n                return \"host\"\n\n        # Default to unset\n        return None\n\n    def _should_pull_image(self, docker_client: \"DockerClient\") -> bool:\n        \"\"\"\n        Decide whether we need to pull the Docker image.\n        \"\"\"\n        image_pull_policy = self._determine_image_pull_policy()\n\n        if image_pull_policy is ImagePullPolicy.ALWAYS:\n            return True\n        elif image_pull_policy is ImagePullPolicy.NEVER:\n            return False\n        elif image_pull_policy is ImagePullPolicy.IF_NOT_PRESENT:\n            try:\n                # NOTE: images.get() wants the tag included with the image\n                # name, while images.pull() wants them split.\n                docker_client.images.get(self.image)\n            except docker.errors.ImageNotFound:\n                self.logger.debug(f\"Could not find Docker image locally: {self.image}\")\n                return True\n        return False\n\n    def _pull_image(self, docker_client: \"DockerClient\"):\n        \"\"\"\n        Pull the image we're going to use to create the container.\n        \"\"\"\n        image, tag = self._get_image_and_tag()\n\n        return docker_client.images.pull(image, tag)\n\n    def _create_container(self, docker_client: \"DockerClient\", **kwargs) -> \"Container\":\n        \"\"\"\n        Create a docker container with retries on name conflicts.\n\n        If the container already exists with the given name, an incremented index is\n        added.\n        \"\"\"\n        # Create the container with retries on name conflicts (with an incremented idx)\n        index = 0\n        container = None\n        name = original_name = kwargs.pop(\"name\")\n\n        while not container:\n            from docker.errors import APIError\n\n            try:\n                display_name = repr(name) if name else \"with auto-generated name\"\n                self.logger.info(f\"Creating Docker container {display_name}...\")\n                container = docker_client.containers.create(name=name, **kwargs)\n            except APIError as exc:\n                if \"Conflict\" in str(exc) and \"container name\" in str(exc):\n                    self.logger.info(\n                        f\"Docker container name {display_name} already exists; \"\n                        \"retrying...\"\n                    )\n                    index += 1\n                    name = f\"{original_name}-{index}\"\n                else:\n                    raise\n\n        self.logger.info(\n            f\"Docker container {container.name!r} has status {container.status!r}\"\n        )\n        return container\n\n    def _watch_container_safe(self, container: \"Container\") -> \"Container\":\n        # Monitor the container capturing the latest snapshot while capturing\n        # not found errors\n        docker_client = self._get_client()\n\n        try:\n            for latest_container in self._watch_container(docker_client, container.id):\n                container = latest_container\n        except docker.errors.NotFound:\n            # The container was removed during watching\n            self.logger.warning(\n                f\"Docker container {container.name} was removed before we could wait \"\n                \"for its completion.\"\n            )\n        finally:\n            docker_client.close()\n\n        return container\n\n    def _watch_container(\n        self, docker_client: \"DockerClient\", container_id: str\n    ) -> Generator[None, None, \"Container\"]:\n        container: \"Container\" = docker_client.containers.get(container_id)\n\n        status = container.status\n        self.logger.info(\n            f\"Docker container {container.name!r} has status {container.status!r}\"\n        )\n        yield container\n\n        if self.stream_output:\n            try:\n                for log in container.logs(stream=True):\n                    log: bytes\n                    print(log.decode().rstrip())\n            except docker.errors.APIError as exc:\n                if \"marked for removal\" in str(exc):\n                    self.logger.warning(\n                        f\"Docker container {container.name} was marked for removal\"\n                        \" before logs could be retrieved. Output will not be\"\n                        \" streamed. \"\n                    )\n                else:\n                    self.logger.exception(\n                        \"An unexpected Docker API error occurred while streaming\"\n                        f\" output from container {container.name}.\"\n                    )\n\n            container.reload()\n            if container.status != status:\n                self.logger.info(\n                    f\"Docker container {container.name!r} has status\"\n                    f\" {container.status!r}\"\n                )\n            yield container\n\n        container.wait()\n        self.logger.info(\n            f\"Docker container {container.name!r} has status {container.status!r}\"\n        )\n        yield container\n\n    def _get_client(self):\n        try:\n            with warnings.catch_warnings():\n                # Silence warnings due to use of deprecated methods within dockerpy\n                # See https://github.com/docker/docker-py/pull/2931\n                warnings.filterwarnings(\n                    \"ignore\",\n                    message=\"distutils Version classes are deprecated.*\",\n                    category=DeprecationWarning,\n                )\n\n                docker_client = docker.from_env()\n\n        except docker.errors.DockerException as exc:\n            raise RuntimeError(\"Could not connect to Docker.\") from exc\n\n        return docker_client\n\n    def _get_container_name(self) -> Optional[str]:\n        \"\"\"\n        Generates a container name to match the configured name, ensuring it is Docker\n        compatible.\n        \"\"\"\n        # Must match `/?[a-zA-Z0-9][a-zA-Z0-9_.-]+` in the end\n        if not self.name:\n            return None\n\n        return (\n            slugify(\n                self.name,\n                lowercase=False,\n                # Docker does not limit length but URL limits apply eventually so\n                # limit the length for safety\n                max_length=250,\n                # Docker allows these characters for container names\n                regex_pattern=r\"[^a-zA-Z0-9_.-]+\",\n            ).lstrip(\n                # Docker does not allow leading underscore, dash, or period\n                \"_-.\"\n            )\n            # Docker does not allow 0 character names so cast to null if the name is\n            # empty after slufification\n            or None\n        )\n\n    def _get_extra_hosts(self, docker_client) -> Dict[str, str]:\n        \"\"\"\n        A host.docker.internal -> host-gateway mapping is necessary for communicating\n        with the API on Linux machines. Docker Desktop on macOS will automatically\n        already have this mapping.\n        \"\"\"\n        if sys.platform == \"linux\" and (\n            # Do not warn if the user has specified a host manually that does not use\n            # a local address\n            \"PREFECT_API_URL\" not in self.env\n            or re.search(\n                \".*(localhost)|(127.0.0.1)|(host.docker.internal).*\",\n                self.env[\"PREFECT_API_URL\"],\n            )\n        ):\n            user_version = packaging.version.parse(\n                format_outlier_version_name(docker_client.version()[\"Version\"])\n            )\n            required_version = packaging.version.parse(\"20.10.0\")\n\n            if user_version < required_version:\n                warnings.warn(\n                    \"`host.docker.internal` could not be automatically resolved to\"\n                    \" your local ip address. This feature is not supported on Docker\"\n                    f\" Engine v{user_version}, upgrade to v{required_version}+ if you\"\n                    \" encounter issues.\"\n                )\n                return {}\n            else:\n                # Compatibility for linux -- https://github.com/docker/cli/issues/2290\n                # Only supported by Docker v20.10.0+ which is our minimum recommend version\n                return {\"host.docker.internal\": \"host-gateway\"}\n\n    def _get_environment_variables(self, network_mode):\n        # If the API URL has been set by the base environment rather than the by the\n        # user, update the value to ensure connectivity when using a bridge network by\n        # updating local connections to use the docker internal host unless the\n        # network mode is \"host\" where localhost is available already.\n        env = {**self._base_environment(), **self.env}\n\n        if (\n            \"PREFECT_API_URL\" in env\n            and \"PREFECT_API_URL\" not in self.env\n            and network_mode != \"host\"\n        ):\n            env[\"PREFECT_API_URL\"] = (\n                env[\"PREFECT_API_URL\"]\n                .replace(\"localhost\", \"host.docker.internal\")\n                .replace(\"127.0.0.1\", \"host.docker.internal\")\n            )\n\n        # Drop null values allowing users to \"unset\" variables\n        return {key: value for key, value in env.items() if value is not None}\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.DockerContainerResult","title":"DockerContainerResult","text":"

    Bases: InfrastructureResult

    Contains information about a completed Docker container

    Source code in prefect/infrastructure/container.py
    class DockerContainerResult(InfrastructureResult):\n    \"\"\"Contains information about a completed Docker container\"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure","title":"Infrastructure","text":"

    Bases: Block, ABC

    Source code in prefect/infrastructure/base.py
    class Infrastructure(Block, abc.ABC):\n    _block_schema_capabilities = [\"run-infrastructure\"]\n\n    type: str\n\n    env: Dict[str, Optional[str]] = pydantic.Field(\n        default_factory=dict,\n        title=\"Environment\",\n        description=\"Environment variables to set in the configured infrastructure.\",\n    )\n    labels: Dict[str, str] = pydantic.Field(\n        default_factory=dict,\n        description=\"Labels applied to the infrastructure for metadata purposes.\",\n    )\n    name: Optional[str] = pydantic.Field(\n        default=None,\n        description=\"Name applied to the infrastructure for identification.\",\n    )\n    command: Optional[List[str]] = pydantic.Field(\n        default=None,\n        description=\"The command to run in the infrastructure.\",\n    )\n\n    async def generate_work_pool_base_job_template(self):\n        if self._block_document_id is None:\n            raise BlockNotSavedError(\n                \"Cannot publish as work pool, block has not been saved. Please call\"\n                \" `.save()` on your block before publishing.\"\n            )\n\n        block_schema = self.__class__.schema()\n        return {\n            \"job_configuration\": {\"block\": \"{{ block }}\"},\n            \"variables\": {\n                \"type\": \"object\",\n                \"properties\": {\n                    \"block\": {\n                        \"title\": \"Block\",\n                        \"description\": (\n                            \"The infrastructure block to use for job creation.\"\n                        ),\n                        \"allOf\": [{\"$ref\": f\"#/definitions/{self.__class__.__name__}\"}],\n                        \"default\": {\n                            \"$ref\": {\"block_document_id\": str(self._block_document_id)}\n                        },\n                    }\n                },\n                \"required\": [\"block\"],\n                \"definitions\": {self.__class__.__name__: block_schema},\n            },\n        }\n\n    def get_corresponding_worker_type(self):\n        return \"block\"\n\n    @sync_compatible\n    async def publish_as_work_pool(self, work_pool_name: Optional[str] = None):\n        \"\"\"\n        Creates a work pool configured to use the given block as the job creator.\n\n        Used to migrate from a agents setup to a worker setup.\n\n        Args:\n            work_pool_name: The name to give to the created work pool. If not provided, the name of the current\n                block will be used.\n        \"\"\"\n\n        base_job_template = await self.generate_work_pool_base_job_template()\n        work_pool_name = work_pool_name or self._block_document_name\n\n        if work_pool_name is None:\n            raise ValueError(\n                \"`work_pool_name` must be provided if the block has not been saved.\"\n            )\n\n        console = Console()\n\n        try:\n            async with prefect.get_client() as client:\n                work_pool = await client.create_work_pool(\n                    work_pool=WorkPoolCreate(\n                        name=work_pool_name,\n                        type=self.get_corresponding_worker_type(),\n                        base_job_template=base_job_template,\n                    )\n                )\n        except ObjectAlreadyExists:\n            console.print(\n                (\n                    f\"Work pool with name {work_pool_name!r} already exists, please use\"\n                    \" a different name.\"\n                ),\n                style=\"red\",\n            )\n            return\n\n        console.print(\n            f\"Work pool {work_pool.name} created!\",\n            style=\"green\",\n        )\n        if PREFECT_UI_URL:\n            console.print(\n                \"You see your new work pool in the UI at\"\n                f\" {PREFECT_UI_URL.value()}/work-pools/work-pool/{work_pool.name}\"\n            )\n\n        deploy_script = (\n            \"my_flow.deploy(work_pool_name='{work_pool.name}', image='my_image:tag')\"\n        )\n        if not hasattr(self, \"image\"):\n            deploy_script = (\n                \"my_flow.from_source(source='https://github.com/org/repo.git',\"\n                f\" entrypoint='flow.py:my_flow').deploy(work_pool_name='{work_pool.name}')\"\n            )\n        console.print(\n            \"\\nYou can deploy a flow to this work pool by calling\"\n            f\" [blue].deploy[/]:\\n\\n\\t{deploy_script}\\n\"\n        )\n        console.print(\n            \"\\nTo start a worker to execute flow runs in this work pool run:\\n\"\n        )\n        console.print(f\"\\t[blue]prefect worker start --pool {work_pool.name}[/]\\n\")\n\n    @abc.abstractmethod\n    async def run(\n        self,\n        task_status: anyio.abc.TaskStatus = None,\n    ) -> InfrastructureResult:\n        \"\"\"\n        Run the infrastructure.\n\n        If provided a `task_status`, the status will be reported as started when the\n        infrastructure is successfully created. The status return value will be an\n        identifier for the infrastructure.\n\n        The call will then monitor the created infrastructure, returning a result at\n        the end containing a status code indicating if the infrastructure exited cleanly\n        or encountered an error.\n        \"\"\"\n        # Note: implementations should include `sync_compatible`\n\n    @abc.abstractmethod\n    def preview(self) -> str:\n        \"\"\"\n        View a preview of the infrastructure that would be run.\n        \"\"\"\n\n    @property\n    def logger(self):\n        return get_logger(f\"prefect.infrastructure.{self.type}\")\n\n    @property\n    def is_using_a_runner(self):\n        return self.command is not None and \"prefect flow-run execute\" in shlex.join(\n            self.command\n        )\n\n    @classmethod\n    def _base_environment(cls) -> Dict[str, str]:\n        \"\"\"\n        Environment variables that should be passed to all created infrastructure.\n\n        These values should be overridable with the `env` field.\n        \"\"\"\n        return get_current_settings().to_environment_variables(exclude_unset=True)\n\n    def prepare_for_flow_run(\n        self: Self,\n        flow_run: \"FlowRun\",\n        deployment: Optional[\"Deployment\"] = None,\n        flow: Optional[\"Flow\"] = None,\n    ) -> Self:\n        \"\"\"\n        Return an infrastructure block that is prepared to execute a flow run.\n        \"\"\"\n        if deployment is not None:\n            deployment_labels = self._base_deployment_labels(deployment)\n        else:\n            deployment_labels = {}\n\n        if flow is not None:\n            flow_labels = self._base_flow_labels(flow)\n        else:\n            flow_labels = {}\n\n        return self.copy(\n            update={\n                \"env\": {**self._base_flow_run_environment(flow_run), **self.env},\n                \"labels\": {\n                    **self._base_flow_run_labels(flow_run),\n                    **deployment_labels,\n                    **flow_labels,\n                    **self.labels,\n                },\n                \"name\": self.name or flow_run.name,\n                \"command\": self.command or self._base_flow_run_command(),\n            }\n        )\n\n    @staticmethod\n    def _base_flow_run_command() -> List[str]:\n        \"\"\"\n        Generate a command for a flow run job.\n        \"\"\"\n        if experiment_enabled(\"enhanced_cancellation\"):\n            if (\n                PREFECT_EXPERIMENTAL_WARN\n                and PREFECT_EXPERIMENTAL_WARN_ENHANCED_CANCELLATION\n            ):\n                warnings.warn(\n                    EXPERIMENTAL_WARNING.format(\n                        feature=\"Enhanced flow run cancellation\",\n                        group=\"enhanced_cancellation\",\n                        help=\"\",\n                    ),\n                    ExperimentalFeature,\n                    stacklevel=3,\n                )\n            return [\"prefect\", \"flow-run\", \"execute\"]\n\n        return [\"python\", \"-m\", \"prefect.engine\"]\n\n    @staticmethod\n    def _base_flow_run_labels(flow_run: \"FlowRun\") -> Dict[str, str]:\n        \"\"\"\n        Generate a dictionary of labels for a flow run job.\n        \"\"\"\n        return {\n            \"prefect.io/flow-run-id\": str(flow_run.id),\n            \"prefect.io/flow-run-name\": flow_run.name,\n            \"prefect.io/version\": prefect.__version__,\n        }\n\n    @staticmethod\n    def _base_flow_run_environment(flow_run: \"FlowRun\") -> Dict[str, str]:\n        \"\"\"\n        Generate a dictionary of environment variables for a flow run job.\n        \"\"\"\n        environment = {}\n        environment[\"PREFECT__FLOW_RUN_ID\"] = str(flow_run.id)\n        return environment\n\n    @staticmethod\n    def _base_deployment_labels(deployment: \"Deployment\") -> Dict[str, str]:\n        labels = {\n            \"prefect.io/deployment-name\": deployment.name,\n        }\n        if deployment.updated is not None:\n            labels[\"prefect.io/deployment-updated\"] = deployment.updated.in_timezone(\n                \"utc\"\n            ).to_iso8601_string()\n        return labels\n\n    @staticmethod\n    def _base_flow_labels(flow: \"Flow\") -> Dict[str, str]:\n        return {\n            \"prefect.io/flow-name\": flow.name,\n        }\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure.prepare_for_flow_run","title":"prepare_for_flow_run","text":"

    Return an infrastructure block that is prepared to execute a flow run.

    Source code in prefect/infrastructure/base.py
    def prepare_for_flow_run(\n    self: Self,\n    flow_run: \"FlowRun\",\n    deployment: Optional[\"Deployment\"] = None,\n    flow: Optional[\"Flow\"] = None,\n) -> Self:\n    \"\"\"\n    Return an infrastructure block that is prepared to execute a flow run.\n    \"\"\"\n    if deployment is not None:\n        deployment_labels = self._base_deployment_labels(deployment)\n    else:\n        deployment_labels = {}\n\n    if flow is not None:\n        flow_labels = self._base_flow_labels(flow)\n    else:\n        flow_labels = {}\n\n    return self.copy(\n        update={\n            \"env\": {**self._base_flow_run_environment(flow_run), **self.env},\n            \"labels\": {\n                **self._base_flow_run_labels(flow_run),\n                **deployment_labels,\n                **flow_labels,\n                **self.labels,\n            },\n            \"name\": self.name or flow_run.name,\n            \"command\": self.command or self._base_flow_run_command(),\n        }\n    )\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure.preview","title":"preview abstractmethod","text":"

    View a preview of the infrastructure that would be run.

    Source code in prefect/infrastructure/base.py
    @abc.abstractmethod\ndef preview(self) -> str:\n    \"\"\"\n    View a preview of the infrastructure that would be run.\n    \"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure.publish_as_work_pool","title":"publish_as_work_pool async","text":"

    Creates a work pool configured to use the given block as the job creator.

    Used to migrate from a agents setup to a worker setup.

    Parameters:

    Name Type Description Default work_pool_name Optional[str]

    The name to give to the created work pool. If not provided, the name of the current block will be used.

    None Source code in prefect/infrastructure/base.py
    @sync_compatible\nasync def publish_as_work_pool(self, work_pool_name: Optional[str] = None):\n    \"\"\"\n    Creates a work pool configured to use the given block as the job creator.\n\n    Used to migrate from a agents setup to a worker setup.\n\n    Args:\n        work_pool_name: The name to give to the created work pool. If not provided, the name of the current\n            block will be used.\n    \"\"\"\n\n    base_job_template = await self.generate_work_pool_base_job_template()\n    work_pool_name = work_pool_name or self._block_document_name\n\n    if work_pool_name is None:\n        raise ValueError(\n            \"`work_pool_name` must be provided if the block has not been saved.\"\n        )\n\n    console = Console()\n\n    try:\n        async with prefect.get_client() as client:\n            work_pool = await client.create_work_pool(\n                work_pool=WorkPoolCreate(\n                    name=work_pool_name,\n                    type=self.get_corresponding_worker_type(),\n                    base_job_template=base_job_template,\n                )\n            )\n    except ObjectAlreadyExists:\n        console.print(\n            (\n                f\"Work pool with name {work_pool_name!r} already exists, please use\"\n                \" a different name.\"\n            ),\n            style=\"red\",\n        )\n        return\n\n    console.print(\n        f\"Work pool {work_pool.name} created!\",\n        style=\"green\",\n    )\n    if PREFECT_UI_URL:\n        console.print(\n            \"You see your new work pool in the UI at\"\n            f\" {PREFECT_UI_URL.value()}/work-pools/work-pool/{work_pool.name}\"\n        )\n\n    deploy_script = (\n        \"my_flow.deploy(work_pool_name='{work_pool.name}', image='my_image:tag')\"\n    )\n    if not hasattr(self, \"image\"):\n        deploy_script = (\n            \"my_flow.from_source(source='https://github.com/org/repo.git',\"\n            f\" entrypoint='flow.py:my_flow').deploy(work_pool_name='{work_pool.name}')\"\n        )\n    console.print(\n        \"\\nYou can deploy a flow to this work pool by calling\"\n        f\" [blue].deploy[/]:\\n\\n\\t{deploy_script}\\n\"\n    )\n    console.print(\n        \"\\nTo start a worker to execute flow runs in this work pool run:\\n\"\n    )\n    console.print(f\"\\t[blue]prefect worker start --pool {work_pool.name}[/]\\n\")\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure.run","title":"run abstractmethod async","text":"

    Run the infrastructure.

    If provided a task_status, the status will be reported as started when the infrastructure is successfully created. The status return value will be an identifier for the infrastructure.

    The call will then monitor the created infrastructure, returning a result at the end containing a status code indicating if the infrastructure exited cleanly or encountered an error.

    Source code in prefect/infrastructure/base.py
    @abc.abstractmethod\nasync def run(\n    self,\n    task_status: anyio.abc.TaskStatus = None,\n) -> InfrastructureResult:\n    \"\"\"\n    Run the infrastructure.\n\n    If provided a `task_status`, the status will be reported as started when the\n    infrastructure is successfully created. The status return value will be an\n    identifier for the infrastructure.\n\n    The call will then monitor the created infrastructure, returning a result at\n    the end containing a status code indicating if the infrastructure exited cleanly\n    or encountered an error.\n    \"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesClusterConfig","title":"KubernetesClusterConfig","text":"

    Bases: Block

    Stores configuration for interaction with Kubernetes clusters.

    See from_file for creation.

    Attributes:

    Name Type Description config Dict

    The entire loaded YAML contents of a kubectl config file

    context_name str

    The name of the kubectl context to use

    Example

    Load a saved Kubernetes cluster config:

    from prefect.blocks.kubernetes import KubernetesClusterConfig\n\ncluster_config_block = KubernetesClusterConfig.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/kubernetes.py
    class KubernetesClusterConfig(Block):\n    \"\"\"\n    Stores configuration for interaction with Kubernetes clusters.\n\n    See `from_file` for creation.\n\n    Attributes:\n        config: The entire loaded YAML contents of a kubectl config file\n        context_name: The name of the kubectl context to use\n\n    Example:\n        Load a saved Kubernetes cluster config:\n        ```python\n        from prefect.blocks.kubernetes import KubernetesClusterConfig\n\n        cluster_config_block = KubernetesClusterConfig.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Kubernetes Cluster Config\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/2d0b896006ad463b49c28aaac14f31e00e32cfab-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig\"\n\n    config: Dict = Field(\n        default=..., description=\"The entire contents of a kubectl config file.\"\n    )\n    context_name: str = Field(\n        default=..., description=\"The name of the kubectl context to use.\"\n    )\n\n    @validator(\"config\", pre=True)\n    def parse_yaml_config(cls, value):\n        if isinstance(value, str):\n            return yaml.safe_load(value)\n        return value\n\n    @classmethod\n    def from_file(cls: Type[Self], path: Path = None, context_name: str = None) -> Self:\n        \"\"\"\n        Create a cluster config from the a Kubernetes config file.\n\n        By default, the current context in the default Kubernetes config file will be\n        used.\n\n        An alternative file or context may be specified.\n\n        The entire config file will be loaded and stored.\n        \"\"\"\n        kube_config = kubernetes.config.kube_config\n\n        path = Path(path or kube_config.KUBE_CONFIG_DEFAULT_LOCATION)\n        path = path.expanduser().resolve()\n\n        # Determine the context\n        existing_contexts, current_context = kube_config.list_kube_config_contexts(\n            config_file=str(path)\n        )\n        context_names = {ctx[\"name\"] for ctx in existing_contexts}\n        if context_name:\n            if context_name not in context_names:\n                raise ValueError(\n                    f\"Context {context_name!r} not found. \"\n                    f\"Specify one of: {listrepr(context_names, sep=', ')}.\"\n                )\n        else:\n            context_name = current_context[\"name\"]\n\n        # Load the entire config file\n        config_file_contents = path.read_text()\n        config_dict = yaml.safe_load(config_file_contents)\n\n        return cls(config=config_dict, context_name=context_name)\n\n    def get_api_client(self) -> \"ApiClient\":\n        \"\"\"\n        Returns a Kubernetes API client for this cluster config.\n        \"\"\"\n        return kubernetes.config.kube_config.new_client_from_config_dict(\n            config_dict=self.config, context=self.context_name\n        )\n\n    def configure_client(self) -> None:\n        \"\"\"\n        Activates this cluster configuration by loading the configuration into the\n        Kubernetes Python client. After calling this, Kubernetes API clients can use\n        this config's context.\n        \"\"\"\n        kubernetes.config.kube_config.load_kube_config_from_dict(\n            config_dict=self.config, context=self.context_name\n        )\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesClusterConfig.configure_client","title":"configure_client","text":"

    Activates this cluster configuration by loading the configuration into the Kubernetes Python client. After calling this, Kubernetes API clients can use this config's context.

    Source code in prefect/blocks/kubernetes.py
    def configure_client(self) -> None:\n    \"\"\"\n    Activates this cluster configuration by loading the configuration into the\n    Kubernetes Python client. After calling this, Kubernetes API clients can use\n    this config's context.\n    \"\"\"\n    kubernetes.config.kube_config.load_kube_config_from_dict(\n        config_dict=self.config, context=self.context_name\n    )\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesClusterConfig.from_file","title":"from_file classmethod","text":"

    Create a cluster config from the a Kubernetes config file.

    By default, the current context in the default Kubernetes config file will be used.

    An alternative file or context may be specified.

    The entire config file will be loaded and stored.

    Source code in prefect/blocks/kubernetes.py
    @classmethod\ndef from_file(cls: Type[Self], path: Path = None, context_name: str = None) -> Self:\n    \"\"\"\n    Create a cluster config from the a Kubernetes config file.\n\n    By default, the current context in the default Kubernetes config file will be\n    used.\n\n    An alternative file or context may be specified.\n\n    The entire config file will be loaded and stored.\n    \"\"\"\n    kube_config = kubernetes.config.kube_config\n\n    path = Path(path or kube_config.KUBE_CONFIG_DEFAULT_LOCATION)\n    path = path.expanduser().resolve()\n\n    # Determine the context\n    existing_contexts, current_context = kube_config.list_kube_config_contexts(\n        config_file=str(path)\n    )\n    context_names = {ctx[\"name\"] for ctx in existing_contexts}\n    if context_name:\n        if context_name not in context_names:\n            raise ValueError(\n                f\"Context {context_name!r} not found. \"\n                f\"Specify one of: {listrepr(context_names, sep=', ')}.\"\n            )\n    else:\n        context_name = current_context[\"name\"]\n\n    # Load the entire config file\n    config_file_contents = path.read_text()\n    config_dict = yaml.safe_load(config_file_contents)\n\n    return cls(config=config_dict, context_name=context_name)\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesClusterConfig.get_api_client","title":"get_api_client","text":"

    Returns a Kubernetes API client for this cluster config.

    Source code in prefect/blocks/kubernetes.py
    def get_api_client(self) -> \"ApiClient\":\n    \"\"\"\n    Returns a Kubernetes API client for this cluster config.\n    \"\"\"\n    return kubernetes.config.kube_config.new_client_from_config_dict(\n        config_dict=self.config, context=self.context_name\n    )\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob","title":"KubernetesJob","text":"

    Bases: Infrastructure

    Runs a command as a Kubernetes Job.

    For a guided tutorial, see How to use Kubernetes with Prefect. For more information, including examples for customizing the resulting manifest, see KubernetesJob infrastructure concepts.

    Attributes:

    Name Type Description cluster_config Optional[KubernetesClusterConfig]

    An optional Kubernetes cluster config to use for this job.

    command Optional[KubernetesClusterConfig]

    A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this.

    customizations JsonPatch

    A list of JSON 6902 patches to apply to the base Job manifest.

    env JsonPatch

    Environment variables to set for the container.

    finished_job_ttl Optional[int]

    The number of seconds to retain jobs after completion. If set, finished jobs will be cleaned up by Kubernetes after the given delay. If None (default), jobs will need to be manually removed.

    image Optional[str]

    An optional string specifying the image reference of a container image to use for the job, for example, docker.io/prefecthq/prefect:2-latest. The behavior is as described in https://kubernetes.io/docs/concepts/containers/images/#image-names. Defaults to the Prefect image.

    image_pull_policy Optional[KubernetesImagePullPolicy]

    The Kubernetes image pull policy to use for job containers.

    job KubernetesManifest

    The base manifest for the Kubernetes Job.

    job_watch_timeout_seconds Optional[int]

    Number of seconds to wait for the job to complete before marking it as crashed. Defaults to None, which means no timeout will be enforced.

    labels Optional[int]

    An optional dictionary of labels to add to the job.

    name Optional[int]

    An optional name for the job.

    namespace Optional[str]

    An optional string signifying the Kubernetes namespace to use.

    pod_watch_timeout_seconds int

    Number of seconds to watch for pod creation before timing out (default 60).

    service_account_name Optional[str]

    An optional string specifying which Kubernetes service account to use.

    stream_output bool

    If set, stream output from the job to local standard output.

    Source code in prefect/infrastructure/kubernetes.py
    class KubernetesJob(Infrastructure):\n    \"\"\"\n    Runs a command as a Kubernetes Job.\n\n    For a guided tutorial, see [How to use Kubernetes with Prefect](https://medium.com/the-prefect-blog/how-to-use-kubernetes-with-prefect-419b2e8b8cb2/).\n    For more information, including examples for customizing the resulting manifest, see [`KubernetesJob` infrastructure concepts](https://docs.prefect.io/concepts/infrastructure/#kubernetesjob).\n\n    Attributes:\n        cluster_config: An optional Kubernetes cluster config to use for this job.\n        command: A list of strings specifying the command to run in the container to\n            start the flow run. In most cases you should not override this.\n        customizations: A list of JSON 6902 patches to apply to the base Job manifest.\n        env: Environment variables to set for the container.\n        finished_job_ttl: The number of seconds to retain jobs after completion. If set, finished jobs will\n            be cleaned up by Kubernetes after the given delay. If None (default), jobs will need to be\n            manually removed.\n        image: An optional string specifying the image reference of a container image\n            to use for the job, for example, docker.io/prefecthq/prefect:2-latest. The\n            behavior is as described in https://kubernetes.io/docs/concepts/containers/images/#image-names.\n            Defaults to the Prefect image.\n        image_pull_policy: The Kubernetes image pull policy to use for job containers.\n        job: The base manifest for the Kubernetes Job.\n        job_watch_timeout_seconds: Number of seconds to wait for the job to complete\n            before marking it as crashed. Defaults to `None`, which means no timeout will be enforced.\n        labels: An optional dictionary of labels to add to the job.\n        name: An optional name for the job.\n        namespace: An optional string signifying the Kubernetes namespace to use.\n        pod_watch_timeout_seconds: Number of seconds to watch for pod creation before timing out (default 60).\n        service_account_name: An optional string specifying which Kubernetes service account to use.\n        stream_output: If set, stream output from the job to local standard output.\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/2d0b896006ad463b49c28aaac14f31e00e32cfab-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob\"\n\n    type: Literal[\"kubernetes-job\"] = Field(\n        default=\"kubernetes-job\", description=\"The type of infrastructure.\"\n    )\n    # shortcuts for the most common user-serviceable settings\n    image: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The image reference of a container image to use for the job, for example,\"\n            \" `docker.io/prefecthq/prefect:2-latest`.The behavior is as described in\"\n            \" the Kubernetes documentation and uses the latest version of Prefect by\"\n            \" default, unless an image is already present in a provided job manifest.\"\n        ),\n    )\n    namespace: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The Kubernetes namespace to use for this job. Defaults to 'default' \"\n            \"unless a namespace is already present in a provided job manifest.\"\n        ),\n    )\n    service_account_name: Optional[str] = Field(\n        default=None, description=\"The Kubernetes service account to use for this job.\"\n    )\n    image_pull_policy: Optional[KubernetesImagePullPolicy] = Field(\n        default=None,\n        description=\"The Kubernetes image pull policy to use for job containers.\",\n    )\n\n    # connection to a cluster\n    cluster_config: Optional[KubernetesClusterConfig] = Field(\n        default=None, description=\"The Kubernetes cluster config to use for this job.\"\n    )\n\n    # settings allowing full customization of the Job\n    job: KubernetesManifest = Field(\n        default_factory=lambda: KubernetesJob.base_job_manifest(),\n        description=\"The base manifest for the Kubernetes Job.\",\n        title=\"Base Job Manifest\",\n    )\n    customizations: JsonPatch = Field(\n        default_factory=lambda: JsonPatch([]),\n        description=\"A list of JSON 6902 patches to apply to the base Job manifest.\",\n    )\n\n    # controls the behavior of execution\n    job_watch_timeout_seconds: Optional[int] = Field(\n        default=None,\n        description=(\n            \"Number of seconds to wait for the job to complete before marking it as\"\n            \" crashed. Defaults to `None`, which means no timeout will be enforced.\"\n        ),\n    )\n    pod_watch_timeout_seconds: int = Field(\n        default=60,\n        description=\"Number of seconds to watch for pod creation before timing out.\",\n    )\n    stream_output: bool = Field(\n        default=True,\n        description=(\n            \"If set, output will be streamed from the job to local standard output.\"\n        ),\n    )\n    finished_job_ttl: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The number of seconds to retain jobs after completion. If set, finished\"\n            \" jobs will be cleaned up by Kubernetes after the given delay. If None\"\n            \" (default), jobs will need to be manually removed.\"\n        ),\n    )\n\n    # internal-use only right now\n    _api_dns_name: Optional[str] = None  # Replaces 'localhost' in API URL\n\n    _block_type_name = \"Kubernetes Job\"\n\n    @validator(\"job\")\n    def ensure_job_includes_all_required_components(cls, value: KubernetesManifest):\n        patch = JsonPatch.from_diff(value, cls.base_job_manifest())\n        missing_paths = sorted([op[\"path\"] for op in patch if op[\"op\"] == \"add\"])\n        if missing_paths:\n            raise ValueError(\n                \"Job is missing required attributes at the following paths: \"\n                f\"{', '.join(missing_paths)}\"\n            )\n        return value\n\n    @validator(\"job\")\n    def ensure_job_has_compatible_values(cls, value: KubernetesManifest):\n        patch = JsonPatch.from_diff(value, cls.base_job_manifest())\n        incompatible = sorted(\n            [\n                f\"{op['path']} must have value {op['value']!r}\"\n                for op in patch\n                if op[\"op\"] == \"replace\"\n            ]\n        )\n        if incompatible:\n            raise ValueError(\n                \"Job has incompatible values for the following attributes: \"\n                f\"{', '.join(incompatible)}\"\n            )\n        return value\n\n    @validator(\"customizations\", pre=True)\n    def cast_customizations_to_a_json_patch(\n        cls, value: Union[List[Dict], JsonPatch, str]\n    ) -> JsonPatch:\n        if isinstance(value, list):\n            return JsonPatch(value)\n        elif isinstance(value, str):\n            try:\n                return JsonPatch(json.loads(value))\n            except json.JSONDecodeError as exc:\n                raise ValueError(\n                    f\"Unable to parse customizations as JSON: {value}. Please make sure\"\n                    \" that the provided value is a valid JSON string.\"\n                ) from exc\n        return value\n\n    @root_validator\n    def default_namespace(cls, values):\n        job = values.get(\"job\")\n\n        namespace = values.get(\"namespace\")\n        job_namespace = job[\"metadata\"].get(\"namespace\") if job else None\n\n        if not namespace and not job_namespace:\n            values[\"namespace\"] = \"default\"\n\n        return values\n\n    @root_validator\n    def default_image(cls, values):\n        job = values.get(\"job\")\n        image = values.get(\"image\")\n        job_image = (\n            job[\"spec\"][\"template\"][\"spec\"][\"containers\"][0].get(\"image\")\n            if job\n            else None\n        )\n\n        if not image and not job_image:\n            values[\"image\"] = get_prefect_image_name()\n\n        return values\n\n    # Support serialization of the 'JsonPatch' type\n    class Config:\n        arbitrary_types_allowed = True\n        json_encoders = {JsonPatch: lambda p: p.patch}\n\n    def dict(self, *args, **kwargs) -> Dict:\n        d = super().dict(*args, **kwargs)\n        d[\"customizations\"] = self.customizations.patch\n        return d\n\n    @classmethod\n    def base_job_manifest(cls) -> KubernetesManifest:\n        \"\"\"Produces the bare minimum allowed Job manifest\"\"\"\n        return {\n            \"apiVersion\": \"batch/v1\",\n            \"kind\": \"Job\",\n            \"metadata\": {\"labels\": {}},\n            \"spec\": {\n                \"template\": {\n                    \"spec\": {\n                        \"parallelism\": 1,\n                        \"completions\": 1,\n                        \"restartPolicy\": \"Never\",\n                        \"containers\": [\n                            {\n                                \"name\": \"prefect-job\",\n                                \"env\": [],\n                            }\n                        ],\n                    }\n                }\n            },\n        }\n\n    # Note that we're using the yaml package to load both YAML and JSON files below.\n    # This works because YAML is a strict superset of JSON:\n    #\n    #   > The YAML 1.23 specification was published in 2009. Its primary focus was\n    #   > making YAML a strict superset of JSON. It also removed many of the problematic\n    #   > implicit typing recommendations.\n    #\n    #   https://yaml.org/spec/1.2.2/#12-yaml-history\n\n    @classmethod\n    def job_from_file(cls, filename: str) -> KubernetesManifest:\n        \"\"\"Load a Kubernetes Job manifest from a YAML or JSON file.\"\"\"\n        with open(filename, \"r\", encoding=\"utf-8\") as f:\n            return yaml.load(f, yaml.SafeLoader)\n\n    @classmethod\n    def customize_from_file(cls, filename: str) -> JsonPatch:\n        \"\"\"Load an RFC 6902 JSON patch from a YAML or JSON file.\"\"\"\n        with open(filename, \"r\", encoding=\"utf-8\") as f:\n            return JsonPatch(yaml.load(f, yaml.SafeLoader))\n\n    @sync_compatible\n    async def run(\n        self,\n        task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> KubernetesJobResult:\n        if not self.command:\n            raise ValueError(\"Kubernetes job cannot be run with empty command.\")\n\n        self._configure_kubernetes_library_client()\n        manifest = self.build_job()\n        job = await run_sync_in_worker_thread(self._create_job, manifest)\n\n        pid = await run_sync_in_worker_thread(self._get_infrastructure_pid, job)\n        # Indicate that the job has started\n        if task_status is not None:\n            task_status.started(pid)\n\n        # Monitor the job until completion\n        status_code = await run_sync_in_worker_thread(\n            self._watch_job, job.metadata.name\n        )\n        return KubernetesJobResult(identifier=pid, status_code=status_code)\n\n    async def kill(self, infrastructure_pid: str, grace_seconds: int = 30):\n        self._configure_kubernetes_library_client()\n        job_cluster_uid, job_namespace, job_name = self._parse_infrastructure_pid(\n            infrastructure_pid\n        )\n\n        if not job_namespace == self.namespace:\n            raise InfrastructureNotAvailable(\n                f\"Unable to kill job {job_name!r}: The job is running in namespace \"\n                f\"{job_namespace!r} but this block is configured to use \"\n                f\"{self.namespace!r}.\"\n            )\n\n        current_cluster_uid = self._get_cluster_uid()\n        if job_cluster_uid != current_cluster_uid:\n            raise InfrastructureNotAvailable(\n                f\"Unable to kill job {job_name!r}: The job is running on another \"\n                \"cluster.\"\n            )\n\n        with self.get_batch_client() as batch_client:\n            try:\n                batch_client.delete_namespaced_job(\n                    name=job_name,\n                    namespace=job_namespace,\n                    grace_period_seconds=grace_seconds,\n                    # Foreground propagation deletes dependent objects before deleting owner objects.\n                    # This ensures that the pods are cleaned up before the job is marked as deleted.\n                    # See: https://kubernetes.io/docs/concepts/architecture/garbage-collection/#foreground-deletion\n                    propagation_policy=\"Foreground\",\n                )\n            except kubernetes.client.exceptions.ApiException as exc:\n                if exc.status == 404:\n                    raise InfrastructureNotFound(\n                        f\"Unable to kill job {job_name!r}: The job was not found.\"\n                    ) from exc\n                else:\n                    raise\n\n    def preview(self):\n        return yaml.dump(self.build_job())\n\n    def get_corresponding_worker_type(self):\n        return \"kubernetes\"\n\n    async def generate_work_pool_base_job_template(self):\n        from prefect.workers.utilities import (\n            get_default_base_job_template_for_infrastructure_type,\n        )\n\n        base_job_template = await get_default_base_job_template_for_infrastructure_type(\n            self.get_corresponding_worker_type()\n        )\n        assert (\n            base_job_template is not None\n        ), \"Failed to retrieve default base job template.\"\n        for key, value in self.dict(exclude_unset=True, exclude_defaults=True).items():\n            if key == \"command\":\n                base_job_template[\"variables\"][\"properties\"][\"command\"][\"default\"] = (\n                    shlex.join(value)\n                )\n            elif key in [\n                \"type\",\n                \"block_type_slug\",\n                \"_block_document_id\",\n                \"_block_document_name\",\n                \"_is_anonymous\",\n                \"job\",\n                \"customizations\",\n            ]:\n                continue\n            elif key == \"image_pull_policy\":\n                base_job_template[\"variables\"][\"properties\"][\"image_pull_policy\"][\n                    \"default\"\n                ] = value.value\n            elif key == \"cluster_config\":\n                base_job_template[\"variables\"][\"properties\"][\"cluster_config\"][\n                    \"default\"\n                ] = {\n                    \"$ref\": {\n                        \"block_document_id\": str(self.cluster_config._block_document_id)\n                    }\n                }\n            elif key in base_job_template[\"variables\"][\"properties\"]:\n                base_job_template[\"variables\"][\"properties\"][key][\"default\"] = value\n            else:\n                self.logger.warning(\n                    f\"Variable {key!r} is not supported by Kubernetes work pools.\"\n                    \" Skipping.\"\n                )\n\n        custom_job_manifest = self.dict(exclude_unset=True, exclude_defaults=True).get(\n            \"job\"\n        )\n        if custom_job_manifest:\n            job_manifest = self.build_job()\n        else:\n            job_manifest = copy.deepcopy(\n                base_job_template[\"job_configuration\"][\"job_manifest\"]\n            )\n            job_manifest = self.customizations.apply(job_manifest)\n        base_job_template[\"job_configuration\"][\"job_manifest\"] = job_manifest\n\n        return base_job_template\n\n    def build_job(self) -> KubernetesManifest:\n        \"\"\"Builds the Kubernetes Job Manifest\"\"\"\n        job_manifest = copy.copy(self.job)\n        job_manifest = self._shortcut_customizations().apply(job_manifest)\n        job_manifest = self.customizations.apply(job_manifest)\n        return job_manifest\n\n    @contextmanager\n    def get_batch_client(self) -> Generator[\"BatchV1Api\", None, None]:\n        with kubernetes.client.ApiClient() as client:\n            try:\n                yield kubernetes.client.BatchV1Api(api_client=client)\n            finally:\n                client.rest_client.pool_manager.clear()\n\n    @contextmanager\n    def get_client(self) -> Generator[\"CoreV1Api\", None, None]:\n        with kubernetes.client.ApiClient() as client:\n            try:\n                yield kubernetes.client.CoreV1Api(api_client=client)\n            finally:\n                client.rest_client.pool_manager.clear()\n\n    def _get_infrastructure_pid(self, job: \"V1Job\") -> str:\n        \"\"\"\n        Generates a Kubernetes infrastructure PID.\n\n        The PID is in the format: \"<cluster uid>:<namespace>:<job name>\".\n        \"\"\"\n        cluster_uid = self._get_cluster_uid()\n        pid = f\"{cluster_uid}:{self.namespace}:{job.metadata.name}\"\n        return pid\n\n    def _parse_infrastructure_pid(\n        self, infrastructure_pid: str\n    ) -> Tuple[str, str, str]:\n        \"\"\"\n        Parse a Kubernetes infrastructure PID into its component parts.\n\n        Returns a cluster UID, namespace, and job name.\n        \"\"\"\n        cluster_uid, namespace, job_name = infrastructure_pid.split(\":\", 2)\n        return cluster_uid, namespace, job_name\n\n    def _get_cluster_uid(self) -> str:\n        \"\"\"\n        Gets a unique id for the current cluster being used.\n\n        There is no real unique identifier for a cluster. However, the `kube-system`\n        namespace is immutable and has a persistence UID that we use instead.\n\n        PREFECT_KUBERNETES_CLUSTER_UID can be set in cases where the `kube-system`\n        namespace cannot be read e.g. when a cluster role cannot be created. If set,\n        this variable will be used and we will not attempt to read the `kube-system`\n        namespace.\n\n        See https://github.com/kubernetes/kubernetes/issues/44954\n        \"\"\"\n        # Default to an environment variable\n        env_cluster_uid = os.environ.get(\"PREFECT_KUBERNETES_CLUSTER_UID\")\n        if env_cluster_uid:\n            return env_cluster_uid\n\n        # Read the UID from the cluster namespace\n        with self.get_client() as client:\n            namespace = client.read_namespace(\"kube-system\")\n        cluster_uid = namespace.metadata.uid\n\n        return cluster_uid\n\n    def _configure_kubernetes_library_client(self) -> None:\n        \"\"\"\n        Set the correct kubernetes client configuration.\n\n        WARNING: This action is not threadsafe and may override the configuration\n                  specified by another `KubernetesJob` instance.\n        \"\"\"\n        # TODO: Investigate returning a configured client so calls on other threads\n        #       will not invalidate the config needed here\n\n        # if a k8s cluster block is provided to the flow runner, use that\n        if self.cluster_config:\n            self.cluster_config.configure_client()\n        else:\n            # If no block specified, try to load Kubernetes configuration within a cluster. If that doesn't\n            # work, try to load the configuration from the local environment, allowing\n            # any further ConfigExceptions to bubble up.\n            try:\n                kubernetes.config.load_incluster_config()\n            except kubernetes.config.ConfigException:\n                kubernetes.config.load_kube_config()\n\n    def _shortcut_customizations(self) -> JsonPatch:\n        \"\"\"Produces the JSON 6902 patch for the most commonly used customizations, like\n        image and namespace, which we offer as top-level parameters (with sensible\n        default values)\"\"\"\n        shortcuts = []\n\n        if self.namespace:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/metadata/namespace\",\n                    \"value\": self.namespace,\n                }\n            )\n\n        if self.image:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/template/spec/containers/0/image\",\n                    \"value\": self.image,\n                }\n            )\n\n        shortcuts += [\n            {\n                \"op\": \"add\",\n                \"path\": (\n                    f\"/metadata/labels/{self._slugify_label_key(key).replace('/', '~1', 1)}\"\n                ),\n                \"value\": self._slugify_label_value(value),\n            }\n            for key, value in self.labels.items()\n        ]\n\n        shortcuts += [\n            {\n                \"op\": \"add\",\n                \"path\": \"/spec/template/spec/containers/0/env/-\",\n                \"value\": {\"name\": key, \"value\": value},\n            }\n            for key, value in self._get_environment_variables().items()\n        ]\n\n        if self.image_pull_policy:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/template/spec/containers/0/imagePullPolicy\",\n                    \"value\": self.image_pull_policy.value,\n                }\n            )\n\n        if self.service_account_name:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/template/spec/serviceAccountName\",\n                    \"value\": self.service_account_name,\n                }\n            )\n\n        if self.finished_job_ttl is not None:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/ttlSecondsAfterFinished\",\n                    \"value\": self.finished_job_ttl,\n                }\n            )\n\n        if self.command:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/template/spec/containers/0/args\",\n                    \"value\": self.command,\n                }\n            )\n\n        if self.name:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/metadata/generateName\",\n                    \"value\": self._slugify_name(self.name) + \"-\",\n                }\n            )\n        else:\n            # Generate name is required\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/metadata/generateName\",\n                    \"value\": (\n                        \"prefect-job-\"\n                        # We generate a name using a hash of the primary job settings\n                        + stable_hash(\n                            *self.command,\n                            *self.env.keys(),\n                            *[v for v in self.env.values() if v is not None],\n                        )\n                        + \"-\"\n                    ),\n                }\n            )\n\n        return JsonPatch(shortcuts)\n\n    def _get_job(self, job_id: str) -> Optional[\"V1Job\"]:\n        with self.get_batch_client() as batch_client:\n            try:\n                job = batch_client.read_namespaced_job(job_id, self.namespace)\n            except kubernetes.client.exceptions.ApiException:\n                self.logger.error(f\"Job {job_id!r} was removed.\", exc_info=True)\n                return None\n            return job\n\n    def _get_job_pod(self, job_name: str) -> \"V1Pod\":\n        \"\"\"Get the first running pod for a job.\"\"\"\n\n        # Wait until we find a running pod for the job\n        # if `pod_watch_timeout_seconds` is None, no timeout will be enforced\n        watch = kubernetes.watch.Watch()\n        self.logger.debug(f\"Job {job_name!r}: Starting watch for pod start...\")\n        last_phase = None\n        with self.get_client() as client:\n            for event in watch.stream(\n                func=client.list_namespaced_pod,\n                namespace=self.namespace,\n                label_selector=f\"job-name={job_name}\",\n                timeout_seconds=self.pod_watch_timeout_seconds,\n            ):\n                phase = event[\"object\"].status.phase\n                if phase != last_phase:\n                    self.logger.info(f\"Job {job_name!r}: Pod has status {phase!r}.\")\n\n                if phase != \"Pending\":\n                    watch.stop()\n                    return event[\"object\"]\n\n                last_phase = phase\n\n        self.logger.error(f\"Job {job_name!r}: Pod never started.\")\n\n    def _watch_job(self, job_name: str) -> int:\n        \"\"\"\n        Watch a job.\n\n        Return the final status code of the first container.\n        \"\"\"\n        self.logger.debug(f\"Job {job_name!r}: Monitoring job...\")\n\n        job = self._get_job(job_name)\n        if not job:\n            return -1\n\n        pod = self._get_job_pod(job_name)\n        if not pod:\n            return -1\n\n        # Calculate the deadline before streaming output\n        deadline = (\n            (time.monotonic() + self.job_watch_timeout_seconds)\n            if self.job_watch_timeout_seconds is not None\n            else None\n        )\n\n        if self.stream_output:\n            with self.get_client() as client:\n                logs = client.read_namespaced_pod_log(\n                    pod.metadata.name,\n                    self.namespace,\n                    follow=True,\n                    _preload_content=False,\n                    container=\"prefect-job\",\n                )\n                try:\n                    for log in logs.stream():\n                        print(log.decode().rstrip())\n\n                        # Check if we have passed the deadline and should stop streaming\n                        # logs\n                        remaining_time = (\n                            deadline - time.monotonic() if deadline else None\n                        )\n                        if deadline and remaining_time <= 0:\n                            break\n\n                except Exception:\n                    self.logger.warning(\n                        (\n                            \"Error occurred while streaming logs - \"\n                            \"Job will continue to run but logs will \"\n                            \"no longer be streamed to stdout.\"\n                        ),\n                        exc_info=True,\n                    )\n\n        with self.get_batch_client() as batch_client:\n            # Check if the job is completed before beginning a watch\n            job = batch_client.read_namespaced_job(\n                name=job_name, namespace=self.namespace\n            )\n            completed = job.status.completion_time is not None\n\n            while not completed:\n                remaining_time = (\n                    math.ceil(deadline - time.monotonic()) if deadline else None\n                )\n                if deadline and remaining_time <= 0:\n                    self.logger.error(\n                        f\"Job {job_name!r}: Job did not complete within \"\n                        f\"timeout of {self.job_watch_timeout_seconds}s.\"\n                    )\n                    return -1\n\n                watch = kubernetes.watch.Watch()\n                # The kubernetes library will disable retries if the timeout kwarg is\n                # present regardless of the value so we do not pass it unless given\n                # https://github.com/kubernetes-client/python/blob/84f5fea2a3e4b161917aa597bf5e5a1d95e24f5a/kubernetes/base/watch/watch.py#LL160\n                timeout_seconds = (\n                    {\"timeout_seconds\": remaining_time} if deadline else {}\n                )\n\n                for event in watch.stream(\n                    func=batch_client.list_namespaced_job,\n                    field_selector=f\"metadata.name={job_name}\",\n                    namespace=self.namespace,\n                    **timeout_seconds,\n                ):\n                    if event[\"type\"] == \"DELETED\":\n                        self.logger.error(f\"Job {job_name!r}: Job has been deleted.\")\n                        completed = True\n                    elif event[\"object\"].status.completion_time:\n                        if not event[\"object\"].status.succeeded:\n                            # Job failed, exit while loop and return pod exit code\n                            self.logger.error(f\"Job {job_name!r}: Job failed.\")\n                        completed = True\n                    # Check if the job has reached its backoff limit\n                    # and stop watching if it has\n                    elif (\n                        event[\"object\"].spec.backoff_limit is not None\n                        and event[\"object\"].status.failed is not None\n                        and event[\"object\"].status.failed\n                        > event[\"object\"].spec.backoff_limit\n                    ):\n                        self.logger.error(\n                            f\"Job {job_name!r}: Job reached backoff limit.\"\n                        )\n                        completed = True\n                    # If the job has no backoff limit, check if it has failed\n                    # and stop watching if it has\n                    elif (\n                        not event[\"object\"].spec.backoff_limit\n                        and event[\"object\"].status.failed\n                    ):\n                        completed = True\n\n                    if completed:\n                        watch.stop()\n                        break\n\n        with self.get_client() as core_client:\n            # Get all pods for the job\n            pods = core_client.list_namespaced_pod(\n                namespace=self.namespace, label_selector=f\"job-name={job_name}\"\n            )\n            # Get the status for only the most recently used pod\n            pods.items.sort(\n                key=lambda pod: pod.metadata.creation_timestamp, reverse=True\n            )\n            most_recent_pod = pods.items[0] if pods.items else None\n            first_container_status = (\n                most_recent_pod.status.container_statuses[0]\n                if most_recent_pod\n                else None\n            )\n            if not first_container_status:\n                self.logger.error(f\"Job {job_name!r}: No pods found for job.\")\n                return -1\n\n            # In some cases, such as spot instance evictions, the pod will be forcibly\n            # terminated and not report a status correctly.\n            elif (\n                first_container_status.state is None\n                or first_container_status.state.terminated is None\n                or first_container_status.state.terminated.exit_code is None\n            ):\n                self.logger.error(\n                    f\"Could not determine exit code for {job_name!r}.\"\n                    \"Exit code will be reported as -1.\"\n                    \"First container status info did not report an exit code.\"\n                    f\"First container info: {first_container_status}.\"\n                )\n                return -1\n\n        return first_container_status.state.terminated.exit_code\n\n    def _create_job(self, job_manifest: KubernetesManifest) -> \"V1Job\":\n        \"\"\"\n        Given a Kubernetes Job Manifest, create the Job on the configured Kubernetes\n        cluster and return its name.\n        \"\"\"\n        with self.get_batch_client() as batch_client:\n            job = batch_client.create_namespaced_job(self.namespace, job_manifest)\n        return job\n\n    def _slugify_name(self, name: str) -> str:\n        \"\"\"\n        Slugify text for use as a name.\n\n        Keeps only alphanumeric characters and dashes, and caps the length\n        of the slug at 45 chars.\n\n        The 45 character length allows room for the k8s utility\n        \"generateName\" to generate a unique name from the slug while\n        keeping the total length of a name below 63 characters, which is\n        the limit for e.g. label names that follow RFC 1123 (hostnames) and\n        RFC 1035 (domain names).\n\n        Args:\n            name: The name of the job\n\n        Returns:\n            the slugified job name\n        \"\"\"\n        slug = slugify(\n            name,\n            max_length=45,  # Leave enough space for generateName\n            regex_pattern=r\"[^a-zA-Z0-9-]+\",\n        )\n\n        # TODO: Handle the case that the name is an empty string after being\n        # slugified.\n\n        return slug\n\n    def _slugify_label_key(self, key: str) -> str:\n        \"\"\"\n        Slugify text for use as a label key.\n\n        Keys are composed of an optional prefix and name, separated by a slash (/).\n\n        Keeps only alphanumeric characters, dashes, underscores, and periods.\n        Limits the length of the label prefix to 253 characters.\n        Limits the length of the label name to 63 characters.\n\n        See https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#syntax-and-character-set\n\n        Args:\n            key: The label key\n\n        Returns:\n            The slugified label key\n        \"\"\"\n        if \"/\" in key:\n            prefix, name = key.split(\"/\", maxsplit=1)\n        else:\n            prefix = None\n            name = key\n\n        name_slug = (\n            slugify(name, max_length=63, regex_pattern=r\"[^a-zA-Z0-9-_.]+\").strip(\n                \"_-.\"  # Must start or end with alphanumeric characters\n            )\n            or name\n        )\n        # Fallback to the original if we end up with an empty slug, this will allow\n        # Kubernetes to throw the validation error\n\n        if prefix:\n            prefix_slug = (\n                slugify(\n                    prefix,\n                    max_length=253,\n                    regex_pattern=r\"[^a-zA-Z0-9-\\.]+\",\n                ).strip(\n                    \"_-.\"\n                )  # Must start or end with alphanumeric characters\n                or prefix\n            )\n\n            return f\"{prefix_slug}/{name_slug}\"\n\n        return name_slug\n\n    def _slugify_label_value(self, value: str) -> str:\n        \"\"\"\n        Slugify text for use as a label value.\n\n        Keeps only alphanumeric characters, dashes, underscores, and periods.\n        Limits the total length of label text to below 63 characters.\n\n        See https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#syntax-and-character-set\n\n        Args:\n            value: The text for the label\n\n        Returns:\n            The slugified value\n        \"\"\"\n        slug = (\n            slugify(value, max_length=63, regex_pattern=r\"[^a-zA-Z0-9-_\\.]+\").strip(\n                \"_-.\"  # Must start or end with alphanumeric characters\n            )\n            or value\n        )\n        # Fallback to the original if we end up with an empty slug, this will allow\n        # Kubernetes to throw the validation error\n\n        return slug\n\n    def _get_environment_variables(self):\n        # If the API URL has been set by the base environment rather than the by the\n        # user, update the value to ensure connectivity when using a bridge network by\n        # updating local connections to use the internal host\n        env = {**self._base_environment(), **self.env}\n\n        if (\n            \"PREFECT_API_URL\" in env\n            and \"PREFECT_API_URL\" not in self.env\n            and self._api_dns_name\n        ):\n            env[\"PREFECT_API_URL\"] = (\n                env[\"PREFECT_API_URL\"]\n                .replace(\"localhost\", self._api_dns_name)\n                .replace(\"127.0.0.1\", self._api_dns_name)\n            )\n\n        # Drop null values allowing users to \"unset\" variables\n        return {key: value for key, value in env.items() if value is not None}\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob.base_job_manifest","title":"base_job_manifest classmethod","text":"

    Produces the bare minimum allowed Job manifest

    Source code in prefect/infrastructure/kubernetes.py
    @classmethod\ndef base_job_manifest(cls) -> KubernetesManifest:\n    \"\"\"Produces the bare minimum allowed Job manifest\"\"\"\n    return {\n        \"apiVersion\": \"batch/v1\",\n        \"kind\": \"Job\",\n        \"metadata\": {\"labels\": {}},\n        \"spec\": {\n            \"template\": {\n                \"spec\": {\n                    \"parallelism\": 1,\n                    \"completions\": 1,\n                    \"restartPolicy\": \"Never\",\n                    \"containers\": [\n                        {\n                            \"name\": \"prefect-job\",\n                            \"env\": [],\n                        }\n                    ],\n                }\n            }\n        },\n    }\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob.build_job","title":"build_job","text":"

    Builds the Kubernetes Job Manifest

    Source code in prefect/infrastructure/kubernetes.py
    def build_job(self) -> KubernetesManifest:\n    \"\"\"Builds the Kubernetes Job Manifest\"\"\"\n    job_manifest = copy.copy(self.job)\n    job_manifest = self._shortcut_customizations().apply(job_manifest)\n    job_manifest = self.customizations.apply(job_manifest)\n    return job_manifest\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob.customize_from_file","title":"customize_from_file classmethod","text":"

    Load an RFC 6902 JSON patch from a YAML or JSON file.

    Source code in prefect/infrastructure/kubernetes.py
    @classmethod\ndef customize_from_file(cls, filename: str) -> JsonPatch:\n    \"\"\"Load an RFC 6902 JSON patch from a YAML or JSON file.\"\"\"\n    with open(filename, \"r\", encoding=\"utf-8\") as f:\n        return JsonPatch(yaml.load(f, yaml.SafeLoader))\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob.job_from_file","title":"job_from_file classmethod","text":"

    Load a Kubernetes Job manifest from a YAML or JSON file.

    Source code in prefect/infrastructure/kubernetes.py
    @classmethod\ndef job_from_file(cls, filename: str) -> KubernetesManifest:\n    \"\"\"Load a Kubernetes Job manifest from a YAML or JSON file.\"\"\"\n    with open(filename, \"r\", encoding=\"utf-8\") as f:\n        return yaml.load(f, yaml.SafeLoader)\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJobResult","title":"KubernetesJobResult","text":"

    Bases: InfrastructureResult

    Contains information about the final state of a completed Kubernetes Job

    Source code in prefect/infrastructure/kubernetes.py
    class KubernetesJobResult(InfrastructureResult):\n    \"\"\"Contains information about the final state of a completed Kubernetes Job\"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Process","title":"Process","text":"

    Bases: Infrastructure

    Run a command in a new process.

    Current environment variables and Prefect settings will be included in the created process. Configured environment variables will override any current environment variables.

    Attributes:

    Name Type Description command

    A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this.

    env

    Environment variables to set for the new process.

    labels

    Labels for the process. Labels are for metadata purposes only and cannot be attached to the process itself.

    name

    A name for the process. For display purposes only.

    stream_output bool

    Whether to stream output to local stdout.

    working_dir Union[str, Path, None]

    Working directory where the process should be opened. If not set, a tmp directory will be used.

    Source code in prefect/infrastructure/process.py
    class Process(Infrastructure):\n    \"\"\"\n    Run a command in a new process.\n\n    Current environment variables and Prefect settings will be included in the created\n    process. Configured environment variables will override any current environment\n    variables.\n\n    Attributes:\n        command: A list of strings specifying the command to run in the container to\n            start the flow run. In most cases you should not override this.\n        env: Environment variables to set for the new process.\n        labels: Labels for the process. Labels are for metadata purposes only and\n            cannot be attached to the process itself.\n        name: A name for the process. For display purposes only.\n        stream_output: Whether to stream output to local stdout.\n        working_dir: Working directory where the process should be opened. If not set,\n            a tmp directory will be used.\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/356e6766a91baf20e1d08bbe16e8b5aaef4d8643-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/infrastructure/#process\"\n\n    type: Literal[\"process\"] = Field(\n        default=\"process\", description=\"The type of infrastructure.\"\n    )\n    stream_output: bool = Field(\n        default=True,\n        description=(\n            \"If set, output will be streamed from the process to local standard output.\"\n        ),\n    )\n    working_dir: Union[str, Path, None] = Field(\n        default=None,\n        description=(\n            \"If set, the process will open within the specified path as the working\"\n            \" directory. Otherwise, a temporary directory will be created.\"\n        ),\n    )  # Underlying accepted types are str, bytes, PathLike[str], None\n\n    @sync_compatible\n    async def run(\n        self,\n        task_status: anyio.abc.TaskStatus = None,\n    ) -> \"ProcessResult\":\n        if not self.command:\n            raise ValueError(\"Process cannot be run with empty command.\")\n\n        _use_threaded_child_watcher()\n        display_name = f\" {self.name!r}\" if self.name else \"\"\n\n        # Open a subprocess to execute the flow run\n        self.logger.info(f\"Opening process{display_name}...\")\n        working_dir_ctx = (\n            tempfile.TemporaryDirectory(suffix=\"prefect\")\n            if not self.working_dir\n            else contextlib.nullcontext(self.working_dir)\n        )\n        with working_dir_ctx as working_dir:\n            self.logger.debug(\n                f\"Process{display_name} running command: {' '.join(self.command)} in\"\n                f\" {working_dir}\"\n            )\n\n            # We must add creationflags to a dict so it is only passed as a function\n            # parameter on Windows, because the presence of creationflags causes\n            # errors on Unix even if set to None\n            kwargs: Dict[str, object] = {}\n            if sys.platform == \"win32\":\n                kwargs[\"creationflags\"] = subprocess.CREATE_NEW_PROCESS_GROUP\n\n            process = await run_process(\n                self.command,\n                stream_output=self.stream_output,\n                task_status=task_status,\n                task_status_handler=_infrastructure_pid_from_process,\n                env=self._get_environment_variables(),\n                cwd=working_dir,\n                **kwargs,\n            )\n\n        # Use the pid for display if no name was given\n        display_name = display_name or f\" {process.pid}\"\n\n        if process.returncode:\n            help_message = None\n            if process.returncode == -9:\n                help_message = (\n                    \"This indicates that the process exited due to a SIGKILL signal. \"\n                    \"Typically, this is either caused by manual cancellation or \"\n                    \"high memory usage causing the operating system to \"\n                    \"terminate the process.\"\n                )\n            if process.returncode == -15:\n                help_message = (\n                    \"This indicates that the process exited due to a SIGTERM signal. \"\n                    \"Typically, this is caused by manual cancellation.\"\n                )\n            elif process.returncode == 247:\n                help_message = (\n                    \"This indicates that the process was terminated due to high \"\n                    \"memory usage.\"\n                )\n            elif (\n                sys.platform == \"win32\" and process.returncode == STATUS_CONTROL_C_EXIT\n            ):\n                help_message = (\n                    \"Process was terminated due to a Ctrl+C or Ctrl+Break signal. \"\n                    \"Typically, this is caused by manual cancellation.\"\n                )\n\n            self.logger.error(\n                f\"Process{display_name} exited with status code: {process.returncode}\"\n                + (f\"; {help_message}\" if help_message else \"\")\n            )\n        else:\n            self.logger.info(f\"Process{display_name} exited cleanly.\")\n\n        return ProcessResult(\n            status_code=process.returncode, identifier=str(process.pid)\n        )\n\n    async def kill(self, infrastructure_pid: str, grace_seconds: int = 30):\n        hostname, pid = _parse_infrastructure_pid(infrastructure_pid)\n\n        if hostname != socket.gethostname():\n            raise InfrastructureNotAvailable(\n                f\"Unable to kill process {pid!r}: The process is running on a different\"\n                f\" host {hostname!r}.\"\n            )\n\n        # In a non-windows environment first send a SIGTERM, then, after\n        # `grace_seconds` seconds have passed subsequent send SIGKILL. In\n        # Windows we use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        if sys.platform == \"win32\":\n            try:\n                os.kill(pid, signal.CTRL_BREAK_EVENT)\n            except (ProcessLookupError, WindowsError):\n                raise InfrastructureNotFound(\n                    f\"Unable to kill process {pid!r}: The process was not found.\"\n                )\n        else:\n            try:\n                os.kill(pid, signal.SIGTERM)\n            except ProcessLookupError:\n                raise InfrastructureNotFound(\n                    f\"Unable to kill process {pid!r}: The process was not found.\"\n                )\n\n            # Throttle how often we check if the process is still alive to keep\n            # from making too many system calls in a short period of time.\n            check_interval = max(grace_seconds / 10, 1)\n\n            with anyio.move_on_after(grace_seconds):\n                while True:\n                    await anyio.sleep(check_interval)\n\n                    # Detect if the process is still alive. If not do an early\n                    # return as the process respected the SIGTERM from above.\n                    try:\n                        os.kill(pid, 0)\n                    except ProcessLookupError:\n                        return\n\n            try:\n                os.kill(pid, signal.SIGKILL)\n            except OSError:\n                # We shouldn't ever end up here, but it's possible that the\n                # process ended right after the check above.\n                return\n\n    def preview(self):\n        environment = self._get_environment_variables(include_os_environ=False)\n        return \" \\\\\\n\".join(\n            [f\"{key}={value}\" for key, value in environment.items()]\n            + [\" \".join(self.command)]\n        )\n\n    def _get_environment_variables(self, include_os_environ: bool = True):\n        os_environ = os.environ if include_os_environ else {}\n        # The base environment must override the current environment or\n        # the Prefect settings context may not be respected\n        env = {**os_environ, **self._base_environment(), **self.env}\n\n        # Drop null values allowing users to \"unset\" variables\n        return {key: value for key, value in env.items() if value is not None}\n\n    def _base_flow_run_command(self):\n        return [get_sys_executable(), \"-m\", \"prefect.engine\"]\n\n    def get_corresponding_worker_type(self):\n        return \"process\"\n\n    async def generate_work_pool_base_job_template(self):\n        from prefect.workers.utilities import (\n            get_default_base_job_template_for_infrastructure_type,\n        )\n\n        base_job_template = await get_default_base_job_template_for_infrastructure_type(\n            self.get_corresponding_worker_type(),\n        )\n        assert (\n            base_job_template is not None\n        ), \"Failed to generate default base job template for Process worker.\"\n        for key, value in self.dict(exclude_unset=True, exclude_defaults=True).items():\n            if key == \"command\":\n                base_job_template[\"variables\"][\"properties\"][\"command\"][\"default\"] = (\n                    shlex.join(value)\n                )\n            elif key in [\n                \"type\",\n                \"block_type_slug\",\n                \"_block_document_id\",\n                \"_block_document_name\",\n                \"_is_anonymous\",\n            ]:\n                continue\n            elif key in base_job_template[\"variables\"][\"properties\"]:\n                base_job_template[\"variables\"][\"properties\"][key][\"default\"] = value\n            else:\n                self.logger.warning(\n                    f\"Variable {key!r} is not supported by Process work pools.\"\n                    \" Skipping.\"\n                )\n\n        return base_job_template\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.ProcessResult","title":"ProcessResult","text":"

    Bases: InfrastructureResult

    Contains information about the final state of a completed process

    Source code in prefect/infrastructure/process.py
    class ProcessResult(InfrastructureResult):\n    \"\"\"Contains information about the final state of a completed process\"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/logging/","title":"Logging","text":"","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/#prefect.logging","title":"prefect.logging","text":"","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/#prefect.logging.get_logger","title":"get_logger cached","text":"

    Get a prefect logger. These loggers are intended for internal use within the prefect package.

    See get_run_logger for retrieving loggers for use within task or flow runs. By default, only run-related loggers are connected to the APILogHandler.

    Source code in prefect/logging/loggers.py
    @lru_cache()\ndef get_logger(name: str = None) -> logging.Logger:\n    \"\"\"\n    Get a `prefect` logger. These loggers are intended for internal use within the\n    `prefect` package.\n\n    See `get_run_logger` for retrieving loggers for use within task or flow runs.\n    By default, only run-related loggers are connected to the `APILogHandler`.\n    \"\"\"\n\n    parent_logger = logging.getLogger(\"prefect\")\n\n    if name:\n        # Append the name if given but allow explicit full names e.g. \"prefect.test\"\n        # should not become \"prefect.prefect.test\"\n        if not name.startswith(parent_logger.name + \".\"):\n            logger = parent_logger.getChild(name)\n        else:\n            logger = logging.getLogger(name)\n    else:\n        logger = parent_logger\n\n    return logger\n
    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/#prefect.logging.get_run_logger","title":"get_run_logger","text":"

    Get a Prefect logger for the current task run or flow run.

    The logger will be named either prefect.task_runs or prefect.flow_runs. Contextual data about the run will be attached to the log records.

    These loggers are connected to the APILogHandler by default to send log records to the API.

    Parameters:

    Name Type Description Default context RunContext

    A specific context may be provided as an override. By default, the context is inferred from global state and this should not be needed.

    None **kwargs str

    Additional keyword arguments will be attached to the log records in addition to the run metadata

    {}

    Raises:

    Type Description RuntimeError

    If no context can be found

    Source code in prefect/logging/loggers.py
    def get_run_logger(\n    context: \"RunContext\" = None, **kwargs: str\n) -> Union[logging.Logger, logging.LoggerAdapter]:\n    \"\"\"\n    Get a Prefect logger for the current task run or flow run.\n\n    The logger will be named either `prefect.task_runs` or `prefect.flow_runs`.\n    Contextual data about the run will be attached to the log records.\n\n    These loggers are connected to the `APILogHandler` by default to send log records to\n    the API.\n\n    Arguments:\n        context: A specific context may be provided as an override. By default, the\n            context is inferred from global state and this should not be needed.\n        **kwargs: Additional keyword arguments will be attached to the log records in\n            addition to the run metadata\n\n    Raises:\n        RuntimeError: If no context can be found\n    \"\"\"\n    # Check for existing contexts\n    task_run_context = prefect.context.TaskRunContext.get()\n    flow_run_context = prefect.context.FlowRunContext.get()\n\n    # Apply the context override\n    if context:\n        if isinstance(context, prefect.context.FlowRunContext):\n            flow_run_context = context\n        elif isinstance(context, prefect.context.TaskRunContext):\n            task_run_context = context\n        else:\n            raise TypeError(\n                f\"Received unexpected type {type(context).__name__!r} for context. \"\n                \"Expected one of 'None', 'FlowRunContext', or 'TaskRunContext'.\"\n            )\n\n    # Determine if this is a task or flow run logger\n    if task_run_context:\n        logger = task_run_logger(\n            task_run=task_run_context.task_run,\n            task=task_run_context.task,\n            flow_run=flow_run_context.flow_run if flow_run_context else None,\n            flow=flow_run_context.flow if flow_run_context else None,\n            **kwargs,\n        )\n    elif flow_run_context:\n        logger = flow_run_logger(\n            flow_run=flow_run_context.flow_run, flow=flow_run_context.flow, **kwargs\n        )\n    elif (\n        get_logger(\"prefect.flow_run\").disabled\n        and get_logger(\"prefect.task_run\").disabled\n    ):\n        logger = logging.getLogger(\"null\")\n    else:\n        raise MissingContextError(\"There is no active flow or task run context.\")\n\n    return logger\n
    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/manifests/","title":"prefect.manifests","text":"","tags":["Python API","deployments"]},{"location":"api-ref/prefect/manifests/#prefect.manifests","title":"prefect.manifests","text":"

    Manifests are portable descriptions of one or more workflows within a given directory structure.

    They are the foundational building blocks for defining Flow Deployments.

    ","tags":["Python API","deployments"]},{"location":"api-ref/prefect/manifests/#prefect.manifests.Manifest","title":"Manifest","text":"

    Bases: BaseModel

    A JSON representation of a flow.

    Source code in prefect/manifests.py
    class Manifest(BaseModel):\n    \"\"\"A JSON representation of a flow.\"\"\"\n\n    flow_name: str = Field(default=..., description=\"The name of the flow.\")\n    import_path: str = Field(\n        default=..., description=\"The relative import path for the flow.\"\n    )\n    parameter_openapi_schema: ParameterSchema = Field(\n        default=..., description=\"The OpenAPI schema of the flow's parameters.\"\n    )\n
    ","tags":["Python API","deployments"]},{"location":"api-ref/prefect/serializers/","title":"prefect.serializers","text":"","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers","title":"prefect.serializers","text":"

    Serializer implementations for converting objects to bytes and bytes to objects.

    All serializers are based on the Serializer class and include a type string that allows them to be referenced without referencing the actual class. For example, you can get often specify the JSONSerializer with the string \"json\". Some serializers support additional settings for configuration of serialization. These are stored on the instance so the same settings can be used to load saved objects.

    All serializers must implement dumps and loads which convert objects to bytes and bytes to an object respectively.

    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.CompressedJSONSerializer","title":"CompressedJSONSerializer","text":"

    Bases: CompressedSerializer

    A compressed serializer preconfigured to use the json serializer.

    Source code in prefect/serializers.py
    class CompressedJSONSerializer(CompressedSerializer):\n    \"\"\"\n    A compressed serializer preconfigured to use the json serializer.\n    \"\"\"\n\n    type: Literal[\"compressed/json\"] = \"compressed/json\"\n    serializer: Serializer = pydantic.Field(default_factory=JSONSerializer)\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.CompressedPickleSerializer","title":"CompressedPickleSerializer","text":"

    Bases: CompressedSerializer

    A compressed serializer preconfigured to use the pickle serializer.

    Source code in prefect/serializers.py
    class CompressedPickleSerializer(CompressedSerializer):\n    \"\"\"\n    A compressed serializer preconfigured to use the pickle serializer.\n    \"\"\"\n\n    type: Literal[\"compressed/pickle\"] = \"compressed/pickle\"\n    serializer: Serializer = pydantic.Field(default_factory=PickleSerializer)\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.CompressedSerializer","title":"CompressedSerializer","text":"

    Bases: Serializer

    Wraps another serializer, compressing its output. Uses lzma by default. See compressionlib for using alternative libraries.

    Attributes:

    Name Type Description serializer Serializer

    The serializer to use before compression.

    compressionlib str

    The import path of a compression module to use. Must have methods compress(bytes) -> bytes and decompress(bytes) -> bytes.

    level str

    If not null, the level of compression to pass to compress.

    Source code in prefect/serializers.py
    class CompressedSerializer(Serializer):\n    \"\"\"\n    Wraps another serializer, compressing its output.\n    Uses `lzma` by default. See `compressionlib` for using alternative libraries.\n\n    Attributes:\n        serializer: The serializer to use before compression.\n        compressionlib: The import path of a compression module to use.\n            Must have methods `compress(bytes) -> bytes` and `decompress(bytes) -> bytes`.\n        level: If not null, the level of compression to pass to `compress`.\n    \"\"\"\n\n    type: Literal[\"compressed\"] = \"compressed\"\n\n    serializer: Serializer\n    compressionlib: str = \"lzma\"\n\n    @pydantic.validator(\"serializer\", pre=True)\n    def cast_type_names_to_serializers(cls, value):\n        if isinstance(value, str):\n            return Serializer(type=value)\n        return value\n\n    @pydantic.validator(\"compressionlib\")\n    def check_compressionlib(cls, value):\n        \"\"\"\n        Check that the given pickle library is importable and has compress/decompress\n        methods.\n        \"\"\"\n        try:\n            compressor = from_qualified_name(value)\n        except (ImportError, AttributeError) as exc:\n            raise ValueError(\n                f\"Failed to import requested compression library: {value!r}.\"\n            ) from exc\n\n        if not callable(getattr(compressor, \"compress\", None)):\n            raise ValueError(\n                f\"Compression library at {value!r} does not have a 'compress' method.\"\n            )\n\n        if not callable(getattr(compressor, \"decompress\", None)):\n            raise ValueError(\n                f\"Compression library at {value!r} does not have a 'decompress' method.\"\n            )\n\n        return value\n\n    def dumps(self, obj: Any) -> bytes:\n        blob = self.serializer.dumps(obj)\n        compressor = from_qualified_name(self.compressionlib)\n        return base64.encodebytes(compressor.compress(blob))\n\n    def loads(self, blob: bytes) -> Any:\n        compressor = from_qualified_name(self.compressionlib)\n        uncompressed = compressor.decompress(base64.decodebytes(blob))\n        return self.serializer.loads(uncompressed)\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.CompressedSerializer.check_compressionlib","title":"check_compressionlib","text":"

    Check that the given pickle library is importable and has compress/decompress methods.

    Source code in prefect/serializers.py
    @pydantic.validator(\"compressionlib\")\ndef check_compressionlib(cls, value):\n    \"\"\"\n    Check that the given pickle library is importable and has compress/decompress\n    methods.\n    \"\"\"\n    try:\n        compressor = from_qualified_name(value)\n    except (ImportError, AttributeError) as exc:\n        raise ValueError(\n            f\"Failed to import requested compression library: {value!r}.\"\n        ) from exc\n\n    if not callable(getattr(compressor, \"compress\", None)):\n        raise ValueError(\n            f\"Compression library at {value!r} does not have a 'compress' method.\"\n        )\n\n    if not callable(getattr(compressor, \"decompress\", None)):\n        raise ValueError(\n            f\"Compression library at {value!r} does not have a 'decompress' method.\"\n        )\n\n    return value\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.JSONSerializer","title":"JSONSerializer","text":"

    Bases: Serializer

    Serializes data to JSON.

    Input types must be compatible with the stdlib json library.

    Wraps the json library to serialize to UTF-8 bytes instead of string types.

    Source code in prefect/serializers.py
    class JSONSerializer(Serializer):\n    \"\"\"\n    Serializes data to JSON.\n\n    Input types must be compatible with the stdlib json library.\n\n    Wraps the `json` library to serialize to UTF-8 bytes instead of string types.\n    \"\"\"\n\n    type: Literal[\"json\"] = \"json\"\n    jsonlib: str = \"json\"\n    object_encoder: Optional[str] = pydantic.Field(\n        default=\"prefect.serializers.prefect_json_object_encoder\",\n        description=(\n            \"An optional callable to use when serializing objects that are not \"\n            \"supported by the JSON encoder. By default, this is set to a callable that \"\n            \"adds support for all types supported by Pydantic.\"\n        ),\n    )\n    object_decoder: Optional[str] = pydantic.Field(\n        default=\"prefect.serializers.prefect_json_object_decoder\",\n        description=(\n            \"An optional callable to use when deserializing objects. This callable \"\n            \"is passed each dictionary encountered during JSON deserialization. \"\n            \"By default, this is set to a callable that deserializes content created \"\n            \"by our default `object_encoder`.\"\n        ),\n    )\n    dumps_kwargs: dict = pydantic.Field(default_factory=dict)\n    loads_kwargs: dict = pydantic.Field(default_factory=dict)\n\n    @pydantic.validator(\"dumps_kwargs\")\n    def dumps_kwargs_cannot_contain_default(cls, value):\n        # `default` is set by `object_encoder`. A user provided callable would make this\n        # class unserializable anyway.\n        if \"default\" in value:\n            raise ValueError(\n                \"`default` cannot be provided. Use `object_encoder` instead.\"\n            )\n        return value\n\n    @pydantic.validator(\"loads_kwargs\")\n    def loads_kwargs_cannot_contain_object_hook(cls, value):\n        # `object_hook` is set by `object_decoder`. A user provided callable would make\n        # this class unserializable anyway.\n        if \"object_hook\" in value:\n            raise ValueError(\n                \"`object_hook` cannot be provided. Use `object_decoder` instead.\"\n            )\n        return value\n\n    def dumps(self, data: Any) -> bytes:\n        json = from_qualified_name(self.jsonlib)\n        kwargs = self.dumps_kwargs.copy()\n        if self.object_encoder:\n            kwargs[\"default\"] = from_qualified_name(self.object_encoder)\n        result = json.dumps(data, **kwargs)\n        if isinstance(result, str):\n            # The standard library returns str but others may return bytes directly\n            result = result.encode()\n        return result\n\n    def loads(self, blob: bytes) -> Any:\n        json = from_qualified_name(self.jsonlib)\n        kwargs = self.loads_kwargs.copy()\n        if self.object_decoder:\n            kwargs[\"object_hook\"] = from_qualified_name(self.object_decoder)\n        return json.loads(blob.decode(), **kwargs)\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.PickleSerializer","title":"PickleSerializer","text":"

    Bases: Serializer

    Serializes objects using the pickle protocol.

    • Uses cloudpickle by default. See picklelib for using alternative libraries.
    • Stores the version of the pickle library to check for compatibility during deserialization.
    • Wraps pickles in base64 for safe transmission.
    Source code in prefect/serializers.py
    class PickleSerializer(Serializer):\n    \"\"\"\n    Serializes objects using the pickle protocol.\n\n    - Uses `cloudpickle` by default. See `picklelib` for using alternative libraries.\n    - Stores the version of the pickle library to check for compatibility during\n        deserialization.\n    - Wraps pickles in base64 for safe transmission.\n    \"\"\"\n\n    type: Literal[\"pickle\"] = \"pickle\"\n\n    picklelib: str = \"cloudpickle\"\n    picklelib_version: str = None\n\n    @pydantic.validator(\"picklelib\")\n    def check_picklelib(cls, value):\n        \"\"\"\n        Check that the given pickle library is importable and has dumps/loads methods.\n        \"\"\"\n        try:\n            pickler = from_qualified_name(value)\n        except (ImportError, AttributeError) as exc:\n            raise ValueError(\n                f\"Failed to import requested pickle library: {value!r}.\"\n            ) from exc\n\n        if not callable(getattr(pickler, \"dumps\", None)):\n            raise ValueError(\n                f\"Pickle library at {value!r} does not have a 'dumps' method.\"\n            )\n\n        if not callable(getattr(pickler, \"loads\", None)):\n            raise ValueError(\n                f\"Pickle library at {value!r} does not have a 'loads' method.\"\n            )\n\n        return value\n\n    @pydantic.root_validator\n    def check_picklelib_version(cls, values):\n        \"\"\"\n        Infers a default value for `picklelib_version` if null or ensures it matches\n        the version retrieved from the `pickelib`.\n        \"\"\"\n        picklelib = values.get(\"picklelib\")\n        picklelib_version = values.get(\"picklelib_version\")\n\n        if not picklelib:\n            raise ValueError(\"Unable to check version of unrecognized picklelib module\")\n\n        pickler = from_qualified_name(picklelib)\n        pickler_version = getattr(pickler, \"__version__\", None)\n\n        if not picklelib_version:\n            values[\"picklelib_version\"] = pickler_version\n        elif picklelib_version != pickler_version:\n            warnings.warn(\n                (\n                    f\"Mismatched {picklelib!r} versions. Found {pickler_version} in the\"\n                    f\" environment but {picklelib_version} was requested. This may\"\n                    \" cause the serializer to fail.\"\n                ),\n                RuntimeWarning,\n                stacklevel=3,\n            )\n\n        return values\n\n    def dumps(self, obj: Any) -> bytes:\n        pickler = from_qualified_name(self.picklelib)\n        blob = pickler.dumps(obj)\n        return base64.encodebytes(blob)\n\n    def loads(self, blob: bytes) -> Any:\n        pickler = from_qualified_name(self.picklelib)\n        return pickler.loads(base64.decodebytes(blob))\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.PickleSerializer.check_picklelib","title":"check_picklelib","text":"

    Check that the given pickle library is importable and has dumps/loads methods.

    Source code in prefect/serializers.py
    @pydantic.validator(\"picklelib\")\ndef check_picklelib(cls, value):\n    \"\"\"\n    Check that the given pickle library is importable and has dumps/loads methods.\n    \"\"\"\n    try:\n        pickler = from_qualified_name(value)\n    except (ImportError, AttributeError) as exc:\n        raise ValueError(\n            f\"Failed to import requested pickle library: {value!r}.\"\n        ) from exc\n\n    if not callable(getattr(pickler, \"dumps\", None)):\n        raise ValueError(\n            f\"Pickle library at {value!r} does not have a 'dumps' method.\"\n        )\n\n    if not callable(getattr(pickler, \"loads\", None)):\n        raise ValueError(\n            f\"Pickle library at {value!r} does not have a 'loads' method.\"\n        )\n\n    return value\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.PickleSerializer.check_picklelib_version","title":"check_picklelib_version","text":"

    Infers a default value for picklelib_version if null or ensures it matches the version retrieved from the pickelib.

    Source code in prefect/serializers.py
    @pydantic.root_validator\ndef check_picklelib_version(cls, values):\n    \"\"\"\n    Infers a default value for `picklelib_version` if null or ensures it matches\n    the version retrieved from the `pickelib`.\n    \"\"\"\n    picklelib = values.get(\"picklelib\")\n    picklelib_version = values.get(\"picklelib_version\")\n\n    if not picklelib:\n        raise ValueError(\"Unable to check version of unrecognized picklelib module\")\n\n    pickler = from_qualified_name(picklelib)\n    pickler_version = getattr(pickler, \"__version__\", None)\n\n    if not picklelib_version:\n        values[\"picklelib_version\"] = pickler_version\n    elif picklelib_version != pickler_version:\n        warnings.warn(\n            (\n                f\"Mismatched {picklelib!r} versions. Found {pickler_version} in the\"\n                f\" environment but {picklelib_version} was requested. This may\"\n                \" cause the serializer to fail.\"\n            ),\n            RuntimeWarning,\n            stacklevel=3,\n        )\n\n    return values\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.Serializer","title":"Serializer","text":"

    Bases: BaseModel, Generic[D], ABC

    A serializer that can encode objects of type 'D' into bytes.

    Source code in prefect/serializers.py
    @add_type_dispatch\nclass Serializer(BaseModel, Generic[D], abc.ABC):\n    \"\"\"\n    A serializer that can encode objects of type 'D' into bytes.\n    \"\"\"\n\n    type: str\n\n    @abc.abstractmethod\n    def dumps(self, obj: D) -> bytes:\n        \"\"\"Encode the object into a blob of bytes.\"\"\"\n\n    @abc.abstractmethod\n    def loads(self, blob: bytes) -> D:\n        \"\"\"Decode the blob of bytes into an object.\"\"\"\n\n    class Config:\n        extra = \"forbid\"\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.Serializer.dumps","title":"dumps abstractmethod","text":"

    Encode the object into a blob of bytes.

    Source code in prefect/serializers.py
    @abc.abstractmethod\ndef dumps(self, obj: D) -> bytes:\n    \"\"\"Encode the object into a blob of bytes.\"\"\"\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.Serializer.loads","title":"loads abstractmethod","text":"

    Decode the blob of bytes into an object.

    Source code in prefect/serializers.py
    @abc.abstractmethod\ndef loads(self, blob: bytes) -> D:\n    \"\"\"Decode the blob of bytes into an object.\"\"\"\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.prefect_json_object_decoder","title":"prefect_json_object_decoder","text":"

    JSONDecoder.object_hook for decoding objects from JSON when previously encoded with prefect_json_object_encoder

    Source code in prefect/serializers.py
    def prefect_json_object_decoder(result: dict):\n    \"\"\"\n    `JSONDecoder.object_hook` for decoding objects from JSON when previously encoded\n    with `prefect_json_object_encoder`\n    \"\"\"\n    if \"__class__\" in result:\n        return pydantic.parse_obj_as(\n            from_qualified_name(result[\"__class__\"]), result[\"data\"]\n        )\n    elif \"__exc_type__\" in result:\n        return from_qualified_name(result[\"__exc_type__\"])(result[\"message\"])\n    else:\n        return result\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.prefect_json_object_encoder","title":"prefect_json_object_encoder","text":"

    JSONEncoder.default for encoding objects into JSON with extended type support.

    Raises a TypeError to fallback on other encoders on failure.

    Source code in prefect/serializers.py
    def prefect_json_object_encoder(obj: Any) -> Any:\n    \"\"\"\n    `JSONEncoder.default` for encoding objects into JSON with extended type support.\n\n    Raises a `TypeError` to fallback on other encoders on failure.\n    \"\"\"\n    if isinstance(obj, BaseException):\n        return {\"__exc_type__\": to_qualified_name(obj.__class__), \"message\": str(obj)}\n    else:\n        return {\n            \"__class__\": to_qualified_name(obj.__class__),\n            \"data\": pydantic_encoder(obj),\n        }\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/settings/","title":"prefect.settings","text":"","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings","title":"prefect.settings","text":"

    Prefect settings management.

    Each setting is defined as a Setting type. The name of each setting is stylized in all caps, matching the environment variable that can be used to change the setting.

    All settings defined in this file are used to generate a dynamic Pydantic settings class called Settings. When instantiated, this class will load settings from environment variables and pull default values from the setting definitions.

    The current instance of Settings being used by the application is stored in a SettingsContext model which allows each instance of the Settings class to be accessed in an async-safe manner.

    Aside from environment variables, we allow settings to be changed during the runtime of the process using profiles. Profiles contain setting overrides that the user may persist without setting environment variables. Profiles are also used internally for managing settings during task run execution where differing settings may be used concurrently in the same process and during testing where we need to override settings to ensure their value is respected as intended.

    The SettingsContext is set when the prefect module is imported. This context is referred to as the \"root\" settings context for clarity. Generally, this is the only settings context that will be used. When this context is entered, we will instantiate a Settings object, loading settings from environment variables and defaults, then we will load the active profile and use it to override settings. See enter_root_settings_context for details on determining the active profile.

    Another SettingsContext may be entered at any time to change the settings being used by the code within the context. Generally, users should not use this. Settings management should be left to Prefect application internals.

    Generally, settings should be accessed with SETTING_VARIABLE.value() which will pull the current Settings instance from the current SettingsContext and retrieve the value of the relevant setting.

    Accessing a setting's value will also call the Setting.value_callback which allows settings to be dynamically modified on retrieval. This allows us to make settings dependent on the value of other settings or perform other dynamic effects.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_HOME","title":"PREFECT_HOME = Setting(Path, default=Path('~') / '.prefect', value_callback=expanduser_in_path) module-attribute","text":"

    Prefect's home directory. Defaults to ~/.prefect. This directory may be created automatically when required.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXTRA_ENTRYPOINTS","title":"PREFECT_EXTRA_ENTRYPOINTS = Setting(str, default='') module-attribute","text":"

    Modules for Prefect to import when Prefect is imported.

    Values should be separated by commas, e.g. my_module,my_other_module. Objects within modules may be specified by a ':' partition, e.g. my_module:my_object. If a callable object is provided, it will be called with no arguments on import.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_DEBUG_MODE","title":"PREFECT_DEBUG_MODE = Setting(bool, default=False) module-attribute","text":"

    If True, places the API in debug mode. This may modify behavior to facilitate debugging, including extra logs and other verbose assistance. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLI_COLORS","title":"PREFECT_CLI_COLORS = Setting(bool, default=True) module-attribute","text":"

    If True, use colors in CLI output. If False, output will not include colors codes. Defaults to True.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLI_PROMPT","title":"PREFECT_CLI_PROMPT = Setting(Optional[bool], default=None) module-attribute","text":"

    If True, use interactive prompts in CLI commands. If False, no interactive prompts will be used. If None, the value will be dynamically determined based on the presence of an interactive-enabled terminal.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLI_WRAP_LINES","title":"PREFECT_CLI_WRAP_LINES = Setting(bool, default=True) module-attribute","text":"

    If True, wrap text by inserting new lines in long lines in CLI output. If False, output will not be wrapped. Defaults to True.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TEST_MODE","title":"PREFECT_TEST_MODE = Setting(bool, default=False) module-attribute","text":"

    If True, places the API in test mode. This may modify behavior to facilitate testing. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_UNIT_TEST_MODE","title":"PREFECT_UNIT_TEST_MODE = Setting(bool, default=False) module-attribute","text":"

    This variable only exists to facilitate unit testing. If True, code is executing in a unit test context. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TEST_SETTING","title":"PREFECT_TEST_SETTING = Setting(Any, default=None, value_callback=only_return_value_in_test_mode) module-attribute","text":"

    This variable only exists to facilitate testing of settings. If accessed when PREFECT_TEST_MODE is not set, None is returned.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_TLS_INSECURE_SKIP_VERIFY","title":"PREFECT_API_TLS_INSECURE_SKIP_VERIFY = Setting(bool, default=False) module-attribute","text":"

    If True, disables SSL checking to allow insecure requests. This is recommended only during development, e.g. when using self-signed certificates.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_URL","title":"PREFECT_API_URL = Setting(str, default=None) module-attribute","text":"

    If provided, the URL of a hosted Prefect API. Defaults to None.

    When using Prefect Cloud, this will include an account and workspace.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_KEY","title":"PREFECT_API_KEY = Setting(str, default=None, is_secret=True) module-attribute","text":"

    API key used to authenticate with a the Prefect API. Defaults to None.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_ENABLE_HTTP2","title":"PREFECT_API_ENABLE_HTTP2 = Setting(bool, default=True) module-attribute","text":"

    If true, enable support for HTTP/2 for communicating with an API.

    If the API does not support HTTP/2, this will have no effect and connections will be made via HTTP/1.1.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLIENT_MAX_RETRIES","title":"PREFECT_CLIENT_MAX_RETRIES = Setting(int, default=5) module-attribute","text":"

    The maximum number of retries to perform on failed HTTP requests.

    Defaults to 5. Set to 0 to disable retries.

    See PREFECT_CLIENT_RETRY_EXTRA_CODES for details on which HTTP status codes are retried.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLIENT_RETRY_JITTER_FACTOR","title":"PREFECT_CLIENT_RETRY_JITTER_FACTOR = Setting(float, default=0.2) module-attribute","text":"

    A value greater than or equal to zero to control the amount of jitter added to retried client requests. Higher values introduce larger amounts of jitter.

    Set to 0 to disable jitter. See clamped_poisson_interval for details on the how jitter can affect retry lengths.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLIENT_RETRY_EXTRA_CODES","title":"PREFECT_CLIENT_RETRY_EXTRA_CODES = Setting(str, default='', value_callback=status_codes_as_integers_in_range) module-attribute","text":"

    A comma-separated list of extra HTTP status codes to retry on. Defaults to an empty string. 429, 502 and 503 are always retried. Please note that not all routes are idempotent and retrying may result in unexpected behavior.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLOUD_API_URL","title":"PREFECT_CLOUD_API_URL = Setting(str, default='https://api.prefect.cloud/api', value_callback=check_for_deprecated_cloud_url) module-attribute","text":"

    API URL for Prefect Cloud. Used for authentication.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLOUD_URL","title":"PREFECT_CLOUD_URL = Setting(str, default=None, deprecated=True, deprecated_start_date='Dec 2022', deprecated_help='Use `PREFECT_CLOUD_API_URL` instead.') module-attribute","text":"","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_UI_URL","title":"PREFECT_UI_URL = Setting(Optional[str], default=None, value_callback=default_ui_url) module-attribute","text":"

    The URL for the UI. By default, this is inferred from the PREFECT_API_URL.

    When using Prefect Cloud, this will include the account and workspace. When using an ephemeral server, this will be None.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLOUD_UI_URL","title":"PREFECT_CLOUD_UI_URL = Setting(str, default=None, value_callback=default_cloud_ui_url) module-attribute","text":"

    The URL for the Cloud UI. By default, this is inferred from the PREFECT_CLOUD_API_URL.

    PREFECT_UI_URL will be workspace specific and will be usable in the open source too.

    In contrast, this value is only valid for Cloud and will not include the workspace.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_REQUEST_TIMEOUT","title":"PREFECT_API_REQUEST_TIMEOUT = Setting(float, default=60.0) module-attribute","text":"

    The default timeout for requests to the API

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN","title":"PREFECT_EXPERIMENTAL_WARN = Setting(bool, default=True) module-attribute","text":"

    If enabled, warn on usage of experimental features.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_PROFILES_PATH","title":"PREFECT_PROFILES_PATH = Setting(Path, default=Path('${PREFECT_HOME}') / 'profiles.toml', value_callback=template_with_settings(PREFECT_HOME)) module-attribute","text":"

    The path to a profiles configuration files.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RESULTS_DEFAULT_SERIALIZER","title":"PREFECT_RESULTS_DEFAULT_SERIALIZER = Setting(str, default='pickle') module-attribute","text":"

    The default serializer to use when not otherwise specified.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RESULTS_PERSIST_BY_DEFAULT","title":"PREFECT_RESULTS_PERSIST_BY_DEFAULT = Setting(bool, default=False) module-attribute","text":"

    The default setting for persisting results when not otherwise specified. If enabled, flow and task results will be persisted unless they opt out.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASKS_REFRESH_CACHE","title":"PREFECT_TASKS_REFRESH_CACHE = Setting(bool, default=False) module-attribute","text":"

    If True, enables a refresh of cached results: re-executing the task will refresh the cached results. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASK_DEFAULT_RETRIES","title":"PREFECT_TASK_DEFAULT_RETRIES = Setting(int, default=0) module-attribute","text":"

    This value sets the default number of retries for all tasks. This value does not overwrite individually set retries values on tasks

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_FLOW_DEFAULT_RETRIES","title":"PREFECT_FLOW_DEFAULT_RETRIES = Setting(int, default=0) module-attribute","text":"

    This value sets the default number of retries for all flows. This value does not overwrite individually set retries values on a flow

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_FLOW_DEFAULT_RETRY_DELAY_SECONDS","title":"PREFECT_FLOW_DEFAULT_RETRY_DELAY_SECONDS = Setting(Union[int, float], default=0) module-attribute","text":"

    This value sets the retry delay seconds for all flows. This value does not overwrite individually set retry delay seconds

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASK_DEFAULT_RETRY_DELAY_SECONDS","title":"PREFECT_TASK_DEFAULT_RETRY_DELAY_SECONDS = Setting(Union[float, int, List[float]], default=0) module-attribute","text":"

    This value sets the default retry delay seconds for all tasks. This value does not overwrite individually set retry delay seconds

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS","title":"PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS = Setting(int, default=30) module-attribute","text":"

    The number of seconds to wait before retrying when a task run cannot secure a concurrency slot from the server.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOCAL_STORAGE_PATH","title":"PREFECT_LOCAL_STORAGE_PATH = Setting(Path, default=Path('${PREFECT_HOME}') / 'storage', value_callback=template_with_settings(PREFECT_HOME)) module-attribute","text":"

    The path to a block storage directory to store things in.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_MEMO_STORE_PATH","title":"PREFECT_MEMO_STORE_PATH = Setting(Path, default=Path('${PREFECT_HOME}') / 'memo_store.toml', value_callback=template_with_settings(PREFECT_HOME)) module-attribute","text":"

    The path to the memo store file.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_MEMOIZE_BLOCK_AUTO_REGISTRATION","title":"PREFECT_MEMOIZE_BLOCK_AUTO_REGISTRATION = Setting(bool, default=True) module-attribute","text":"

    Controls whether or not block auto-registration on start up should be memoized. Setting to False may result in slower server start up times.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_LEVEL","title":"PREFECT_LOGGING_LEVEL = Setting(str, default='INFO', value_callback=debug_mode_log_level) module-attribute","text":"

    The default logging level for Prefect loggers. Defaults to \"INFO\" during normal operation. Is forced to \"DEBUG\" during debug mode.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_INTERNAL_LEVEL","title":"PREFECT_LOGGING_INTERNAL_LEVEL = Setting(str, default='ERROR', value_callback=debug_mode_log_level) module-attribute","text":"

    The default logging level for Prefect's internal machinery loggers. Defaults to \"ERROR\" during normal operation. Is forced to \"DEBUG\" during debug mode.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_SERVER_LEVEL","title":"PREFECT_LOGGING_SERVER_LEVEL = Setting(str, default='WARNING') module-attribute","text":"

    The default logging level for the Prefect API server.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_SETTINGS_PATH","title":"PREFECT_LOGGING_SETTINGS_PATH = Setting(Path, default=Path('${PREFECT_HOME}') / 'logging.yml', value_callback=template_with_settings(PREFECT_HOME)) module-attribute","text":"

    The path to a custom YAML logging configuration file. If no file is found, the default logging.yml is used. Defaults to a logging.yml in the Prefect home directory.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_EXTRA_LOGGERS","title":"PREFECT_LOGGING_EXTRA_LOGGERS = Setting(str, default='', value_callback=get_extra_loggers) module-attribute","text":"

    Additional loggers to attach to Prefect logging at runtime. Values should be comma separated. The handlers attached to the 'prefect' logger will be added to these loggers. Additionally, if the level is not set, it will be set to the same level as the 'prefect' logger.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_LOG_PRINTS","title":"PREFECT_LOGGING_LOG_PRINTS = Setting(bool, default=False) module-attribute","text":"

    If set, print statements in flows and tasks will be redirected to the Prefect logger for the given run. This setting can be overridden by individual tasks and flows.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_ENABLED","title":"PREFECT_LOGGING_TO_API_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Toggles sending logs to the API. If False, logs sent to the API log handler will not be sent to the API.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_BATCH_INTERVAL","title":"PREFECT_LOGGING_TO_API_BATCH_INTERVAL = Setting(float, default=2.0) module-attribute","text":"

    The number of seconds between batched writes of logs to the API.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_BATCH_SIZE","title":"PREFECT_LOGGING_TO_API_BATCH_SIZE = Setting(int, default=4000000) module-attribute","text":"

    The maximum size in bytes for a batch of logs.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_MAX_LOG_SIZE","title":"PREFECT_LOGGING_TO_API_MAX_LOG_SIZE = Setting(int, default=1000000) module-attribute","text":"

    The maximum size in bytes for a single log.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_WHEN_MISSING_FLOW","title":"PREFECT_LOGGING_TO_API_WHEN_MISSING_FLOW = Setting(Literal['warn', 'error', 'ignore'], default='warn') module-attribute","text":"

    Controls the behavior when loggers attempt to send logs to the API handler from outside of a flow.

    All logs sent to the API must be associated with a flow run. The API log handler can only be used outside of a flow by manually providing a flow run identifier. Logs that are not associated with a flow run will not be sent to the API. This setting can be used to determine if a warning or error is displayed when the identifier is missing.

    The following options are available:

    • \"warn\": Log a warning message.
    • \"error\": Raise an error.
    • \"ignore\": Do not log a warning message or raise an error.
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SQLALCHEMY_POOL_SIZE","title":"PREFECT_SQLALCHEMY_POOL_SIZE = Setting(int, default=None) module-attribute","text":"

    Controls connection pool size when using a PostgreSQL database with the Prefect API. If not set, the default SQLAlchemy pool size will be used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SQLALCHEMY_MAX_OVERFLOW","title":"PREFECT_SQLALCHEMY_MAX_OVERFLOW = Setting(int, default=None) module-attribute","text":"

    Controls maximum overflow of the connection pool when using a PostgreSQL database with the Prefect API. If not set, the default SQLAlchemy maximum overflow value will be used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_COLORS","title":"PREFECT_LOGGING_COLORS = Setting(bool, default=True) module-attribute","text":"

    Whether to style console logs with color.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_MARKUP","title":"PREFECT_LOGGING_MARKUP = Setting(bool, default=False) module-attribute","text":"

    Whether to interpret strings wrapped in square brackets as a style. This allows styles to be conveniently added to log messages, e.g. [red]This is a red message.[/red]. However, the downside is, if enabled, strings that contain square brackets may be inaccurately interpreted and lead to incomplete output, e.g. DROP TABLE [dbo].[SomeTable];\" outputs DROP TABLE .[SomeTable];.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD","title":"PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD = Setting(float, default=10.0) module-attribute","text":"

    Threshold time in seconds for logging a warning if task parameter introspection exceeds this duration. Parameter introspection can be a significant performance hit when the parameter is a large collection object, e.g. a large dictionary or DataFrame, and each element needs to be inspected. See prefect.utilities.annotations.quote for more details. Defaults to 10.0. Set to 0 to disable logging the warning.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_AGENT_QUERY_INTERVAL","title":"PREFECT_AGENT_QUERY_INTERVAL = Setting(float, default=15) module-attribute","text":"

    The agent loop interval, in seconds. Agents will check for new runs this often. Defaults to 15.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_AGENT_PREFETCH_SECONDS","title":"PREFECT_AGENT_PREFETCH_SECONDS = Setting(int, default=15) module-attribute","text":"

    Agents will look for scheduled runs this many seconds in the future and attempt to run them. This accounts for any additional infrastructure spin-up time or latency in preparing a flow run. Note flow runs will not start before their scheduled time, even if they are prefetched. Defaults to 15.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_ASYNC_FETCH_STATE_RESULT","title":"PREFECT_ASYNC_FETCH_STATE_RESULT = Setting(bool, default=False) module-attribute","text":"

    Determines whether State.result() fetches results automatically or not. In Prefect 2.6.0, the State.result() method was updated to be async to facilitate automatic retrieval of results from storage which means when writing async code you must await the call. For backwards compatibility, the result is not retrieved by default for async users. You may opt into this per call by passing fetch=True or toggle this setting to change the behavior globally. This setting does not affect users writing synchronous tasks and flows. This setting does not affect retrieval of results when using Future.result().

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_BLOCKS_REGISTER_ON_START","title":"PREFECT_API_BLOCKS_REGISTER_ON_START = Setting(bool, default=True) module-attribute","text":"

    If set, any block types that have been imported will be registered with the backend on application startup. If not set, block types must be manually registered.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_PASSWORD","title":"PREFECT_API_DATABASE_PASSWORD = Setting(str, default=None, is_secret=True) module-attribute","text":"

    Password to template into the PREFECT_API_DATABASE_CONNECTION_URL. This is useful if the password must be provided separately from the connection URL. To use this setting, you must include it in your connection URL.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_CONNECTION_URL","title":"PREFECT_API_DATABASE_CONNECTION_URL = Setting(str, default=None, value_callback=default_database_connection_url, is_secret=True) module-attribute","text":"

    A database connection URL in a SQLAlchemy-compatible format. Prefect currently supports SQLite and Postgres. Note that all Prefect database engines must use an async driver - for SQLite, use sqlite+aiosqlite and for Postgres use postgresql+asyncpg.

    SQLite in-memory databases can be used by providing the url sqlite+aiosqlite:///file::memory:?cache=shared&uri=true&check_same_thread=false, which will allow the database to be accessed by multiple threads. Note that in-memory databases can not be accessed from multiple processes and should only be used for simple tests.

    Defaults to a sqlite database stored in the Prefect home directory.

    If you need to provide password via a different environment variable, you use the PREFECT_API_DATABASE_PASSWORD setting. For example:

    PREFECT_API_DATABASE_PASSWORD='mypassword'\nPREFECT_API_DATABASE_CONNECTION_URL='postgresql+asyncpg://postgres:${PREFECT_API_DATABASE_PASSWORD}@localhost/prefect'\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_ECHO","title":"PREFECT_API_DATABASE_ECHO = Setting(bool, default=False) module-attribute","text":"

    If True, SQLAlchemy will log all SQL issued to the database. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_MIGRATE_ON_START","title":"PREFECT_API_DATABASE_MIGRATE_ON_START = Setting(bool, default=True) module-attribute","text":"

    If True, the database will be upgraded on application creation. If False, the database will need to be upgraded manually.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_TIMEOUT","title":"PREFECT_API_DATABASE_TIMEOUT = Setting(Optional[float], default=10.0) module-attribute","text":"

    A statement timeout, in seconds, applied to all database interactions made by the API. Defaults to 10 seconds.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_CONNECTION_TIMEOUT","title":"PREFECT_API_DATABASE_CONNECTION_TIMEOUT = Setting(Optional[float], default=5) module-attribute","text":"

    A connection timeout, in seconds, applied to database connections. Defaults to 5.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS","title":"PREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS = Setting(float, default=60) module-attribute","text":"

    The scheduler loop interval, in seconds. This determines how often the scheduler will attempt to schedule new flow runs, but has no impact on how quickly either flow runs or task runs are actually executed. Defaults to 60.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_DEPLOYMENT_BATCH_SIZE","title":"PREFECT_API_SERVICES_SCHEDULER_DEPLOYMENT_BATCH_SIZE = Setting(int, default=100) module-attribute","text":"

    The number of deployments the scheduler will attempt to schedule in a single batch. If there are more deployments than the batch size, the scheduler immediately attempts to schedule the next batch; it does not sleep for scheduler_loop_seconds until it has visited every deployment once. Defaults to 100.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_MAX_RUNS","title":"PREFECT_API_SERVICES_SCHEDULER_MAX_RUNS = Setting(int, default=100) module-attribute","text":"

    The scheduler will attempt to schedule up to this many auto-scheduled runs in the future. Note that runs may have fewer than this many scheduled runs, depending on the value of scheduler_max_scheduled_time. Defaults to 100.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_MIN_RUNS","title":"PREFECT_API_SERVICES_SCHEDULER_MIN_RUNS = Setting(int, default=3) module-attribute","text":"

    The scheduler will attempt to schedule at least this many auto-scheduled runs in the future. Note that runs may have more than this many scheduled runs, depending on the value of scheduler_min_scheduled_time. Defaults to 3.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME","title":"PREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME = Setting(timedelta, default=timedelta(days=100)) module-attribute","text":"

    The scheduler will create new runs up to this far in the future. Note that this setting will take precedence over scheduler_max_runs: if a flow runs once a month and scheduler_max_scheduled_time is three months, then only three runs will be scheduled. Defaults to 100 days (8640000 seconds).

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME","title":"PREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME = Setting(timedelta, default=timedelta(hours=1)) module-attribute","text":"

    The scheduler will create new runs at least this far in the future. Note that this setting will take precedence over scheduler_min_runs: if a flow runs every hour and scheduler_min_scheduled_time is three hours, then three runs will be scheduled even if scheduler_min_runs is 1. Defaults to 1 hour (3600 seconds).

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_INSERT_BATCH_SIZE","title":"PREFECT_API_SERVICES_SCHEDULER_INSERT_BATCH_SIZE = Setting(int, default=500) module-attribute","text":"

    The number of flow runs the scheduler will attempt to insert in one batch across all deployments. If the number of flow runs to schedule exceeds this amount, the runs will be inserted in batches of this size. Defaults to 500.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_LATE_RUNS_LOOP_SECONDS","title":"PREFECT_API_SERVICES_LATE_RUNS_LOOP_SECONDS = Setting(float, default=5) module-attribute","text":"

    The late runs service will look for runs to mark as late this often. Defaults to 5.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_LATE_RUNS_AFTER_SECONDS","title":"PREFECT_API_SERVICES_LATE_RUNS_AFTER_SECONDS = Setting(timedelta, default=timedelta(seconds=5)) module-attribute","text":"

    The late runs service will mark runs as late after they have exceeded their scheduled start time by this many seconds. Defaults to 5 seconds.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_LOOP_SECONDS","title":"PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_LOOP_SECONDS = Setting(float, default=5) module-attribute","text":"

    The pause expiration service will look for runs to mark as failed this often. Defaults to 5.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_CANCELLATION_CLEANUP_LOOP_SECONDS","title":"PREFECT_API_SERVICES_CANCELLATION_CLEANUP_LOOP_SECONDS = Setting(float, default=20) module-attribute","text":"

    The cancellation cleanup service will look non-terminal tasks and subflows this often. Defaults to 20.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DEFAULT_LIMIT","title":"PREFECT_API_DEFAULT_LIMIT = Setting(int, default=200) module-attribute","text":"

    The default limit applied to queries that can return multiple objects, such as POST /flow_runs/filter.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SERVER_API_HOST","title":"PREFECT_SERVER_API_HOST = Setting(str, default='127.0.0.1') module-attribute","text":"

    The API's host address (defaults to 127.0.0.1).

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SERVER_API_PORT","title":"PREFECT_SERVER_API_PORT = Setting(int, default=4200) module-attribute","text":"

    The API's port address (defaults to 4200).

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SERVER_API_KEEPALIVE_TIMEOUT","title":"PREFECT_SERVER_API_KEEPALIVE_TIMEOUT = Setting(int, default=5) module-attribute","text":"

    The API's keep alive timeout (defaults to 5). Refer to https://www.uvicorn.org/settings/#timeouts for details.

    When the API is hosted behind a load balancer, you may want to set this to a value greater than the load balancer's idle timeout.

    Note this setting only applies when calling prefect server start; if hosting the API with another tool you will need to configure this there instead.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_UI_ENABLED","title":"PREFECT_UI_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to serve the Prefect UI.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_UI_API_URL","title":"PREFECT_UI_API_URL = Setting(str, default=None, value_callback=default_ui_api_url) module-attribute","text":"

    The connection url for communication from the UI to the API. Defaults to PREFECT_API_URL if set. Otherwise, the default URL is generated from PREFECT_SERVER_API_HOST and PREFECT_SERVER_API_PORT. If providing a custom value, the aforementioned settings may be templated into the given string.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SERVER_ANALYTICS_ENABLED","title":"PREFECT_SERVER_ANALYTICS_ENABLED = Setting(bool, default=True) module-attribute","text":"

    When enabled, Prefect sends anonymous data (e.g. count of flow runs, package version) on server startup to help us improve our product.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_ENABLED","title":"PREFECT_API_SERVICES_SCHEDULER_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the scheduling service in the server application. If disabled, you will need to run this service separately to schedule runs for deployments.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_LATE_RUNS_ENABLED","title":"PREFECT_API_SERVICES_LATE_RUNS_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the late runs service in the server application. If disabled, you will need to run this service separately to have runs past their scheduled start time marked as late.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_FLOW_RUN_NOTIFICATIONS_ENABLED","title":"PREFECT_API_SERVICES_FLOW_RUN_NOTIFICATIONS_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the flow run notifications service in the server application. If disabled, you will need to run this service separately to send flow run notifications.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_ENABLED","title":"PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the paused flow run expiration service in the server application. If disabled, paused flows that have timed out will remain in a Paused state until a resume attempt.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH","title":"PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH = Setting(int, default=2000) module-attribute","text":"

    The maximum number of characters allowed for a task run cache key. This setting cannot be changed client-side, it must be set on the server.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_CANCELLATION_CLEANUP_ENABLED","title":"PREFECT_API_SERVICES_CANCELLATION_CLEANUP_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the cancellation cleanup service in the server application. If disabled, task runs and subflow runs belonging to cancelled flows may remain in non-terminal states.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_MAX_FLOW_RUN_GRAPH_NODES","title":"PREFECT_API_MAX_FLOW_RUN_GRAPH_NODES = Setting(int, default=10000) module-attribute","text":"

    The maximum size of a flow run graph on the v2 API

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_EVENTS_CLIENT","title":"PREFECT_EXPERIMENTAL_ENABLE_EVENTS_CLIENT = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental Prefect work pools.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_EVENTS_CLIENT","title":"PREFECT_EXPERIMENTAL_WARN_EVENTS_CLIENT = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect work pools are used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_WORK_POOLS","title":"PREFECT_EXPERIMENTAL_ENABLE_WORK_POOLS = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental Prefect work pools.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_WORK_POOLS","title":"PREFECT_EXPERIMENTAL_WARN_WORK_POOLS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect work pools are used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_WORKERS","title":"PREFECT_EXPERIMENTAL_ENABLE_WORKERS = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental Prefect workers.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_WORKERS","title":"PREFECT_EXPERIMENTAL_WARN_WORKERS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect workers are used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_VISUALIZE","title":"PREFECT_EXPERIMENTAL_WARN_VISUALIZE = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect visualize is used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_ENHANCED_CANCELLATION","title":"PREFECT_EXPERIMENTAL_ENABLE_ENHANCED_CANCELLATION = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental enhanced flow run cancellation.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_ENHANCED_CANCELLATION","title":"PREFECT_EXPERIMENTAL_WARN_ENHANCED_CANCELLATION = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental enhanced flow run cancellation is used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_DEPLOYMENT_STATUS","title":"PREFECT_EXPERIMENTAL_ENABLE_DEPLOYMENT_STATUS = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable deployment status in the UI

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_DEPLOYMENT_STATUS","title":"PREFECT_EXPERIMENTAL_WARN_DEPLOYMENT_STATUS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when deployment status is used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_FLOW_RUN_INPUT","title":"PREFECT_EXPERIMENTAL_FLOW_RUN_INPUT = Setting(bool, default=False) module-attribute","text":"

    Whether or not to enable flow run input.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_FLOW_RUN_INPUT","title":"PREFECT_EXPERIMENTAL_WARN_FLOW_RUN_INPUT = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable flow run input.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_PROCESS_LIMIT","title":"PREFECT_RUNNER_PROCESS_LIMIT = Setting(int, default=5) module-attribute","text":"

    Maximum number of processes a runner will execute in parallel.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_POLL_FREQUENCY","title":"PREFECT_RUNNER_POLL_FREQUENCY = Setting(int, default=10) module-attribute","text":"

    Number of seconds a runner should wait between queries for scheduled work.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_SERVER_MISSED_POLLS_TOLERANCE","title":"PREFECT_RUNNER_SERVER_MISSED_POLLS_TOLERANCE = Setting(int, default=2) module-attribute","text":"

    Number of missed polls before a runner is considered unhealthy by its webserver.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_SERVER_HOST","title":"PREFECT_RUNNER_SERVER_HOST = Setting(str, default='0.0.0.0') module-attribute","text":"

    The host address the runner's webserver should bind to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_SERVER_PORT","title":"PREFECT_RUNNER_SERVER_PORT = Setting(int, default=8080) module-attribute","text":"

    The port the runner's webserver should bind to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_SERVER_LOG_LEVEL","title":"PREFECT_RUNNER_SERVER_LOG_LEVEL = Setting(str, default='error') module-attribute","text":"

    The log level of the runner's webserver.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_HEARTBEAT_SECONDS","title":"PREFECT_WORKER_HEARTBEAT_SECONDS = Setting(float, default=30) module-attribute","text":"

    Number of seconds a worker should wait between sending a heartbeat.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_QUERY_SECONDS","title":"PREFECT_WORKER_QUERY_SECONDS = Setting(float, default=10) module-attribute","text":"

    Number of seconds a worker should wait between queries for scheduled flow runs.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_PREFETCH_SECONDS","title":"PREFECT_WORKER_PREFETCH_SECONDS = Setting(float, default=10) module-attribute","text":"

    The number of seconds into the future a worker should query for scheduled flow runs. Can be used to compensate for infrastructure start up time for a worker.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_WEBSERVER_HOST","title":"PREFECT_WORKER_WEBSERVER_HOST = Setting(str, default='0.0.0.0') module-attribute","text":"

    The host address the worker's webserver should bind to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_WEBSERVER_PORT","title":"PREFECT_WORKER_WEBSERVER_PORT = Setting(int, default=8080) module-attribute","text":"

    The port the worker's webserver should bind to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_EXTRA_RUNNER_ENDPOINTS","title":"PREFECT_EXPERIMENTAL_ENABLE_EXTRA_RUNNER_ENDPOINTS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to enable experimental worker webserver endpoints.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_ARTIFACTS","title":"PREFECT_EXPERIMENTAL_ENABLE_ARTIFACTS = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental Prefect artifacts.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_ARTIFACTS","title":"PREFECT_EXPERIMENTAL_WARN_ARTIFACTS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect artifacts are used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_WORKSPACE_DASHBOARD","title":"PREFECT_EXPERIMENTAL_ENABLE_WORKSPACE_DASHBOARD = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable the experimental workspace dashboard.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_WORKSPACE_DASHBOARD","title":"PREFECT_EXPERIMENTAL_WARN_WORKSPACE_DASHBOARD = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when the experimental workspace dashboard is enabled.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_DEFAULT_RESULT_STORAGE_BLOCK","title":"PREFECT_DEFAULT_RESULT_STORAGE_BLOCK = Setting(str, default=None) module-attribute","text":"

    The block-type/block-document slug of a block to use as the default result storage.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_DEFAULT_WORK_POOL_NAME","title":"PREFECT_DEFAULT_WORK_POOL_NAME = Setting(str, default=None) module-attribute","text":"

    The default work pool to deploy to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE","title":"PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE = Setting(str, default=None) module-attribute","text":"

    The default Docker namespace to use when building images.

    Can be either an organization/username or a registry URL with an organization/username.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Setting","title":"Setting","text":"

    Bases: Generic[T]

    Setting definition type.

    Source code in prefect/settings.py
    class Setting(Generic[T]):\n    \"\"\"\n    Setting definition type.\n    \"\"\"\n\n    def __init__(\n        self,\n        type: Type[T],\n        *,\n        deprecated: bool = False,\n        deprecated_start_date: Optional[str] = None,\n        deprecated_end_date: Optional[str] = None,\n        deprecated_help: str = \"\",\n        deprecated_when_message: str = \"\",\n        deprecated_when: Optional[Callable[[Any], bool]] = None,\n        deprecated_renamed_to: Optional[\"Setting[T]\"] = None,\n        value_callback: Optional[Callable[[\"Settings\", T], T]] = None,\n        is_secret: bool = False,\n        **kwargs: Any,\n    ) -> None:\n        self.field: fields.FieldInfo = Field(**kwargs)\n        self.type = type\n        self.value_callback = value_callback\n        self._name = None\n        self.is_secret = is_secret\n        self.deprecated = deprecated\n        self.deprecated_start_date = deprecated_start_date\n        self.deprecated_end_date = deprecated_end_date\n        self.deprecated_help = deprecated_help\n        self.deprecated_when = deprecated_when or (lambda _: True)\n        self.deprecated_when_message = deprecated_when_message\n        self.deprecated_renamed_to = deprecated_renamed_to\n        self.deprecated_renamed_from = None\n        self.__doc__ = self.field.description\n\n        # Validate the deprecation settings, will throw an error at setting definition\n        # time if the developer has not configured it correctly\n        if deprecated:\n            generate_deprecation_message(\n                name=\"...\",  # setting names not populated until after init\n                start_date=self.deprecated_start_date,\n                end_date=self.deprecated_end_date,\n                help=self.deprecated_help,\n                when=self.deprecated_when_message,\n            )\n\n        if deprecated_renamed_to is not None:\n            # Track the deprecation both ways\n            deprecated_renamed_to.deprecated_renamed_from = self\n\n    def value(self, bypass_callback: bool = False) -> T:\n        \"\"\"\n        Get the current value of a setting.\n\n        Example:\n        ```python\n        from prefect.settings import PREFECT_API_URL\n        PREFECT_API_URL.value()\n        ```\n        \"\"\"\n        return self.value_from(get_current_settings(), bypass_callback=bypass_callback)\n\n    def value_from(self, settings: \"Settings\", bypass_callback: bool = False) -> T:\n        \"\"\"\n        Get the value of a setting from a settings object\n\n        Example:\n        ```python\n        from prefect.settings import get_default_settings\n        PREFECT_API_URL.value_from(get_default_settings())\n        ```\n        \"\"\"\n        value = settings.value_of(self, bypass_callback=bypass_callback)\n\n        if not bypass_callback and self.deprecated and self.deprecated_when(value):\n            # Check if this setting is deprecated and someone is accessing the value\n            # via the old name\n            warnings.warn(self.deprecated_message, DeprecationWarning, stacklevel=3)\n\n            # If the the value is empty, return the new setting's value for compat\n            if value is None and self.deprecated_renamed_to is not None:\n                return self.deprecated_renamed_to.value_from(settings)\n\n        if not bypass_callback and self.deprecated_renamed_from is not None:\n            # Check if this setting is a rename of a deprecated setting and the\n            # deprecated setting is set and should be used for compatibility\n            deprecated_value = self.deprecated_renamed_from.value_from(\n                settings, bypass_callback=True\n            )\n            if deprecated_value is not None:\n                warnings.warn(\n                    (\n                        f\"{self.deprecated_renamed_from.deprecated_message} Because\"\n                        f\" {self.deprecated_renamed_from.name!r} is set it will be used\"\n                        f\" instead of {self.name!r} for backwards compatibility.\"\n                    ),\n                    DeprecationWarning,\n                    stacklevel=3,\n                )\n            return deprecated_value or value\n\n        return value\n\n    @property\n    def name(self):\n        if self._name:\n            return self._name\n\n        # Lookup the name on first access\n        for name, val in tuple(globals().items()):\n            if val == self:\n                self._name = name\n                return name\n\n        raise ValueError(\"Setting not found in `prefect.settings` module.\")\n\n    @name.setter\n    def name(self, value: str):\n        self._name = value\n\n    @property\n    def deprecated_message(self):\n        return generate_deprecation_message(\n            name=f\"Setting {self.name!r}\",\n            start_date=self.deprecated_start_date,\n            end_date=self.deprecated_end_date,\n            help=self.deprecated_help,\n            when=self.deprecated_when_message,\n        )\n\n    def __repr__(self) -> str:\n        return f\"<{self.name}: {self.type.__name__}>\"\n\n    def __bool__(self) -> bool:\n        \"\"\"\n        Returns a truthy check of the current value.\n        \"\"\"\n        return bool(self.value())\n\n    def __eq__(self, __o: object) -> bool:\n        return __o.__eq__(self.value())\n\n    def __hash__(self) -> int:\n        return hash((type(self), self.name))\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Setting.value","title":"value","text":"

    Get the current value of a setting.

    Example:

    from prefect.settings import PREFECT_API_URL\nPREFECT_API_URL.value()\n

    Source code in prefect/settings.py
    def value(self, bypass_callback: bool = False) -> T:\n    \"\"\"\n    Get the current value of a setting.\n\n    Example:\n    ```python\n    from prefect.settings import PREFECT_API_URL\n    PREFECT_API_URL.value()\n    ```\n    \"\"\"\n    return self.value_from(get_current_settings(), bypass_callback=bypass_callback)\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Setting.value_from","title":"value_from","text":"

    Get the value of a setting from a settings object

    Example:

    from prefect.settings import get_default_settings\nPREFECT_API_URL.value_from(get_default_settings())\n

    Source code in prefect/settings.py
    def value_from(self, settings: \"Settings\", bypass_callback: bool = False) -> T:\n    \"\"\"\n    Get the value of a setting from a settings object\n\n    Example:\n    ```python\n    from prefect.settings import get_default_settings\n    PREFECT_API_URL.value_from(get_default_settings())\n    ```\n    \"\"\"\n    value = settings.value_of(self, bypass_callback=bypass_callback)\n\n    if not bypass_callback and self.deprecated and self.deprecated_when(value):\n        # Check if this setting is deprecated and someone is accessing the value\n        # via the old name\n        warnings.warn(self.deprecated_message, DeprecationWarning, stacklevel=3)\n\n        # If the the value is empty, return the new setting's value for compat\n        if value is None and self.deprecated_renamed_to is not None:\n            return self.deprecated_renamed_to.value_from(settings)\n\n    if not bypass_callback and self.deprecated_renamed_from is not None:\n        # Check if this setting is a rename of a deprecated setting and the\n        # deprecated setting is set and should be used for compatibility\n        deprecated_value = self.deprecated_renamed_from.value_from(\n            settings, bypass_callback=True\n        )\n        if deprecated_value is not None:\n            warnings.warn(\n                (\n                    f\"{self.deprecated_renamed_from.deprecated_message} Because\"\n                    f\" {self.deprecated_renamed_from.name!r} is set it will be used\"\n                    f\" instead of {self.name!r} for backwards compatibility.\"\n                ),\n                DeprecationWarning,\n                stacklevel=3,\n            )\n        return deprecated_value or value\n\n    return value\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings","title":"Settings","text":"

    Bases: SettingsFieldsMixin

    Contains validated Prefect settings.

    Settings should be accessed using the relevant Setting object. For example:

    from prefect.settings import PREFECT_HOME\nPREFECT_HOME.value()\n

    Accessing a setting attribute directly will ignore any value_callback mutations. This is not recommended:

    from prefect.settings import Settings\nSettings().PREFECT_PROFILES_PATH  # PosixPath('${PREFECT_HOME}/profiles.toml')\n

    Source code in prefect/settings.py
    @add_cloudpickle_reduction\nclass Settings(SettingsFieldsMixin):\n    \"\"\"\n    Contains validated Prefect settings.\n\n    Settings should be accessed using the relevant `Setting` object. For example:\n    ```python\n    from prefect.settings import PREFECT_HOME\n    PREFECT_HOME.value()\n    ```\n\n    Accessing a setting attribute directly will ignore any `value_callback` mutations.\n    This is not recommended:\n    ```python\n    from prefect.settings import Settings\n    Settings().PREFECT_PROFILES_PATH  # PosixPath('${PREFECT_HOME}/profiles.toml')\n    ```\n    \"\"\"\n\n    def value_of(self, setting: Setting[T], bypass_callback: bool = False) -> T:\n        \"\"\"\n        Retrieve a setting's value.\n        \"\"\"\n        value = getattr(self, setting.name)\n        if setting.value_callback and not bypass_callback:\n            value = setting.value_callback(self, value)\n        return value\n\n    @validator(PREFECT_LOGGING_LEVEL.name, PREFECT_LOGGING_SERVER_LEVEL.name)\n    def check_valid_log_level(cls, value):\n        if isinstance(value, str):\n            value = value.upper()\n        logging._checkLevel(value)\n        return value\n\n    @root_validator\n    def post_root_validators(cls, values):\n        \"\"\"\n        Add root validation functions for settings here.\n        \"\"\"\n        # TODO: We could probably register these dynamically but this is the simpler\n        #       approach for now. We can explore more interesting validation features\n        #       in the future.\n        values = max_log_size_smaller_than_batch_size(values)\n        values = warn_on_database_password_value_without_usage(values)\n        values = warn_on_misconfigured_api_url(values)\n        return values\n\n    def copy_with_update(\n        self,\n        updates: Mapping[Setting, Any] = None,\n        set_defaults: Mapping[Setting, Any] = None,\n        restore_defaults: Iterable[Setting] = None,\n    ) -> \"Settings\":\n        \"\"\"\n        Create a new `Settings` object with validation.\n\n        Arguments:\n            updates: A mapping of settings to new values. Existing values for the\n                given settings will be overridden.\n            set_defaults: A mapping of settings to new default values. Existing values for\n                the given settings will only be overridden if they were not set.\n            restore_defaults: An iterable of settings to restore to their default values.\n\n        Returns:\n            A new `Settings` object.\n        \"\"\"\n        updates = updates or {}\n        set_defaults = set_defaults or {}\n        restore_defaults = restore_defaults or set()\n        restore_defaults_names = {setting.name for setting in restore_defaults}\n\n        return self.__class__(\n            **{\n                **{setting.name: value for setting, value in set_defaults.items()},\n                **self.dict(exclude_unset=True, exclude=restore_defaults_names),\n                **{setting.name: value for setting, value in updates.items()},\n            }\n        )\n\n    def with_obfuscated_secrets(self):\n        \"\"\"\n        Returns a copy of this settings object with secret setting values obfuscated.\n        \"\"\"\n        settings = self.copy(\n            update={\n                setting.name: obfuscate(self.value_of(setting))\n                for setting in SETTING_VARIABLES.values()\n                if setting.is_secret\n                # Exclude deprecated settings with null values to avoid warnings\n                and not (setting.deprecated and self.value_of(setting) is None)\n            }\n        )\n        # Ensure that settings that have not been marked as \"set\" before are still so\n        # after we have updated their value above\n        settings.__fields_set__.intersection_update(self.__fields_set__)\n        return settings\n\n    def to_environment_variables(\n        self, include: Iterable[Setting] = None, exclude_unset: bool = False\n    ) -> Dict[str, str]:\n        \"\"\"\n        Convert the settings object to environment variables.\n\n        Note that setting values will not be run through their `value_callback` allowing\n        dynamic resolution to occur when loaded from the returned environment.\n\n        Args:\n            include_keys: An iterable of settings to include in the return value.\n                If not set, all settings are used.\n            exclude_unset: Only include settings that have been set (i.e. the value is\n                not from the default). If set, unset keys will be dropped even if they\n                are set in `include_keys`.\n\n        Returns:\n            A dictionary of settings with values cast to strings\n        \"\"\"\n        include = set(include or SETTING_VARIABLES.values())\n\n        if exclude_unset:\n            set_keys = {\n                # Collect all of the \"set\" keys and cast to `Setting` objects\n                SETTING_VARIABLES[key]\n                for key in self.dict(exclude_unset=True)\n            }\n            include.intersection_update(set_keys)\n\n        # Validate the types of items in `include` to prevent exclusion bugs\n        for key in include:\n            if not isinstance(key, Setting):\n                raise TypeError(\n                    \"Invalid type {type(key).__name__!r} for key in `include`.\"\n                )\n\n        env = {\n            # Use `getattr` instead of `value_of` to avoid value callback resolution\n            key: getattr(self, key)\n            for key, setting in SETTING_VARIABLES.items()\n            if setting in include\n        }\n\n        # Cast to strings and drop null values\n        return {key: str(value) for key, value in env.items() if value is not None}\n\n    class Config:\n        frozen = True\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings.value_of","title":"value_of","text":"

    Retrieve a setting's value.

    Source code in prefect/settings.py
    def value_of(self, setting: Setting[T], bypass_callback: bool = False) -> T:\n    \"\"\"\n    Retrieve a setting's value.\n    \"\"\"\n    value = getattr(self, setting.name)\n    if setting.value_callback and not bypass_callback:\n        value = setting.value_callback(self, value)\n    return value\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings.post_root_validators","title":"post_root_validators","text":"

    Add root validation functions for settings here.

    Source code in prefect/settings.py
    @root_validator\ndef post_root_validators(cls, values):\n    \"\"\"\n    Add root validation functions for settings here.\n    \"\"\"\n    # TODO: We could probably register these dynamically but this is the simpler\n    #       approach for now. We can explore more interesting validation features\n    #       in the future.\n    values = max_log_size_smaller_than_batch_size(values)\n    values = warn_on_database_password_value_without_usage(values)\n    values = warn_on_misconfigured_api_url(values)\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings.with_obfuscated_secrets","title":"with_obfuscated_secrets","text":"

    Returns a copy of this settings object with secret setting values obfuscated.

    Source code in prefect/settings.py
    def with_obfuscated_secrets(self):\n    \"\"\"\n    Returns a copy of this settings object with secret setting values obfuscated.\n    \"\"\"\n    settings = self.copy(\n        update={\n            setting.name: obfuscate(self.value_of(setting))\n            for setting in SETTING_VARIABLES.values()\n            if setting.is_secret\n            # Exclude deprecated settings with null values to avoid warnings\n            and not (setting.deprecated and self.value_of(setting) is None)\n        }\n    )\n    # Ensure that settings that have not been marked as \"set\" before are still so\n    # after we have updated their value above\n    settings.__fields_set__.intersection_update(self.__fields_set__)\n    return settings\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings.to_environment_variables","title":"to_environment_variables","text":"

    Convert the settings object to environment variables.

    Note that setting values will not be run through their value_callback allowing dynamic resolution to occur when loaded from the returned environment.

    Parameters:

    Name Type Description Default include_keys

    An iterable of settings to include in the return value. If not set, all settings are used.

    required exclude_unset bool

    Only include settings that have been set (i.e. the value is not from the default). If set, unset keys will be dropped even if they are set in include_keys.

    False

    Returns:

    Type Description Dict[str, str]

    A dictionary of settings with values cast to strings

    Source code in prefect/settings.py
    def to_environment_variables(\n    self, include: Iterable[Setting] = None, exclude_unset: bool = False\n) -> Dict[str, str]:\n    \"\"\"\n    Convert the settings object to environment variables.\n\n    Note that setting values will not be run through their `value_callback` allowing\n    dynamic resolution to occur when loaded from the returned environment.\n\n    Args:\n        include_keys: An iterable of settings to include in the return value.\n            If not set, all settings are used.\n        exclude_unset: Only include settings that have been set (i.e. the value is\n            not from the default). If set, unset keys will be dropped even if they\n            are set in `include_keys`.\n\n    Returns:\n        A dictionary of settings with values cast to strings\n    \"\"\"\n    include = set(include or SETTING_VARIABLES.values())\n\n    if exclude_unset:\n        set_keys = {\n            # Collect all of the \"set\" keys and cast to `Setting` objects\n            SETTING_VARIABLES[key]\n            for key in self.dict(exclude_unset=True)\n        }\n        include.intersection_update(set_keys)\n\n    # Validate the types of items in `include` to prevent exclusion bugs\n    for key in include:\n        if not isinstance(key, Setting):\n            raise TypeError(\n                \"Invalid type {type(key).__name__!r} for key in `include`.\"\n            )\n\n    env = {\n        # Use `getattr` instead of `value_of` to avoid value callback resolution\n        key: getattr(self, key)\n        for key, setting in SETTING_VARIABLES.items()\n        if setting in include\n    }\n\n    # Cast to strings and drop null values\n    return {key: str(value) for key, value in env.items() if value is not None}\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Profile","title":"Profile","text":"

    Bases: BaseModel

    A user profile containing settings.

    Source code in prefect/settings.py
    class Profile(BaseModel):\n    \"\"\"\n    A user profile containing settings.\n    \"\"\"\n\n    name: str\n    settings: Dict[Setting, Any] = Field(default_factory=dict)\n    source: Optional[Path]\n\n    @validator(\"settings\", pre=True)\n    def map_names_to_settings(cls, value):\n        if value is None:\n            return value\n\n        # Cast string setting names to variables\n        validated = {}\n        for setting, val in value.items():\n            if isinstance(setting, str) and setting in SETTING_VARIABLES:\n                validated[SETTING_VARIABLES[setting]] = val\n            elif isinstance(setting, Setting):\n                validated[setting] = val\n            else:\n                raise ValueError(f\"Unknown setting {setting!r}.\")\n\n        return validated\n\n    def validate_settings(self) -> None:\n        \"\"\"\n        Validate the settings contained in this profile.\n\n        Raises:\n            pydantic.ValidationError: When settings do not have valid values.\n        \"\"\"\n        # Create a new `Settings` instance with the settings from this profile relying\n        # on Pydantic validation to raise an error.\n        # We do not return the `Settings` object because this is not the recommended\n        # path for constructing settings with a profile. See `use_profile` instead.\n        Settings(**{setting.name: value for setting, value in self.settings.items()})\n\n    def convert_deprecated_renamed_settings(self) -> List[Tuple[Setting, Setting]]:\n        \"\"\"\n        Update settings in place to replace deprecated settings with new settings when\n        renamed.\n\n        Returns a list of tuples with the old and new setting.\n        \"\"\"\n        changed = []\n        for setting in tuple(self.settings):\n            if (\n                setting.deprecated\n                and setting.deprecated_renamed_to\n                and setting.deprecated_renamed_to not in self.settings\n            ):\n                self.settings[setting.deprecated_renamed_to] = self.settings.pop(\n                    setting\n                )\n                changed.append((setting, setting.deprecated_renamed_to))\n        return changed\n\n    class Config:\n        arbitrary_types_allowed = True\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Profile.validate_settings","title":"validate_settings","text":"

    Validate the settings contained in this profile.

    Raises:

    Type Description ValidationError

    When settings do not have valid values.

    Source code in prefect/settings.py
    def validate_settings(self) -> None:\n    \"\"\"\n    Validate the settings contained in this profile.\n\n    Raises:\n        pydantic.ValidationError: When settings do not have valid values.\n    \"\"\"\n    # Create a new `Settings` instance with the settings from this profile relying\n    # on Pydantic validation to raise an error.\n    # We do not return the `Settings` object because this is not the recommended\n    # path for constructing settings with a profile. See `use_profile` instead.\n    Settings(**{setting.name: value for setting, value in self.settings.items()})\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Profile.convert_deprecated_renamed_settings","title":"convert_deprecated_renamed_settings","text":"

    Update settings in place to replace deprecated settings with new settings when renamed.

    Returns a list of tuples with the old and new setting.

    Source code in prefect/settings.py
    def convert_deprecated_renamed_settings(self) -> List[Tuple[Setting, Setting]]:\n    \"\"\"\n    Update settings in place to replace deprecated settings with new settings when\n    renamed.\n\n    Returns a list of tuples with the old and new setting.\n    \"\"\"\n    changed = []\n    for setting in tuple(self.settings):\n        if (\n            setting.deprecated\n            and setting.deprecated_renamed_to\n            and setting.deprecated_renamed_to not in self.settings\n        ):\n            self.settings[setting.deprecated_renamed_to] = self.settings.pop(\n                setting\n            )\n            changed.append((setting, setting.deprecated_renamed_to))\n    return changed\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection","title":"ProfilesCollection","text":"

    \" A utility class for working with a collection of profiles.

    Profiles in the collection must have unique names.

    The collection may store the name of the active profile.

    Source code in prefect/settings.py
    class ProfilesCollection:\n    \"\"\" \"\n    A utility class for working with a collection of profiles.\n\n    Profiles in the collection must have unique names.\n\n    The collection may store the name of the active profile.\n    \"\"\"\n\n    def __init__(\n        self, profiles: Iterable[Profile], active: Optional[str] = None\n    ) -> None:\n        self.profiles_by_name = {profile.name: profile for profile in profiles}\n        self.active_name = active\n\n    @property\n    def names(self) -> Set[str]:\n        \"\"\"\n        Return a set of profile names in this collection.\n        \"\"\"\n        return set(self.profiles_by_name.keys())\n\n    @property\n    def active_profile(self) -> Optional[Profile]:\n        \"\"\"\n        Retrieve the active profile in this collection.\n        \"\"\"\n        if self.active_name is None:\n            return None\n        return self[self.active_name]\n\n    def set_active(self, name: Optional[str], check: bool = True):\n        \"\"\"\n        Set the active profile name in the collection.\n\n        A null value may be passed to indicate that this collection does not determine\n        the active profile.\n        \"\"\"\n        if check and name is not None and name not in self.names:\n            raise ValueError(f\"Unknown profile name {name!r}.\")\n        self.active_name = name\n\n    def update_profile(\n        self, name: str, settings: Mapping[Union[Dict, str], Any], source: Path = None\n    ) -> Profile:\n        \"\"\"\n        Add a profile to the collection or update the existing on if the name is already\n        present in this collection.\n\n        If updating an existing profile, the settings will be merged. Settings can\n        be dropped from the existing profile by setting them to `None` in the new\n        profile.\n\n        Returns the new profile object.\n        \"\"\"\n        existing = self.profiles_by_name.get(name)\n\n        # Convert the input to a `Profile` to cast settings to the correct type\n        profile = Profile(name=name, settings=settings, source=source)\n\n        if existing:\n            new_settings = {**existing.settings, **profile.settings}\n\n            # Drop null keys to restore to default\n            for key, value in tuple(new_settings.items()):\n                if value is None:\n                    new_settings.pop(key)\n\n            new_profile = Profile(\n                name=profile.name,\n                settings=new_settings,\n                source=source or profile.source,\n            )\n        else:\n            new_profile = profile\n\n        self.profiles_by_name[new_profile.name] = new_profile\n\n        return new_profile\n\n    def add_profile(self, profile: Profile) -> None:\n        \"\"\"\n        Add a profile to the collection.\n\n        If the profile name already exists, an exception will be raised.\n        \"\"\"\n        if profile.name in self.profiles_by_name:\n            raise ValueError(\n                f\"Profile name {profile.name!r} already exists in collection.\"\n            )\n\n        self.profiles_by_name[profile.name] = profile\n\n    def remove_profile(self, name: str) -> None:\n        \"\"\"\n        Remove a profile from the collection.\n        \"\"\"\n        self.profiles_by_name.pop(name)\n\n    def without_profile_source(self, path: Optional[Path]) -> \"ProfilesCollection\":\n        \"\"\"\n        Remove profiles that were loaded from a given path.\n\n        Returns a new collection.\n        \"\"\"\n        return ProfilesCollection(\n            [\n                profile\n                for profile in self.profiles_by_name.values()\n                if profile.source != path\n            ],\n            active=self.active_name,\n        )\n\n    def to_dict(self):\n        \"\"\"\n        Convert to a dictionary suitable for writing to disk.\n        \"\"\"\n        return {\n            \"active\": self.active_name,\n            \"profiles\": {\n                profile.name: {\n                    setting.name: value for setting, value in profile.settings.items()\n                }\n                for profile in self.profiles_by_name.values()\n            },\n        }\n\n    def __getitem__(self, name: str) -> Profile:\n        return self.profiles_by_name[name]\n\n    def __iter__(self):\n        return self.profiles_by_name.__iter__()\n\n    def items(self):\n        return self.profiles_by_name.items()\n\n    def __eq__(self, __o: object) -> bool:\n        if not isinstance(__o, ProfilesCollection):\n            return False\n\n        return (\n            self.profiles_by_name == __o.profiles_by_name\n            and self.active_name == __o.active_name\n        )\n\n    def __repr__(self) -> str:\n        return (\n            f\"ProfilesCollection(profiles={list(self.profiles_by_name.values())!r},\"\n            f\" active={self.active_name!r})>\"\n        )\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.names","title":"names: Set[str] property","text":"

    Return a set of profile names in this collection.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.active_profile","title":"active_profile: Optional[Profile] property","text":"

    Retrieve the active profile in this collection.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.set_active","title":"set_active","text":"

    Set the active profile name in the collection.

    A null value may be passed to indicate that this collection does not determine the active profile.

    Source code in prefect/settings.py
    def set_active(self, name: Optional[str], check: bool = True):\n    \"\"\"\n    Set the active profile name in the collection.\n\n    A null value may be passed to indicate that this collection does not determine\n    the active profile.\n    \"\"\"\n    if check and name is not None and name not in self.names:\n        raise ValueError(f\"Unknown profile name {name!r}.\")\n    self.active_name = name\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.update_profile","title":"update_profile","text":"

    Add a profile to the collection or update the existing on if the name is already present in this collection.

    If updating an existing profile, the settings will be merged. Settings can be dropped from the existing profile by setting them to None in the new profile.

    Returns the new profile object.

    Source code in prefect/settings.py
    def update_profile(\n    self, name: str, settings: Mapping[Union[Dict, str], Any], source: Path = None\n) -> Profile:\n    \"\"\"\n    Add a profile to the collection or update the existing on if the name is already\n    present in this collection.\n\n    If updating an existing profile, the settings will be merged. Settings can\n    be dropped from the existing profile by setting them to `None` in the new\n    profile.\n\n    Returns the new profile object.\n    \"\"\"\n    existing = self.profiles_by_name.get(name)\n\n    # Convert the input to a `Profile` to cast settings to the correct type\n    profile = Profile(name=name, settings=settings, source=source)\n\n    if existing:\n        new_settings = {**existing.settings, **profile.settings}\n\n        # Drop null keys to restore to default\n        for key, value in tuple(new_settings.items()):\n            if value is None:\n                new_settings.pop(key)\n\n        new_profile = Profile(\n            name=profile.name,\n            settings=new_settings,\n            source=source or profile.source,\n        )\n    else:\n        new_profile = profile\n\n    self.profiles_by_name[new_profile.name] = new_profile\n\n    return new_profile\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.add_profile","title":"add_profile","text":"

    Add a profile to the collection.

    If the profile name already exists, an exception will be raised.

    Source code in prefect/settings.py
    def add_profile(self, profile: Profile) -> None:\n    \"\"\"\n    Add a profile to the collection.\n\n    If the profile name already exists, an exception will be raised.\n    \"\"\"\n    if profile.name in self.profiles_by_name:\n        raise ValueError(\n            f\"Profile name {profile.name!r} already exists in collection.\"\n        )\n\n    self.profiles_by_name[profile.name] = profile\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.remove_profile","title":"remove_profile","text":"

    Remove a profile from the collection.

    Source code in prefect/settings.py
    def remove_profile(self, name: str) -> None:\n    \"\"\"\n    Remove a profile from the collection.\n    \"\"\"\n    self.profiles_by_name.pop(name)\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.without_profile_source","title":"without_profile_source","text":"

    Remove profiles that were loaded from a given path.

    Returns a new collection.

    Source code in prefect/settings.py
    def without_profile_source(self, path: Optional[Path]) -> \"ProfilesCollection\":\n    \"\"\"\n    Remove profiles that were loaded from a given path.\n\n    Returns a new collection.\n    \"\"\"\n    return ProfilesCollection(\n        [\n            profile\n            for profile in self.profiles_by_name.values()\n            if profile.source != path\n        ],\n        active=self.active_name,\n    )\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.get_extra_loggers","title":"get_extra_loggers","text":"

    value_callback for PREFECT_LOGGING_EXTRA_LOGGERSthat parses the CSV string into a list and trims whitespace from logger names.

    Source code in prefect/settings.py
    def get_extra_loggers(_: \"Settings\", value: str) -> List[str]:\n    \"\"\"\n    `value_callback` for `PREFECT_LOGGING_EXTRA_LOGGERS`that parses the CSV string into a\n    list and trims whitespace from logger names.\n    \"\"\"\n    return [name.strip() for name in value.split(\",\")] if value else []\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.debug_mode_log_level","title":"debug_mode_log_level","text":"

    value_callback for PREFECT_LOGGING_LEVEL that overrides the log level to DEBUG when debug mode is enabled.

    Source code in prefect/settings.py
    def debug_mode_log_level(settings, value):\n    \"\"\"\n    `value_callback` for `PREFECT_LOGGING_LEVEL` that overrides the log level to DEBUG\n    when debug mode is enabled.\n    \"\"\"\n    if PREFECT_DEBUG_MODE.value_from(settings):\n        return \"DEBUG\"\n    else:\n        return value\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.only_return_value_in_test_mode","title":"only_return_value_in_test_mode","text":"

    value_callback for PREFECT_TEST_SETTING that only allows access during test mode

    Source code in prefect/settings.py
    def only_return_value_in_test_mode(settings, value):\n    \"\"\"\n    `value_callback` for `PREFECT_TEST_SETTING` that only allows access during test mode\n    \"\"\"\n    if PREFECT_TEST_MODE.value_from(settings):\n        return value\n    else:\n        return None\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.default_ui_api_url","title":"default_ui_api_url","text":"

    value_callback for PREFECT_UI_API_URL that sets the default value to relative path '/api', otherwise it constructs an API URL from the API settings.

    Source code in prefect/settings.py
    def default_ui_api_url(settings, value):\n    \"\"\"\n    `value_callback` for `PREFECT_UI_API_URL` that sets the default value to\n    relative path '/api', otherwise it constructs an API URL from the API settings.\n    \"\"\"\n    if value is None:\n        # Set a default value\n        value = \"/api\"\n\n    return template_with_settings(\n        PREFECT_SERVER_API_HOST, PREFECT_SERVER_API_PORT, PREFECT_API_URL\n    )(settings, value)\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.status_codes_as_integers_in_range","title":"status_codes_as_integers_in_range","text":"

    value_callback for PREFECT_CLIENT_RETRY_EXTRA_CODES that ensures status codes are integers in the range 100-599.

    Source code in prefect/settings.py
    def status_codes_as_integers_in_range(_, value):\n    \"\"\"\n    `value_callback` for `PREFECT_CLIENT_RETRY_EXTRA_CODES` that ensures status codes\n    are integers in the range 100-599.\n    \"\"\"\n    if value == \"\":\n        return set()\n\n    values = {v.strip() for v in value.split(\",\")}\n\n    if any(not v.isdigit() or int(v) < 100 or int(v) > 599 for v in values):\n        raise ValueError(\n            \"PREFECT_CLIENT_RETRY_EXTRA_CODES must be a comma separated list of \"\n            \"integers between 100 and 599.\"\n        )\n\n    values = {int(v) for v in values}\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.template_with_settings","title":"template_with_settings","text":"

    Returns a value_callback that will template the given settings into the runtime value for the setting.

    Source code in prefect/settings.py
    def template_with_settings(*upstream_settings: Setting) -> Callable[[\"Settings\", T], T]:\n    \"\"\"\n    Returns a `value_callback` that will template the given settings into the runtime\n    value for the setting.\n    \"\"\"\n\n    def templater(settings, value):\n        if value is None:\n            return value  # Do not attempt to template a null string\n\n        original_type = type(value)\n        template_values = {\n            setting.name: setting.value_from(settings) for setting in upstream_settings\n        }\n        template = string.Template(str(value))\n        return original_type(template.substitute(template_values))\n\n    return templater\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.max_log_size_smaller_than_batch_size","title":"max_log_size_smaller_than_batch_size","text":"

    Validator for settings asserting the batch size and match log size are compatible

    Source code in prefect/settings.py
    def max_log_size_smaller_than_batch_size(values):\n    \"\"\"\n    Validator for settings asserting the batch size and match log size are compatible\n    \"\"\"\n    if (\n        values[\"PREFECT_LOGGING_TO_API_BATCH_SIZE\"]\n        < values[\"PREFECT_LOGGING_TO_API_MAX_LOG_SIZE\"]\n    ):\n        raise ValueError(\n            \"`PREFECT_LOGGING_TO_API_MAX_LOG_SIZE` cannot be larger than\"\n            \" `PREFECT_LOGGING_TO_API_BATCH_SIZE`\"\n        )\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.warn_on_database_password_value_without_usage","title":"warn_on_database_password_value_without_usage","text":"

    Validator for settings warning if the database password is set but not used.

    Source code in prefect/settings.py
    def warn_on_database_password_value_without_usage(values):\n    \"\"\"\n    Validator for settings warning if the database password is set but not used.\n    \"\"\"\n    value = values[\"PREFECT_API_DATABASE_PASSWORD\"]\n    if (\n        value\n        and not value.startswith(OBFUSCATED_PREFIX)\n        and (\n            \"PREFECT_API_DATABASE_PASSWORD\"\n            not in values[\"PREFECT_API_DATABASE_CONNECTION_URL\"]\n        )\n    ):\n        warnings.warn(\n            \"PREFECT_API_DATABASE_PASSWORD is set but not included in the \"\n            \"PREFECT_API_DATABASE_CONNECTION_URL. \"\n            \"The provided password will be ignored.\"\n        )\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.warn_on_misconfigured_api_url","title":"warn_on_misconfigured_api_url","text":"

    Validator for settings warning if the API URL is misconfigured.

    Source code in prefect/settings.py
    def warn_on_misconfigured_api_url(values):\n    \"\"\"\n    Validator for settings warning if the API URL is misconfigured.\n    \"\"\"\n    api_url = values[\"PREFECT_API_URL\"]\n    if api_url is not None:\n        misconfigured_mappings = {\n            \"app.prefect.cloud\": (\n                \"`PREFECT_API_URL` points to `app.prefect.cloud`. Did you\"\n                \" mean `api.prefect.cloud`?\"\n            ),\n            \"account/\": (\n                \"`PREFECT_API_URL` uses `/account/` but should use `/accounts/`.\"\n            ),\n            \"workspace/\": (\n                \"`PREFECT_API_URL` uses `/workspace/` but should use `/workspaces/`.\"\n            ),\n        }\n        warnings_list = []\n\n        for misconfig, warning in misconfigured_mappings.items():\n            if misconfig in api_url:\n                warnings_list.append(warning)\n\n        parsed_url = urlparse(api_url)\n        if parsed_url.path and not parsed_url.path.startswith(\"/api\"):\n            warnings_list.append(\n                \"`PREFECT_API_URL` should have `/api` after the base URL.\"\n            )\n\n        if warnings_list:\n            example = (\n                'e.g. PREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]\"'\n            )\n            warnings_list.append(example)\n\n            warnings.warn(\"\\n\".join(warnings_list), stacklevel=2)\n\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.get_current_settings","title":"get_current_settings","text":"

    Returns a settings object populated with values from the current settings context or, if no settings context is active, the environment.

    Source code in prefect/settings.py
    def get_current_settings() -> Settings:\n    \"\"\"\n    Returns a settings object populated with values from the current settings context\n    or, if no settings context is active, the environment.\n    \"\"\"\n    from prefect.context import SettingsContext\n\n    settings_context = SettingsContext.get()\n    if settings_context is not None:\n        return settings_context.settings\n\n    return get_settings_from_env()\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.get_settings_from_env","title":"get_settings_from_env","text":"

    Returns a settings object populated with default values and overrides from environment variables, ignoring any values in profiles.

    Calls with the same environment return a cached object instead of reconstructing to avoid validation overhead.

    Source code in prefect/settings.py
    def get_settings_from_env() -> Settings:\n    \"\"\"\n    Returns a settings object populated with default values and overrides from\n    environment variables, ignoring any values in profiles.\n\n    Calls with the same environment return a cached object instead of reconstructing\n    to avoid validation overhead.\n    \"\"\"\n    # Since os.environ is a Dict[str, str] we can safely hash it by contents, but we\n    # must be careful to avoid hashing a generator instead of a tuple\n    cache_key = hash(tuple((key, value) for key, value in os.environ.items()))\n\n    if cache_key not in _FROM_ENV_CACHE:\n        _FROM_ENV_CACHE[cache_key] = Settings()\n\n    return _FROM_ENV_CACHE[cache_key]\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.get_default_settings","title":"get_default_settings","text":"

    Returns a settings object populated with default values, ignoring any overrides from environment variables or profiles.

    This is cached since the defaults should not change during the lifetime of the module.

    Source code in prefect/settings.py
    def get_default_settings() -> Settings:\n    \"\"\"\n    Returns a settings object populated with default values, ignoring any overrides\n    from environment variables or profiles.\n\n    This is cached since the defaults should not change during the lifetime of the\n    module.\n    \"\"\"\n    global _DEFAULTS_CACHE\n\n    if not _DEFAULTS_CACHE:\n        old = os.environ\n        try:\n            os.environ = {}\n            settings = get_settings_from_env()\n        finally:\n            os.environ = old\n\n        _DEFAULTS_CACHE = settings\n\n    return _DEFAULTS_CACHE\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.temporary_settings","title":"temporary_settings","text":"

    Temporarily override the current settings by entering a new profile.

    See Settings.copy_with_update for details on different argument behavior.

    Examples:

    >>> from prefect.settings import PREFECT_API_URL\n>>>\n>>> with temporary_settings(updates={PREFECT_API_URL: \"foo\"}):\n>>>    assert PREFECT_API_URL.value() == \"foo\"\n>>>\n>>>    with temporary_settings(set_defaults={PREFECT_API_URL: \"bar\"}):\n>>>         assert PREFECT_API_URL.value() == \"foo\"\n>>>\n>>>    with temporary_settings(restore_defaults={PREFECT_API_URL}):\n>>>         assert PREFECT_API_URL.value() is None\n>>>\n>>>         with temporary_settings(set_defaults={PREFECT_API_URL: \"bar\"})\n>>>             assert PREFECT_API_URL.value() == \"bar\"\n>>> assert PREFECT_API_URL.value() is None\n
    Source code in prefect/settings.py
    @contextmanager\ndef temporary_settings(\n    updates: Mapping[Setting, Any] = None,\n    set_defaults: Mapping[Setting, Any] = None,\n    restore_defaults: Iterable[Setting] = None,\n) -> Settings:\n    \"\"\"\n    Temporarily override the current settings by entering a new profile.\n\n    See `Settings.copy_with_update` for details on different argument behavior.\n\n    Examples:\n        >>> from prefect.settings import PREFECT_API_URL\n        >>>\n        >>> with temporary_settings(updates={PREFECT_API_URL: \"foo\"}):\n        >>>    assert PREFECT_API_URL.value() == \"foo\"\n        >>>\n        >>>    with temporary_settings(set_defaults={PREFECT_API_URL: \"bar\"}):\n        >>>         assert PREFECT_API_URL.value() == \"foo\"\n        >>>\n        >>>    with temporary_settings(restore_defaults={PREFECT_API_URL}):\n        >>>         assert PREFECT_API_URL.value() is None\n        >>>\n        >>>         with temporary_settings(set_defaults={PREFECT_API_URL: \"bar\"})\n        >>>             assert PREFECT_API_URL.value() == \"bar\"\n        >>> assert PREFECT_API_URL.value() is None\n    \"\"\"\n    import prefect.context\n\n    context = prefect.context.get_settings_context()\n\n    new_settings = context.settings.copy_with_update(\n        updates=updates, set_defaults=set_defaults, restore_defaults=restore_defaults\n    )\n\n    with prefect.context.SettingsContext(\n        profile=context.profile, settings=new_settings\n    ):\n        yield new_settings\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.load_profiles","title":"load_profiles","text":"

    Load all profiles from the default and current profile paths.

    Source code in prefect/settings.py
    def load_profiles() -> ProfilesCollection:\n    \"\"\"\n    Load all profiles from the default and current profile paths.\n    \"\"\"\n    profiles = _read_profiles_from(DEFAULT_PROFILES_PATH)\n\n    user_profiles_path = PREFECT_PROFILES_PATH.value()\n    if user_profiles_path.exists():\n        user_profiles = _read_profiles_from(user_profiles_path)\n\n        # Merge all of the user profiles with the defaults\n        for name in user_profiles:\n            profiles.update_profile(\n                name,\n                settings=user_profiles[name].settings,\n                source=user_profiles[name].source,\n            )\n\n        if user_profiles.active_name:\n            profiles.set_active(user_profiles.active_name, check=False)\n\n    return profiles\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.load_current_profile","title":"load_current_profile","text":"

    Load the current profile from the default and current profile paths.

    This will not include settings from the current settings context. Only settings that have been persisted to the profiles file will be saved.

    Source code in prefect/settings.py
    def load_current_profile():\n    \"\"\"\n    Load the current profile from the default and current profile paths.\n\n    This will _not_ include settings from the current settings context. Only settings\n    that have been persisted to the profiles file will be saved.\n    \"\"\"\n    from prefect.context import SettingsContext\n\n    profiles = load_profiles()\n    context = SettingsContext.get()\n\n    if context:\n        profiles.set_active(context.profile.name)\n\n    return profiles.active_profile\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.save_profiles","title":"save_profiles","text":"

    Writes all non-default profiles to the current profiles path.

    Source code in prefect/settings.py
    def save_profiles(profiles: ProfilesCollection) -> None:\n    \"\"\"\n    Writes all non-default profiles to the current profiles path.\n    \"\"\"\n    profiles_path = PREFECT_PROFILES_PATH.value()\n    profiles = profiles.without_profile_source(DEFAULT_PROFILES_PATH)\n    return _write_profiles_to(profiles_path, profiles)\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.load_profile","title":"load_profile","text":"

    Load a single profile by name.

    Source code in prefect/settings.py
    def load_profile(name: str) -> Profile:\n    \"\"\"\n    Load a single profile by name.\n    \"\"\"\n    profiles = load_profiles()\n    try:\n        return profiles[name]\n    except KeyError:\n        raise ValueError(f\"Profile {name!r} not found.\")\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.update_current_profile","title":"update_current_profile","text":"

    Update the persisted data for the profile currently in-use.

    If the profile does not exist in the profiles file, it will be created.

    Given settings will be merged with the existing settings as described in ProfilesCollection.update_profile.

    Returns:

    Type Description Profile

    The new profile.

    Source code in prefect/settings.py
    def update_current_profile(settings: Dict[Union[str, Setting], Any]) -> Profile:\n    \"\"\"\n    Update the persisted data for the profile currently in-use.\n\n    If the profile does not exist in the profiles file, it will be created.\n\n    Given settings will be merged with the existing settings as described in\n    `ProfilesCollection.update_profile`.\n\n    Returns:\n        The new profile.\n    \"\"\"\n    import prefect.context\n\n    current_profile = prefect.context.get_settings_context().profile\n\n    if not current_profile:\n        raise MissingProfileError(\"No profile is currently in use.\")\n\n    profiles = load_profiles()\n\n    # Ensure the current profile's settings are present\n    profiles.update_profile(current_profile.name, current_profile.settings)\n    # Then merge the new settings in\n    new_profile = profiles.update_profile(current_profile.name, settings)\n\n    # Validate before saving\n    new_profile.validate_settings()\n\n    save_profiles(profiles)\n\n    return profiles[current_profile.name]\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/software/","title":"prefect.software","text":"","tags":["Python API","software"]},{"location":"api-ref/prefect/software/#prefect.software","title":"prefect.software","text":"","tags":["Python API","software"]},{"location":"api-ref/prefect/states/","title":"prefect.states","text":"","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states","title":"prefect.states","text":"","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.AwaitingRetry","title":"AwaitingRetry","text":"

    Convenience function for creating AwaitingRetry states.

    Returns:

    Name Type Description State State

    a AwaitingRetry state

    Source code in prefect/states.py
    def AwaitingRetry(\n    cls: Type[State] = State, scheduled_time: datetime.datetime = None, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `AwaitingRetry` states.\n\n    Returns:\n        State: a AwaitingRetry state\n    \"\"\"\n    return Scheduled(\n        cls=cls, scheduled_time=scheduled_time, name=\"AwaitingRetry\", **kwargs\n    )\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Cancelled","title":"Cancelled","text":"

    Convenience function for creating Cancelled states.

    Returns:

    Name Type Description State State

    a Cancelled state

    Source code in prefect/states.py
    def Cancelled(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Cancelled` states.\n\n    Returns:\n        State: a Cancelled state\n    \"\"\"\n    return cls(type=StateType.CANCELLED, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Cancelling","title":"Cancelling","text":"

    Convenience function for creating Cancelling states.

    Returns:

    Name Type Description State State

    a Cancelling state

    Source code in prefect/states.py
    def Cancelling(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Cancelling` states.\n\n    Returns:\n        State: a Cancelling state\n    \"\"\"\n    return cls(type=StateType.CANCELLING, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Completed","title":"Completed","text":"

    Convenience function for creating Completed states.

    Returns:

    Name Type Description State State

    a Completed state

    Source code in prefect/states.py
    def Completed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Completed` states.\n\n    Returns:\n        State: a Completed state\n    \"\"\"\n    return cls(type=StateType.COMPLETED, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Crashed","title":"Crashed","text":"

    Convenience function for creating Crashed states.

    Returns:

    Name Type Description State State

    a Crashed state

    Source code in prefect/states.py
    def Crashed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Crashed` states.\n\n    Returns:\n        State: a Crashed state\n    \"\"\"\n    return cls(type=StateType.CRASHED, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Failed","title":"Failed","text":"

    Convenience function for creating Failed states.

    Returns:

    Name Type Description State State

    a Failed state

    Source code in prefect/states.py
    def Failed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Failed` states.\n\n    Returns:\n        State: a Failed state\n    \"\"\"\n    return cls(type=StateType.FAILED, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Late","title":"Late","text":"

    Convenience function for creating Late states.

    Returns:

    Name Type Description State State

    a Late state

    Source code in prefect/states.py
    def Late(\n    cls: Type[State] = State, scheduled_time: datetime.datetime = None, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `Late` states.\n\n    Returns:\n        State: a Late state\n    \"\"\"\n    return Scheduled(cls=cls, scheduled_time=scheduled_time, name=\"Late\", **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Paused","title":"Paused","text":"

    Convenience function for creating Paused states.

    Returns:

    Name Type Description State State

    a Paused state

    Source code in prefect/states.py
    def Paused(\n    cls: Type[State] = State,\n    timeout_seconds: Optional[int] = None,\n    pause_expiration_time: Optional[datetime.datetime] = None,\n    reschedule: bool = False,\n    pause_key: Optional[str] = None,\n    **kwargs,\n) -> State:\n    \"\"\"Convenience function for creating `Paused` states.\n\n    Returns:\n        State: a Paused state\n    \"\"\"\n    state_details = StateDetails.parse_obj(kwargs.pop(\"state_details\", {}))\n\n    if state_details.pause_timeout:\n        raise ValueError(\"An extra pause timeout was provided in state_details\")\n\n    if pause_expiration_time is not None and timeout_seconds is not None:\n        raise ValueError(\n            \"Cannot supply both a pause_expiration_time and timeout_seconds\"\n        )\n\n    if pause_expiration_time is None and timeout_seconds is None:\n        pass\n    else:\n        state_details.pause_timeout = pause_expiration_time or (\n            pendulum.now(\"UTC\") + pendulum.Duration(seconds=timeout_seconds)\n        )\n\n    state_details.pause_reschedule = reschedule\n    state_details.pause_key = pause_key\n\n    return cls(type=StateType.PAUSED, state_details=state_details, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Pending","title":"Pending","text":"

    Convenience function for creating Pending states.

    Returns:

    Name Type Description State State

    a Pending state

    Source code in prefect/states.py
    def Pending(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Pending` states.\n\n    Returns:\n        State: a Pending state\n    \"\"\"\n    return cls(type=StateType.PENDING, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Retrying","title":"Retrying","text":"

    Convenience function for creating Retrying states.

    Returns:

    Name Type Description State State

    a Retrying state

    Source code in prefect/states.py
    def Retrying(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Retrying` states.\n\n    Returns:\n        State: a Retrying state\n    \"\"\"\n    return cls(type=StateType.RUNNING, name=\"Retrying\", **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Running","title":"Running","text":"

    Convenience function for creating Running states.

    Returns:

    Name Type Description State State

    a Running state

    Source code in prefect/states.py
    def Running(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Running` states.\n\n    Returns:\n        State: a Running state\n    \"\"\"\n    return cls(type=StateType.RUNNING, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Scheduled","title":"Scheduled","text":"

    Convenience function for creating Scheduled states.

    Returns:

    Name Type Description State State

    a Scheduled state

    Source code in prefect/states.py
    def Scheduled(\n    cls: Type[State] = State, scheduled_time: datetime.datetime = None, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `Scheduled` states.\n\n    Returns:\n        State: a Scheduled state\n    \"\"\"\n    state_details = StateDetails.parse_obj(kwargs.pop(\"state_details\", {}))\n    if scheduled_time is None:\n        scheduled_time = pendulum.now(\"UTC\")\n    elif state_details.scheduled_time:\n        raise ValueError(\"An extra scheduled_time was provided in state_details\")\n    state_details.scheduled_time = scheduled_time\n\n    return cls(type=StateType.SCHEDULED, state_details=state_details, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Suspended","title":"Suspended","text":"

    Convenience function for creating Suspended states.

    Returns:

    Name Type Description State

    a Suspended state

    Source code in prefect/states.py
    def Suspended(\n    cls: Type[State] = State,\n    timeout_seconds: Optional[int] = None,\n    pause_expiration_time: Optional[datetime.datetime] = None,\n    pause_key: Optional[str] = None,\n    **kwargs,\n):\n    \"\"\"Convenience function for creating `Suspended` states.\n\n    Returns:\n        State: a Suspended state\n    \"\"\"\n    return Paused(\n        cls=cls,\n        name=\"Suspended\",\n        reschedule=True,\n        timeout_seconds=timeout_seconds,\n        pause_expiration_time=pause_expiration_time,\n        pause_key=pause_key,\n        **kwargs,\n    )\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.exception_to_crashed_state","title":"exception_to_crashed_state async","text":"

    Takes an exception that occurs outside of user code and converts it to a 'Crash' exception with a 'Crashed' state.

    Source code in prefect/states.py
    async def exception_to_crashed_state(\n    exc: BaseException,\n    result_factory: Optional[ResultFactory] = None,\n) -> State:\n    \"\"\"\n    Takes an exception that occurs _outside_ of user code and converts it to a\n    'Crash' exception with a 'Crashed' state.\n    \"\"\"\n    state_message = None\n\n    if isinstance(exc, anyio.get_cancelled_exc_class()):\n        state_message = \"Execution was cancelled by the runtime environment.\"\n\n    elif isinstance(exc, KeyboardInterrupt):\n        state_message = \"Execution was aborted by an interrupt signal.\"\n\n    elif isinstance(exc, TerminationSignal):\n        state_message = \"Execution was aborted by a termination signal.\"\n\n    elif isinstance(exc, SystemExit):\n        state_message = \"Execution was aborted by Python system exit call.\"\n\n    elif isinstance(exc, (httpx.TimeoutException, httpx.ConnectError)):\n        try:\n            request: httpx.Request = exc.request\n        except RuntimeError:\n            # The request property is not set\n            state_message = (\n                \"Request failed while attempting to contact the server:\"\n                f\" {format_exception(exc)}\"\n            )\n        else:\n            # TODO: We can check if this is actually our API url\n            state_message = f\"Request to {request.url} failed: {format_exception(exc)}.\"\n\n    else:\n        state_message = (\n            \"Execution was interrupted by an unexpected exception:\"\n            f\" {format_exception(exc)}\"\n        )\n\n    if result_factory:\n        data = await result_factory.create_result(exc)\n    else:\n        # Attach the exception for local usage, will not be available when retrieved\n        # from the API\n        data = exc\n\n    return Crashed(message=state_message, data=data)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.exception_to_failed_state","title":"exception_to_failed_state async","text":"

    Convenience function for creating Failed states from exceptions

    Source code in prefect/states.py
    async def exception_to_failed_state(\n    exc: Optional[BaseException] = None,\n    result_factory: Optional[ResultFactory] = None,\n    **kwargs,\n) -> State:\n    \"\"\"\n    Convenience function for creating `Failed` states from exceptions\n    \"\"\"\n    if not exc:\n        _, exc, _ = sys.exc_info()\n        if exc is None:\n            raise ValueError(\n                \"Exception was not passed and no active exception could be found.\"\n            )\n    else:\n        pass\n\n    if result_factory:\n        data = await result_factory.create_result(exc)\n    else:\n        # Attach the exception for local usage, will not be available when retrieved\n        # from the API\n        data = exc\n\n    existing_message = kwargs.pop(\"message\", \"\")\n    if existing_message and not existing_message.endswith(\" \"):\n        existing_message += \" \"\n\n    # TODO: Consider if we want to include traceback information, it is intentionally\n    #       excluded from messages for now\n    message = existing_message + format_exception(exc)\n\n    return Failed(data=data, message=message, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.get_state_exception","title":"get_state_exception async","text":"

    If not given a FAILED or CRASHED state, this raise a value error.

    If the state result is a state, its exception will be returned.

    If the state result is an iterable of states, the exception of the first failure will be returned.

    If the state result is a string, a wrapper exception will be returned with the string as the message.

    If the state result is null, a wrapper exception will be returned with the state message attached.

    If the state result is not of a known type, a TypeError will be returned.

    When a wrapper exception is returned, the type will be: - FailedRun if the state type is FAILED. - CrashedRun if the state type is CRASHED. - CancelledRun if the state type is CANCELLED.

    Source code in prefect/states.py
    @sync_compatible\nasync def get_state_exception(state: State) -> BaseException:\n    \"\"\"\n    If not given a FAILED or CRASHED state, this raise a value error.\n\n    If the state result is a state, its exception will be returned.\n\n    If the state result is an iterable of states, the exception of the first failure\n    will be returned.\n\n    If the state result is a string, a wrapper exception will be returned with the\n    string as the message.\n\n    If the state result is null, a wrapper exception will be returned with the state\n    message attached.\n\n    If the state result is not of a known type, a `TypeError` will be returned.\n\n    When a wrapper exception is returned, the type will be:\n        - `FailedRun` if the state type is FAILED.\n        - `CrashedRun` if the state type is CRASHED.\n        - `CancelledRun` if the state type is CANCELLED.\n    \"\"\"\n\n    if state.is_failed():\n        wrapper = FailedRun\n        default_message = \"Run failed.\"\n    elif state.is_crashed():\n        wrapper = CrashedRun\n        default_message = \"Run crashed.\"\n    elif state.is_cancelled():\n        wrapper = CancelledRun\n        default_message = \"Run cancelled.\"\n    else:\n        raise ValueError(f\"Expected failed or crashed state got {state!r}.\")\n\n    if isinstance(state.data, BaseResult):\n        result = await state.data.get()\n    elif state.data is None:\n        result = None\n    else:\n        result = state.data\n\n    if result is None:\n        return wrapper(state.message or default_message)\n\n    if isinstance(result, Exception):\n        return result\n\n    elif isinstance(result, BaseException):\n        return result\n\n    elif isinstance(result, str):\n        return wrapper(result)\n\n    elif is_state(result):\n        # Return the exception from the inner state\n        return await get_state_exception(result)\n\n    elif is_state_iterable(result):\n        # Return the first failure\n        for state in result:\n            if state.is_failed() or state.is_crashed() or state.is_cancelled():\n                return await get_state_exception(state)\n\n        raise ValueError(\n            \"Failed state result was an iterable of states but none were failed.\"\n        )\n\n    else:\n        raise TypeError(\n            f\"Unexpected result for failed state: {result!r} \u2014\u2014 \"\n            f\"{type(result).__name__} cannot be resolved into an exception\"\n        )\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.get_state_result","title":"get_state_result","text":"

    Get the result from a state.

    See State.result()

    Source code in prefect/states.py
    def get_state_result(\n    state: State[R], raise_on_failure: bool = True, fetch: Optional[bool] = None\n) -> R:\n    \"\"\"\n    Get the result from a state.\n\n    See `State.result()`\n    \"\"\"\n\n    if fetch is None and (\n        PREFECT_ASYNC_FETCH_STATE_RESULT or not in_async_main_thread()\n    ):\n        # Fetch defaults to `True` for sync users or async users who have opted in\n        fetch = True\n\n    if not fetch:\n        if fetch is None and in_async_main_thread():\n            warnings.warn(\n                (\n                    \"State.result() was called from an async context but not awaited. \"\n                    \"This method will be updated to return a coroutine by default in \"\n                    \"the future. Pass `fetch=True` and `await` the call to get rid of \"\n                    \"this warning.\"\n                ),\n                DeprecationWarning,\n                stacklevel=2,\n            )\n        # Backwards compatibility\n        if isinstance(state.data, DataDocument):\n            return result_from_state_with_data_document(\n                state, raise_on_failure=raise_on_failure\n            )\n        else:\n            return state.data\n    else:\n        return _get_state_result(state, raise_on_failure=raise_on_failure)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.is_state","title":"is_state","text":"

    Check if the given object is a state instance

    Source code in prefect/states.py
    def is_state(obj: Any) -> TypeGuard[State]:\n    \"\"\"\n    Check if the given object is a state instance\n    \"\"\"\n    # We may want to narrow this to client-side state types but for now this provides\n    # backwards compatibility\n    try:\n        from prefect.server.schemas.states import State as State_\n\n        classes_ = (State, State_)\n    except ImportError:\n        classes_ = State\n\n    # return isinstance(obj, (State, State_))\n    return isinstance(obj, classes_)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.is_state_iterable","title":"is_state_iterable","text":"

    Check if a the given object is an iterable of states types

    Supported iterables are: - set - list - tuple

    Other iterables will return False even if they contain states.

    Source code in prefect/states.py
    def is_state_iterable(obj: Any) -> TypeGuard[Iterable[State]]:\n    \"\"\"\n    Check if a the given object is an iterable of states types\n\n    Supported iterables are:\n    - set\n    - list\n    - tuple\n\n    Other iterables will return `False` even if they contain states.\n    \"\"\"\n    # We do not check for arbitrary iterables because this is not intended to be used\n    # for things like dictionaries, dataframes, or pydantic models\n    if (\n        not isinstance(obj, BaseAnnotation)\n        and isinstance(obj, (list, set, tuple))\n        and obj\n    ):\n        return all([is_state(o) for o in obj])\n    else:\n        return False\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.raise_state_exception","title":"raise_state_exception async","text":"

    Given a FAILED or CRASHED state, raise the contained exception.

    Source code in prefect/states.py
    @sync_compatible\nasync def raise_state_exception(state: State) -> None:\n    \"\"\"\n    Given a FAILED or CRASHED state, raise the contained exception.\n    \"\"\"\n    if not (state.is_failed() or state.is_crashed() or state.is_cancelled()):\n        return None\n\n    raise await get_state_exception(state)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.return_value_to_state","title":"return_value_to_state async","text":"

    Given a return value from a user's function, create a State the run should be placed in.

    • If data is returned, we create a 'COMPLETED' state with the data
    • If a single, manually created state is returned, we use that state as given (manual creation is determined by the lack of ids)
    • If an upstream state or iterable of upstream states is returned, we apply the aggregate rule

    The aggregate rule says that given multiple states we will determine the final state such that:

    • If any states are not COMPLETED the final state is FAILED
    • If all of the states are COMPLETED the final state is COMPLETED
    • The states will be placed in the final state data attribute

    Callers should resolve all futures into states before passing return values to this function.

    Source code in prefect/states.py
    async def return_value_to_state(retval: R, result_factory: ResultFactory) -> State[R]:\n    \"\"\"\n    Given a return value from a user's function, create a `State` the run should\n    be placed in.\n\n    - If data is returned, we create a 'COMPLETED' state with the data\n    - If a single, manually created state is returned, we use that state as given\n        (manual creation is determined by the lack of ids)\n    - If an upstream state or iterable of upstream states is returned, we apply the\n        aggregate rule\n\n    The aggregate rule says that given multiple states we will determine the final state\n    such that:\n\n    - If any states are not COMPLETED the final state is FAILED\n    - If all of the states are COMPLETED the final state is COMPLETED\n    - The states will be placed in the final state `data` attribute\n\n    Callers should resolve all futures into states before passing return values to this\n    function.\n    \"\"\"\n\n    if (\n        is_state(retval)\n        # Check for manual creation\n        and not retval.state_details.flow_run_id\n        and not retval.state_details.task_run_id\n    ):\n        state = retval\n\n        # Do not modify states with data documents attached; backwards compatibility\n        if isinstance(state.data, DataDocument):\n            return state\n\n        # Unless the user has already constructed a result explicitly, use the factory\n        # to update the data to the correct type\n        if not isinstance(state.data, BaseResult):\n            state.data = await result_factory.create_result(state.data)\n\n        return state\n\n    # Determine a new state from the aggregate of contained states\n    if is_state(retval) or is_state_iterable(retval):\n        states = StateGroup(ensure_iterable(retval))\n\n        # Determine the new state type\n        if states.all_completed():\n            new_state_type = StateType.COMPLETED\n        elif states.any_cancelled():\n            new_state_type = StateType.CANCELLED\n        elif states.any_paused():\n            new_state_type = StateType.PAUSED\n        else:\n            new_state_type = StateType.FAILED\n\n        # Generate a nice message for the aggregate\n        if states.all_completed():\n            message = \"All states completed.\"\n        elif states.any_cancelled():\n            message = f\"{states.cancelled_count}/{states.total_count} states cancelled.\"\n        elif states.any_paused():\n            message = f\"{states.paused_count}/{states.total_count} states paused.\"\n        elif states.any_failed():\n            message = f\"{states.fail_count}/{states.total_count} states failed.\"\n        elif not states.all_final():\n            message = (\n                f\"{states.not_final_count}/{states.total_count} states are not final.\"\n            )\n        else:\n            message = \"Given states: \" + states.counts_message()\n\n        # TODO: We may actually want to set the data to a `StateGroup` object and just\n        #       allow it to be unpacked into a tuple and such so users can interact with\n        #       it\n        return State(\n            type=new_state_type,\n            message=message,\n            data=await result_factory.create_result(retval),\n        )\n\n    # Generators aren't portable, implicitly convert them to a list.\n    if isinstance(retval, GeneratorType):\n        data = list(retval)\n    else:\n        data = retval\n\n    # Otherwise, they just gave data and this is a completed retval\n    return Completed(data=await result_factory.create_result(data))\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/task-runners/","title":"prefect.task_runners","text":"","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners","title":"prefect.task_runners","text":"

    Interface and implementations of various task runners.

    Task Runners in Prefect are responsible for managing the execution of Prefect task runs. Generally speaking, users are not expected to interact with task runners outside of configuring and initializing them for a flow.

    Example
    >>> from prefect import flow, task\n>>> from prefect.task_runners import SequentialTaskRunner\n>>> from typing import List\n>>>\n>>> @task\n>>> def say_hello(name):\n...     print(f\"hello {name}\")\n>>>\n>>> @task\n>>> def say_goodbye(name):\n...     print(f\"goodbye {name}\")\n>>>\n>>> @flow(task_runner=SequentialTaskRunner())\n>>> def greetings(names: List[str]):\n...     for name in names:\n...         say_hello(name)\n...         say_goodbye(name)\n>>>\n>>> greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\nhello arthur\ngoodbye arthur\nhello trillian\ngoodbye trillian\nhello ford\ngoodbye ford\nhello marvin\ngoodbye marvin\n

    Switching to a DaskTaskRunner:

    >>> from prefect_dask.task_runners import DaskTaskRunner\n>>> flow.task_runner = DaskTaskRunner()\n>>> greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\nhello arthur\ngoodbye arthur\nhello trillian\nhello ford\ngoodbye marvin\nhello marvin\ngoodbye ford\ngoodbye trillian\n

    For usage details, see the Task Runners documentation.

    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner","title":"BaseTaskRunner","text":"Source code in prefect/task_runners.py
    class BaseTaskRunner(metaclass=abc.ABCMeta):\n    def __init__(self) -> None:\n        self.logger = get_logger(f\"task_runner.{self.name}\")\n        self._started: bool = False\n\n    @property\n    @abc.abstractmethod\n    def concurrency_type(self) -> TaskConcurrencyType:\n        pass  # noqa\n\n    @property\n    def name(self):\n        return type(self).__name__.lower().replace(\"taskrunner\", \"\")\n\n    def duplicate(self):\n        \"\"\"\n        Return a new task runner instance with the same options.\n        \"\"\"\n        # The base class returns `NotImplemented` to indicate that this is not yet\n        # implemented by a given task runner.\n        return NotImplemented\n\n    def __eq__(self, other: object) -> bool:\n        \"\"\"\n        Returns true if the task runners use the same options.\n        \"\"\"\n        if type(other) == type(self) and (\n            # Compare public attributes for naive equality check\n            # Subclasses should implement this method with a check init option equality\n            {k: v for k, v in self.__dict__.items() if not k.startswith(\"_\")}\n            == {k: v for k, v in other.__dict__.items() if not k.startswith(\"_\")}\n        ):\n            return True\n        else:\n            return NotImplemented\n\n    @abc.abstractmethod\n    async def submit(\n        self,\n        key: UUID,\n        call: Callable[..., Awaitable[State[R]]],\n    ) -> None:\n        \"\"\"\n        Submit a call for execution and return a `PrefectFuture` that can be used to\n        get the call result.\n\n        Args:\n            task_run: The task run being submitted.\n            task_key: A unique key for this orchestration run of the task. Can be used\n                for caching.\n            call: The function to be executed\n            run_kwargs: A dict of keyword arguments to pass to `call`\n\n        Returns:\n            A future representing the result of `call` execution\n        \"\"\"\n        raise NotImplementedError()\n\n    @abc.abstractmethod\n    async def wait(self, key: UUID, timeout: float = None) -> Optional[State]:\n        \"\"\"\n        Given a `PrefectFuture`, wait for its return state up to `timeout` seconds.\n        If it is not finished after the timeout expires, `None` should be returned.\n\n        Implementers should be careful to ensure that this function never returns or\n        raises an exception.\n        \"\"\"\n        raise NotImplementedError()\n\n    @asynccontextmanager\n    async def start(\n        self: T,\n    ) -> AsyncIterator[T]:\n        \"\"\"\n        Start the task runner, preparing any resources necessary for task submission.\n\n        Children should implement `_start` to prepare and clean up resources.\n\n        Yields:\n            The prepared task runner\n        \"\"\"\n        if self._started:\n            raise RuntimeError(\"The task runner is already started!\")\n\n        async with AsyncExitStack() as exit_stack:\n            self.logger.debug(\"Starting task runner...\")\n            try:\n                await self._start(exit_stack)\n                self._started = True\n                yield self\n            finally:\n                self.logger.debug(\"Shutting down task runner...\")\n                self._started = False\n\n    async def _start(self, exit_stack: AsyncExitStack) -> None:\n        \"\"\"\n        Create any resources required for this task runner to submit work.\n\n        Cleanup of resources should be submitted to the `exit_stack`.\n        \"\"\"\n        pass  # noqa\n\n    def __str__(self) -> str:\n        return type(self).__name__\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner.duplicate","title":"duplicate","text":"

    Return a new task runner instance with the same options.

    Source code in prefect/task_runners.py
    def duplicate(self):\n    \"\"\"\n    Return a new task runner instance with the same options.\n    \"\"\"\n    # The base class returns `NotImplemented` to indicate that this is not yet\n    # implemented by a given task runner.\n    return NotImplemented\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner.start","title":"start async","text":"

    Start the task runner, preparing any resources necessary for task submission.

    Children should implement _start to prepare and clean up resources.

    Yields:

    Type Description AsyncIterator[T]

    The prepared task runner

    Source code in prefect/task_runners.py
    @asynccontextmanager\nasync def start(\n    self: T,\n) -> AsyncIterator[T]:\n    \"\"\"\n    Start the task runner, preparing any resources necessary for task submission.\n\n    Children should implement `_start` to prepare and clean up resources.\n\n    Yields:\n        The prepared task runner\n    \"\"\"\n    if self._started:\n        raise RuntimeError(\"The task runner is already started!\")\n\n    async with AsyncExitStack() as exit_stack:\n        self.logger.debug(\"Starting task runner...\")\n        try:\n            await self._start(exit_stack)\n            self._started = True\n            yield self\n        finally:\n            self.logger.debug(\"Shutting down task runner...\")\n            self._started = False\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner.submit","title":"submit abstractmethod async","text":"

    Submit a call for execution and return a PrefectFuture that can be used to get the call result.

    Parameters:

    Name Type Description Default task_run

    The task run being submitted.

    required task_key

    A unique key for this orchestration run of the task. Can be used for caching.

    required call Callable[..., Awaitable[State[R]]]

    The function to be executed

    required run_kwargs

    A dict of keyword arguments to pass to call

    required

    Returns:

    Type Description None

    A future representing the result of call execution

    Source code in prefect/task_runners.py
    @abc.abstractmethod\nasync def submit(\n    self,\n    key: UUID,\n    call: Callable[..., Awaitable[State[R]]],\n) -> None:\n    \"\"\"\n    Submit a call for execution and return a `PrefectFuture` that can be used to\n    get the call result.\n\n    Args:\n        task_run: The task run being submitted.\n        task_key: A unique key for this orchestration run of the task. Can be used\n            for caching.\n        call: The function to be executed\n        run_kwargs: A dict of keyword arguments to pass to `call`\n\n    Returns:\n        A future representing the result of `call` execution\n    \"\"\"\n    raise NotImplementedError()\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner.wait","title":"wait abstractmethod async","text":"

    Given a PrefectFuture, wait for its return state up to timeout seconds. If it is not finished after the timeout expires, None should be returned.

    Implementers should be careful to ensure that this function never returns or raises an exception.

    Source code in prefect/task_runners.py
    @abc.abstractmethod\nasync def wait(self, key: UUID, timeout: float = None) -> Optional[State]:\n    \"\"\"\n    Given a `PrefectFuture`, wait for its return state up to `timeout` seconds.\n    If it is not finished after the timeout expires, `None` should be returned.\n\n    Implementers should be careful to ensure that this function never returns or\n    raises an exception.\n    \"\"\"\n    raise NotImplementedError()\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.ConcurrentTaskRunner","title":"ConcurrentTaskRunner","text":"

    Bases: BaseTaskRunner

    A concurrent task runner that allows tasks to switch when blocking on IO. Synchronous tasks will be submitted to a thread pool maintained by anyio.

    Example
    Using a thread for concurrency:\n>>> from prefect import flow\n>>> from prefect.task_runners import ConcurrentTaskRunner\n>>> @flow(task_runner=ConcurrentTaskRunner)\n>>> def my_flow():\n>>>     ...\n
    Source code in prefect/task_runners.py
    class ConcurrentTaskRunner(BaseTaskRunner):\n    \"\"\"\n    A concurrent task runner that allows tasks to switch when blocking on IO.\n    Synchronous tasks will be submitted to a thread pool maintained by `anyio`.\n\n    Example:\n        ```\n        Using a thread for concurrency:\n        >>> from prefect import flow\n        >>> from prefect.task_runners import ConcurrentTaskRunner\n        >>> @flow(task_runner=ConcurrentTaskRunner)\n        >>> def my_flow():\n        >>>     ...\n        ```\n    \"\"\"\n\n    def __init__(self):\n        # TODO: Consider adding `max_workers` support using anyio capacity limiters\n\n        # Runtime attributes\n        self._task_group: anyio.abc.TaskGroup = None\n        self._result_events: Dict[UUID, Event] = {}\n        self._results: Dict[UUID, Any] = {}\n        self._keys: Set[UUID] = set()\n\n        super().__init__()\n\n    @property\n    def concurrency_type(self) -> TaskConcurrencyType:\n        return TaskConcurrencyType.CONCURRENT\n\n    def duplicate(self):\n        return type(self)()\n\n    async def submit(\n        self,\n        key: UUID,\n        call: Callable[[], Awaitable[State[R]]],\n    ) -> None:\n        if not self._started:\n            raise RuntimeError(\n                \"The task runner must be started before submitting work.\"\n            )\n\n        if not self._task_group:\n            raise RuntimeError(\n                \"The concurrent task runner cannot be used to submit work after \"\n                \"serialization.\"\n            )\n\n        # Create an event to set on completion\n        self._result_events[key] = Event()\n\n        # Rely on the event loop for concurrency\n        self._task_group.start_soon(self._run_and_store_result, key, call)\n\n    async def wait(\n        self,\n        key: UUID,\n        timeout: float = None,\n    ) -> Optional[State]:\n        if not self._task_group:\n            raise RuntimeError(\n                \"The concurrent task runner cannot be used to wait for work after \"\n                \"serialization.\"\n            )\n\n        return await self._get_run_result(key, timeout)\n\n    async def _run_and_store_result(\n        self, key: UUID, call: Callable[[], Awaitable[State[R]]]\n    ):\n        \"\"\"\n        Simple utility to store the orchestration result in memory on completion\n\n        Since this run is occurring on the main thread, we capture exceptions to prevent\n        task crashes from crashing the flow run.\n        \"\"\"\n        try:\n            result = await call()\n        except BaseException as exc:\n            result = await exception_to_crashed_state(exc)\n\n        self._results[key] = result\n        self._result_events[key].set()\n\n    async def _get_run_result(\n        self, key: UUID, timeout: float = None\n    ) -> Optional[State]:\n        \"\"\"\n        Block until the run result has been populated.\n        \"\"\"\n        result = None  # retval on timeout\n\n        # Note we do not use `asyncio.wrap_future` and instead use an `Event` to avoid\n        # stdlib behavior where the wrapped future is cancelled if the parent future is\n        # cancelled (as it would be during a timeout here)\n        with anyio.move_on_after(timeout):\n            await self._result_events[key].wait()\n            result = self._results[key]\n\n        return result  # timeout reached\n\n    async def _start(self, exit_stack: AsyncExitStack):\n        \"\"\"\n        Start the process pool\n        \"\"\"\n        self._task_group = await exit_stack.enter_async_context(\n            anyio.create_task_group()\n        )\n\n    def __getstate__(self):\n        \"\"\"\n        Allow the `ConcurrentTaskRunner` to be serialized by dropping the task group.\n        \"\"\"\n        data = self.__dict__.copy()\n        data.update({k: None for k in {\"_task_group\"}})\n        return data\n\n    def __setstate__(self, data: dict):\n        \"\"\"\n        When deserialized, we will no longer have a reference to the task group.\n        \"\"\"\n        self.__dict__.update(data)\n        self._task_group = None\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.SequentialTaskRunner","title":"SequentialTaskRunner","text":"

    Bases: BaseTaskRunner

    A simple task runner that executes calls as they are submitted.

    If writing synchronous tasks, this runner will always execute tasks sequentially. If writing async tasks, this runner will execute tasks sequentially unless grouped using anyio.create_task_group or asyncio.gather.

    Source code in prefect/task_runners.py
    class SequentialTaskRunner(BaseTaskRunner):\n    \"\"\"\n    A simple task runner that executes calls as they are submitted.\n\n    If writing synchronous tasks, this runner will always execute tasks sequentially.\n    If writing async tasks, this runner will execute tasks sequentially unless grouped\n    using `anyio.create_task_group` or `asyncio.gather`.\n    \"\"\"\n\n    def __init__(self) -> None:\n        super().__init__()\n        self._results: Dict[str, State] = {}\n\n    @property\n    def concurrency_type(self) -> TaskConcurrencyType:\n        return TaskConcurrencyType.SEQUENTIAL\n\n    def duplicate(self):\n        return type(self)()\n\n    async def submit(\n        self,\n        key: UUID,\n        call: Callable[..., Awaitable[State[R]]],\n    ) -> None:\n        # Run the function immediately and store the result in memory\n        try:\n            result = await call()\n        except BaseException as exc:\n            result = await exception_to_crashed_state(exc)\n\n        self._results[key] = result\n\n    async def wait(self, key: UUID, timeout: float = None) -> Optional[State]:\n        return self._results[key]\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/tasks/","title":"prefect.tasks","text":"","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks","title":"prefect.tasks","text":"

    Module containing the base workflow task class and decorator - for most use cases, using the @task decorator is preferred.

    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.Task","title":"Task","text":"

    Bases: Generic[P, R]

    A Prefect task definition.

    Note

    We recommend using the @task decorator for most use-cases.

    Wraps a function with an entrypoint to the Prefect engine. Calling this class within a flow function creates a new task run.

    To preserve the input and output types, we use the generic type variables P and R for \"Parameters\" and \"Returns\" respectively.

    Parameters:

    Name Type Description Default fn Callable[P, R]

    The function defining the task.

    required name str

    An optional name for the task; if not provided, the name will be inferred from the given function.

    None description str

    An optional string description for the task.

    None tags Iterable[str]

    An optional set of tags to be associated with runs of this task. These tags are combined with any tags defined by a prefect.tags context at task runtime.

    None version str

    An optional string specifying the version of this task definition

    None cache_key_fn Callable[[TaskRunContext, Dict[str, Any]], Optional[str]]

    An optional callable that, given the task run context and call parameters, generates a string key; if the key matches a previous completed state, that state result will be restored instead of running the task again.

    None cache_expiration timedelta

    An optional amount of time indicating how long cached states for this task should be restorable; if not provided, cached states will never expire.

    None task_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this task; this name can be provided as a string template with the task's keyword arguments as variables, or a function that returns a string.

    None retries Optional[int]

    An optional number of times to retry on task run failure.

    None retry_delay_seconds Optional[Union[float, int, List[float], Callable[[int], List[float]]]]

    Optionally configures how long to wait before retrying the task after failure. This is only applicable if retries is nonzero. This setting can either be a number of seconds, a list of retry delays, or a callable that, given the total number of retries, generates a list of retry delays. If a number of seconds, that delay will be applied to all retries. If a list, each retry will wait for the corresponding delay before retrying. When passing a callable or a list, the number of configured retry delays cannot exceed 50.

    None retry_jitter_factor Optional[float]

    An optional factor that defines the factor to which a retry can be jittered in order to avoid a \"thundering herd\".

    None persist_result Optional[bool]

    An optional toggle indicating whether the result of this task should be persisted to result storage. Defaults to None, which indicates that Prefect should choose whether the result should be persisted depending on the features being used.

    None result_storage Optional[ResultStorage]

    An optional block to use to persist the result of this task. Defaults to the value set in the flow the task is called in.

    None result_storage_key Optional[str]

    An optional key to store the result in storage at when persisted. Defaults to a unique identifier.

    None result_serializer Optional[ResultSerializer]

    An optional serializer to use to serialize the result of this task for persistence. Defaults to the value set in the flow the task is called in.

    None timeout_seconds Union[int, float]

    An optional number of seconds indicating a maximum runtime for the task. If the task exceeds this runtime, it will be marked as failed.

    None log_prints Optional[bool]

    If set, print statements in the task will be redirected to the Prefect logger for the task run. Defaults to None, which indicates that the value from the flow should be used.

    False refresh_cache Optional[bool]

    If set, cached results for the cache key are not used. Defaults to None, which indicates that a cached result from a previous execution with matching cache key is used.

    None on_failure Optional[List[Callable[[Task, TaskRun, State], None]]]

    An optional list of callables to run when the task enters a failed state.

    None on_completion Optional[List[Callable[[Task, TaskRun, State], None]]]

    An optional list of callables to run when the task enters a completed state.

    None retry_condition_fn Optional[Callable[[Task, TaskRun, State], bool]]

    An optional callable run when a task run returns a Failed state. Should return True if the task should continue to its retry policy (e.g. retries=3), and False if the task should end as failed. Defaults to None, indicating the task should always continue to its retry policy.

    None viz_return_value Optional[Any]

    An optional value to return when the task dependency tree is visualized.

    None Source code in prefect/tasks.py
    @PrefectObjectRegistry.register_instances\nclass Task(Generic[P, R]):\n    \"\"\"\n    A Prefect task definition.\n\n    !!! note\n        We recommend using [the `@task` decorator][prefect.tasks.task] for most use-cases.\n\n    Wraps a function with an entrypoint to the Prefect engine. Calling this class within a flow function\n    creates a new task run.\n\n    To preserve the input and output types, we use the generic type variables P and R for \"Parameters\" and\n    \"Returns\" respectively.\n\n    Args:\n        fn: The function defining the task.\n        name: An optional name for the task; if not provided, the name will be inferred\n            from the given function.\n        description: An optional string description for the task.\n        tags: An optional set of tags to be associated with runs of this task. These\n            tags are combined with any tags defined by a `prefect.tags` context at\n            task runtime.\n        version: An optional string specifying the version of this task definition\n        cache_key_fn: An optional callable that, given the task run context and call\n            parameters, generates a string key; if the key matches a previous completed\n            state, that state result will be restored instead of running the task again.\n        cache_expiration: An optional amount of time indicating how long cached states\n            for this task should be restorable; if not provided, cached states will\n            never expire.\n        task_run_name: An optional name to distinguish runs of this task; this name can be provided\n            as a string template with the task's keyword arguments as variables,\n            or a function that returns a string.\n        retries: An optional number of times to retry on task run failure.\n        retry_delay_seconds: Optionally configures how long to wait before retrying the\n            task after failure. This is only applicable if `retries` is nonzero. This\n            setting can either be a number of seconds, a list of retry delays, or a\n            callable that, given the total number of retries, generates a list of retry\n            delays. If a number of seconds, that delay will be applied to all retries.\n            If a list, each retry will wait for the corresponding delay before retrying.\n            When passing a callable or a list, the number of configured retry delays\n            cannot exceed 50.\n        retry_jitter_factor: An optional factor that defines the factor to which a retry\n            can be jittered in order to avoid a \"thundering herd\".\n        persist_result: An optional toggle indicating whether the result of this task\n            should be persisted to result storage. Defaults to `None`, which indicates\n            that Prefect should choose whether the result should be persisted depending on\n            the features being used.\n        result_storage: An optional block to use to persist the result of this task.\n            Defaults to the value set in the flow the task is called in.\n        result_storage_key: An optional key to store the result in storage at when persisted.\n            Defaults to a unique identifier.\n        result_serializer: An optional serializer to use to serialize the result of this\n            task for persistence. Defaults to the value set in the flow the task is\n            called in.\n        timeout_seconds: An optional number of seconds indicating a maximum runtime for\n            the task. If the task exceeds this runtime, it will be marked as failed.\n        log_prints: If set, `print` statements in the task will be redirected to the\n            Prefect logger for the task run. Defaults to `None`, which indicates\n            that the value from the flow should be used.\n        refresh_cache: If set, cached results for the cache key are not used.\n            Defaults to `None`, which indicates that a cached result from a previous\n            execution with matching cache key is used.\n        on_failure: An optional list of callables to run when the task enters a failed state.\n        on_completion: An optional list of callables to run when the task enters a completed state.\n        retry_condition_fn: An optional callable run when a task run returns a Failed state. Should\n            return `True` if the task should continue to its retry policy (e.g. `retries=3`), and `False` if the task\n            should end as failed. Defaults to `None`, indicating the task should always continue\n            to its retry policy.\n        viz_return_value: An optional value to return when the task dependency tree is visualized.\n    \"\"\"\n\n    # NOTE: These parameters (types, defaults, and docstrings) should be duplicated\n    #       exactly in the @task decorator\n    def __init__(\n        self,\n        fn: Callable[P, R],\n        name: str = None,\n        description: str = None,\n        tags: Iterable[str] = None,\n        version: str = None,\n        cache_key_fn: Callable[\n            [\"TaskRunContext\", Dict[str, Any]], Optional[str]\n        ] = None,\n        cache_expiration: datetime.timedelta = None,\n        task_run_name: Optional[Union[Callable[[], str], str]] = None,\n        retries: Optional[int] = None,\n        retry_delay_seconds: Optional[\n            Union[\n                float,\n                int,\n                List[float],\n                Callable[[int], List[float]],\n            ]\n        ] = None,\n        retry_jitter_factor: Optional[float] = None,\n        persist_result: Optional[bool] = None,\n        result_storage: Optional[ResultStorage] = None,\n        result_serializer: Optional[ResultSerializer] = None,\n        result_storage_key: Optional[str] = None,\n        cache_result_in_memory: bool = True,\n        timeout_seconds: Union[int, float] = None,\n        log_prints: Optional[bool] = False,\n        refresh_cache: Optional[bool] = None,\n        on_completion: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n        on_failure: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n        retry_condition_fn: Optional[Callable[[\"Task\", TaskRun, State], bool]] = None,\n        viz_return_value: Optional[Any] = None,\n    ):\n        # Validate if hook passed is list and contains callables\n        hook_categories = [on_completion, on_failure]\n        hook_names = [\"on_completion\", \"on_failure\"]\n        for hooks, hook_name in zip(hook_categories, hook_names):\n            if hooks is not None:\n                if not hooks:\n                    raise ValueError(f\"Empty list passed for '{hook_name}'\")\n                try:\n                    hooks = list(hooks)\n                except TypeError:\n                    raise TypeError(\n                        f\"Expected iterable for '{hook_name}'; got\"\n                        f\" {type(hooks).__name__} instead. Please provide a list of\"\n                        f\" hooks to '{hook_name}':\\n\\n\"\n                        f\"@flow({hook_name}=[hook1, hook2])\\ndef\"\n                        \" my_flow():\\n\\tpass\"\n                    )\n\n                for hook in hooks:\n                    if not callable(hook):\n                        raise TypeError(\n                            f\"Expected callables in '{hook_name}'; got\"\n                            f\" {type(hook).__name__} instead. Please provide a list of\"\n                            f\" hooks to '{hook_name}':\\n\\n\"\n                            f\"@flow({hook_name}=[hook1, hook2])\\ndef\"\n                            \" my_flow():\\n\\tpass\"\n                        )\n\n        if not callable(fn):\n            raise TypeError(\"'fn' must be callable\")\n\n        self.description = description or inspect.getdoc(fn)\n        update_wrapper(self, fn)\n        self.fn = fn\n        self.isasync = inspect.iscoroutinefunction(self.fn)\n\n        if not name:\n            if not hasattr(self.fn, \"__name__\"):\n                self.name = type(self.fn).__name__\n            else:\n                self.name = self.fn.__name__\n        else:\n            self.name = name\n\n        if task_run_name is not None:\n            if not isinstance(task_run_name, str) and not callable(task_run_name):\n                raise TypeError(\n                    \"Expected string or callable for 'task_run_name'; got\"\n                    f\" {type(task_run_name).__name__} instead.\"\n                )\n        self.task_run_name = task_run_name\n\n        self.version = version\n        self.log_prints = log_prints\n\n        raise_for_reserved_arguments(self.fn, [\"return_state\", \"wait_for\"])\n\n        self.tags = set(tags if tags else [])\n\n        if not hasattr(self.fn, \"__qualname__\"):\n            self.task_key = to_qualified_name(type(self.fn))\n        else:\n            self.task_key = to_qualified_name(self.fn)\n\n        self.cache_key_fn = cache_key_fn\n        self.cache_expiration = cache_expiration\n        self.refresh_cache = refresh_cache\n\n        # TaskRunPolicy settings\n        # TODO: We can instantiate a `TaskRunPolicy` and add Pydantic bound checks to\n        #       validate that the user passes positive numbers here\n\n        self.retries = (\n            retries if retries is not None else PREFECT_TASK_DEFAULT_RETRIES.value()\n        )\n        if retry_delay_seconds is None:\n            retry_delay_seconds = PREFECT_TASK_DEFAULT_RETRY_DELAY_SECONDS.value()\n\n        if callable(retry_delay_seconds):\n            self.retry_delay_seconds = retry_delay_seconds(retries)\n        else:\n            self.retry_delay_seconds = retry_delay_seconds\n\n        if isinstance(self.retry_delay_seconds, list) and (\n            len(self.retry_delay_seconds) > 50\n        ):\n            raise ValueError(\"Can not configure more than 50 retry delays per task.\")\n\n        if retry_jitter_factor is not None and retry_jitter_factor < 0:\n            raise ValueError(\"`retry_jitter_factor` must be >= 0.\")\n\n        self.retry_jitter_factor = retry_jitter_factor\n        self.persist_result = persist_result\n        self.result_storage = result_storage\n        self.result_serializer = result_serializer\n        self.result_storage_key = result_storage_key\n        self.cache_result_in_memory = cache_result_in_memory\n        self.timeout_seconds = float(timeout_seconds) if timeout_seconds else None\n        # Warn if this task's `name` conflicts with another task while having a\n        # different function. This is to detect the case where two or more tasks\n        # share a name or are lambdas, which should result in a warning, and to\n        # differentiate it from the case where the task was 'copied' via\n        # `with_options`, which should not result in a warning.\n        registry = PrefectObjectRegistry.get()\n\n        if registry and any(\n            other\n            for other in registry.get_instances(Task)\n            if other.name == self.name and id(other.fn) != id(self.fn)\n        ):\n            try:\n                file = inspect.getsourcefile(self.fn)\n                line_number = inspect.getsourcelines(self.fn)[1]\n            except TypeError:\n                file = \"unknown\"\n                line_number = \"unknown\"\n\n            warnings.warn(\n                f\"A task named {self.name!r} and defined at '{file}:{line_number}' \"\n                \"conflicts with another task. Consider specifying a unique `name` \"\n                \"parameter in the task definition:\\n\\n \"\n                \"`@task(name='my_unique_name', ...)`\"\n            )\n        self.on_completion = on_completion\n        self.on_failure = on_failure\n\n        # retry_condition_fn must be a callable or None. If it is neither, raise a TypeError\n        if retry_condition_fn is not None and not (callable(retry_condition_fn)):\n            raise TypeError(\n                \"Expected `retry_condition_fn` to be callable, got\"\n                f\" {type(retry_condition_fn).__name__} instead.\"\n            )\n\n        self.retry_condition_fn = retry_condition_fn\n        self.viz_return_value = viz_return_value\n\n    def with_options(\n        self,\n        *,\n        name: str = None,\n        description: str = None,\n        tags: Iterable[str] = None,\n        cache_key_fn: Callable[\n            [\"TaskRunContext\", Dict[str, Any]], Optional[str]\n        ] = None,\n        task_run_name: Optional[Union[Callable[[], str], str]] = None,\n        cache_expiration: datetime.timedelta = None,\n        retries: Optional[int] = NotSet,\n        retry_delay_seconds: Union[\n            float,\n            int,\n            List[float],\n            Callable[[int], List[float]],\n        ] = NotSet,\n        retry_jitter_factor: Optional[float] = NotSet,\n        persist_result: Optional[bool] = NotSet,\n        result_storage: Optional[ResultStorage] = NotSet,\n        result_serializer: Optional[ResultSerializer] = NotSet,\n        result_storage_key: Optional[str] = NotSet,\n        cache_result_in_memory: Optional[bool] = None,\n        timeout_seconds: Union[int, float] = None,\n        log_prints: Optional[bool] = NotSet,\n        refresh_cache: Optional[bool] = NotSet,\n        on_completion: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n        on_failure: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n        retry_condition_fn: Optional[Callable[[\"Task\", TaskRun, State], bool]] = None,\n        viz_return_value: Optional[Any] = None,\n    ):\n        \"\"\"\n        Create a new task from the current object, updating provided options.\n\n        Args:\n            name: A new name for the task.\n            description: A new description for the task.\n            tags: A new set of tags for the task. If given, existing tags are ignored,\n                not merged.\n            cache_key_fn: A new cache key function for the task.\n            cache_expiration: A new cache expiration time for the task.\n            task_run_name: An optional name to distinguish runs of this task; this name can be provided\n                as a string template with the task's keyword arguments as variables,\n                or a function that returns a string.\n            retries: A new number of times to retry on task run failure.\n            retry_delay_seconds: Optionally configures how long to wait before retrying\n                the task after failure. This is only applicable if `retries` is nonzero.\n                This setting can either be a number of seconds, a list of retry delays,\n                or a callable that, given the total number of retries, generates a list\n                of retry delays. If a number of seconds, that delay will be applied to\n                all retries. If a list, each retry will wait for the corresponding delay\n                before retrying. When passing a callable or a list, the number of\n                configured retry delays cannot exceed 50.\n            retry_jitter_factor: An optional factor that defines the factor to which a\n                retry can be jittered in order to avoid a \"thundering herd\".\n            persist_result: A new option for enabling or disabling result persistence.\n            result_storage: A new storage type to use for results.\n            result_serializer: A new serializer to use for results.\n            result_storage_key: A new key for the persisted result to be stored at.\n            timeout_seconds: A new maximum time for the task to complete in seconds.\n            log_prints: A new option for enabling or disabling redirection of `print` statements.\n            refresh_cache: A new option for enabling or disabling cache refresh.\n            on_completion: A new list of callables to run when the task enters a completed state.\n            on_failure: A new list of callables to run when the task enters a failed state.\n            retry_condition_fn: An optional callable run when a task run returns a Failed state.\n                Should return `True` if the task should continue to its retry policy, and `False`\n                if the task should end as failed. Defaults to `None`, indicating the task should\n                always continue to its retry policy.\n            viz_return_value: An optional value to return when the task dependency tree is visualized.\n\n        Returns:\n            A new `Task` instance.\n\n        Examples:\n\n            Create a new task from an existing task and update the name\n\n            >>> @task(name=\"My task\")\n            >>> def my_task():\n            >>>     return 1\n            >>>\n            >>> new_task = my_task.with_options(name=\"My new task\")\n\n            Create a new task from an existing task and update the retry settings\n\n            >>> from random import randint\n            >>>\n            >>> @task(retries=1, retry_delay_seconds=5)\n            >>> def my_task():\n            >>>     x = randint(0, 5)\n            >>>     if x >= 3:  # Make a task that fails sometimes\n            >>>         raise ValueError(\"Retry me please!\")\n            >>>     return x\n            >>>\n            >>> new_task = my_task.with_options(retries=5, retry_delay_seconds=2)\n\n            Use a task with updated options within a flow\n\n            >>> @task(name=\"My task\")\n            >>> def my_task():\n            >>>     return 1\n            >>>\n            >>> @flow\n            >>> my_flow():\n            >>>     new_task = my_task.with_options(name=\"My new task\")\n            >>>     new_task()\n        \"\"\"\n        return Task(\n            fn=self.fn,\n            name=name or self.name,\n            description=description or self.description,\n            tags=tags or copy(self.tags),\n            cache_key_fn=cache_key_fn or self.cache_key_fn,\n            cache_expiration=cache_expiration or self.cache_expiration,\n            task_run_name=task_run_name,\n            retries=retries if retries is not NotSet else self.retries,\n            retry_delay_seconds=(\n                retry_delay_seconds\n                if retry_delay_seconds is not NotSet\n                else self.retry_delay_seconds\n            ),\n            retry_jitter_factor=(\n                retry_jitter_factor\n                if retry_jitter_factor is not NotSet\n                else self.retry_jitter_factor\n            ),\n            persist_result=(\n                persist_result if persist_result is not NotSet else self.persist_result\n            ),\n            result_storage=(\n                result_storage if result_storage is not NotSet else self.result_storage\n            ),\n            result_storage_key=(\n                result_storage_key\n                if result_storage_key is not NotSet\n                else self.result_storage_key\n            ),\n            result_serializer=(\n                result_serializer\n                if result_serializer is not NotSet\n                else self.result_serializer\n            ),\n            cache_result_in_memory=(\n                cache_result_in_memory\n                if cache_result_in_memory is not None\n                else self.cache_result_in_memory\n            ),\n            timeout_seconds=(\n                timeout_seconds if timeout_seconds is not None else self.timeout_seconds\n            ),\n            log_prints=(log_prints if log_prints is not NotSet else self.log_prints),\n            refresh_cache=(\n                refresh_cache if refresh_cache is not NotSet else self.refresh_cache\n            ),\n            on_completion=on_completion or self.on_completion,\n            on_failure=on_failure or self.on_failure,\n            retry_condition_fn=retry_condition_fn or self.retry_condition_fn,\n            viz_return_value=viz_return_value or self.viz_return_value,\n        )\n\n    @overload\n    def __call__(\n        self: \"Task[P, NoReturn]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> None:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def __call__(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> T:\n        ...\n\n    @overload\n    def __call__(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        return_state: Literal[True],\n        **kwargs: P.kwargs,\n    ) -> State[T]:\n        ...\n\n    def __call__(\n        self,\n        *args: P.args,\n        return_state: bool = False,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: P.kwargs,\n    ):\n        \"\"\"\n        Run the task and return the result. If `return_state` is True returns\n        the result is wrapped in a Prefect State which provides error handling.\n        \"\"\"\n        from prefect.engine import enter_task_run_engine\n        from prefect.task_runners import SequentialTaskRunner\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n\n        return_type = \"state\" if return_state else \"result\"\n\n        task_run_tracker = get_task_viz_tracker()\n        if task_run_tracker:\n            return track_viz_task(\n                self.isasync, self.name, parameters, self.viz_return_value\n            )\n\n        return enter_task_run_engine(\n            self,\n            parameters=parameters,\n            wait_for=wait_for,\n            task_runner=SequentialTaskRunner(),\n            return_type=return_type,\n            mapped=False,\n        )\n\n    @overload\n    def _run(\n        self: \"Task[P, NoReturn]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> PrefectFuture[None, Sync]:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def _run(\n        self: \"Task[P, Coroutine[Any, Any, T]]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> Awaitable[State[T]]:\n        ...\n\n    @overload\n    def _run(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> State[T]:\n        ...\n\n    def _run(\n        self,\n        *args: P.args,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: P.kwargs,\n    ) -> Union[State, Awaitable[State]]:\n        \"\"\"\n        Run the task and return the final state.\n        \"\"\"\n        from prefect.engine import enter_task_run_engine\n        from prefect.task_runners import SequentialTaskRunner\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n\n        return enter_task_run_engine(\n            self,\n            parameters=parameters,\n            wait_for=wait_for,\n            return_type=\"state\",\n            task_runner=SequentialTaskRunner(),\n            mapped=False,\n        )\n\n    @overload\n    def submit(\n        self: \"Task[P, NoReturn]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> PrefectFuture[None, Sync]:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def submit(\n        self: \"Task[P, Coroutine[Any, Any, T]]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> Awaitable[PrefectFuture[T, Async]]:\n        ...\n\n    @overload\n    def submit(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> PrefectFuture[T, Sync]:\n        ...\n\n    @overload\n    def submit(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        return_state: Literal[True],\n        **kwargs: P.kwargs,\n    ) -> State[T]:\n        ...\n\n    def submit(\n        self,\n        *args: Any,\n        return_state: bool = False,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: Any,\n    ) -> Union[PrefectFuture, Awaitable[PrefectFuture]]:\n        \"\"\"\n        Submit a run of the task to a worker.\n\n        Must be called within a flow function. If writing an async task, this call must\n        be awaited.\n\n        Will create a new task run in the backing API and submit the task to the flow's\n        task runner. This call only blocks execution while the task is being submitted,\n        once it is submitted, the flow function will continue executing. However, note\n        that the `SequentialTaskRunner` does not implement parallel execution for sync tasks\n        and they are fully resolved on submission.\n\n        Args:\n            *args: Arguments to run the task with\n            return_state: Return the result of the flow run wrapped in a\n                Prefect State.\n            wait_for: Upstream task futures to wait for before starting the task\n            **kwargs: Keyword arguments to run the task with\n\n        Returns:\n            If `return_state` is False a future allowing asynchronous access to\n                the state of the task\n            If `return_state` is True a future wrapped in a Prefect State allowing asynchronous access to\n                the state of the task\n\n        Examples:\n\n            Define a task\n\n            >>> from prefect import task\n            >>> @task\n            >>> def my_task():\n            >>>     return \"hello\"\n\n            Run a task in a flow\n\n            >>> from prefect import flow\n            >>> @flow\n            >>> def my_flow():\n            >>>     my_task.submit()\n\n            Wait for a task to finish\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     my_task.submit().wait()\n\n            Use the result from a task in a flow\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     print(my_task.submit().result())\n            >>>\n            >>> my_flow()\n            hello\n\n            Run an async task in an async flow\n\n            >>> @task\n            >>> async def my_async_task():\n            >>>     pass\n            >>>\n            >>> @flow\n            >>> async def my_flow():\n            >>>     await my_async_task.submit()\n\n            Run a sync task in an async flow\n\n            >>> @flow\n            >>> async def my_flow():\n            >>>     my_task.submit()\n\n            Enforce ordering between tasks that do not exchange data\n            >>> @task\n            >>> def task_1():\n            >>>     pass\n            >>>\n            >>> @task\n            >>> def task_2():\n            >>>     pass\n            >>>\n            >>> @flow\n            >>> def my_flow():\n            >>>     x = task_1.submit()\n            >>>\n            >>>     # task 2 will wait for task_1 to complete\n            >>>     y = task_2.submit(wait_for=[x])\n\n        \"\"\"\n\n        from prefect.engine import enter_task_run_engine\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n        return_type = \"state\" if return_state else \"future\"\n\n        task_viz_tracker = get_task_viz_tracker()\n        if task_viz_tracker:\n            raise VisualizationUnsupportedError(\n                \"`task.submit()` is not currently supported by `flow.visualize()`\"\n            )\n\n        return enter_task_run_engine(\n            self,\n            parameters=parameters,\n            wait_for=wait_for,\n            return_type=return_type,\n            task_runner=None,  # Use the flow's task runner\n            mapped=False,\n        )\n\n    @overload\n    def map(\n        self: \"Task[P, NoReturn]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> List[PrefectFuture[None, Sync]]:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def map(\n        self: \"Task[P, Coroutine[Any, Any, T]]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> Awaitable[List[PrefectFuture[T, Async]]]:\n        ...\n\n    @overload\n    def map(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> List[PrefectFuture[T, Sync]]:\n        ...\n\n    @overload\n    def map(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        return_state: Literal[True],\n        **kwargs: P.kwargs,\n    ) -> List[State[T]]:\n        ...\n\n    def map(\n        self,\n        *args: Any,\n        return_state: bool = False,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: Any,\n    ) -> Any:\n        \"\"\"\n        Submit a mapped run of the task to a worker.\n\n        Must be called within a flow function. If writing an async task, this\n        call must be awaited.\n\n        Must be called with at least one iterable and all iterables must be\n        the same length. Any arguments that are not iterable will be treated as\n        a static value and each task run will receive the same value.\n\n        Will create as many task runs as the length of the iterable(s) in the\n        backing API and submit the task runs to the flow's task runner. This\n        call blocks if given a future as input while the future is resolved. It\n        also blocks while the tasks are being submitted, once they are\n        submitted, the flow function will continue executing. However, note\n        that the `SequentialTaskRunner` does not implement parallel execution\n        for sync tasks and they are fully resolved on submission.\n\n        Args:\n            *args: Iterable and static arguments to run the tasks with\n            return_state: Return a list of Prefect States that wrap the results\n                of each task run.\n            wait_for: Upstream task futures to wait for before starting the\n                task\n            **kwargs: Keyword iterable arguments to run the task with\n\n        Returns:\n            A list of futures allowing asynchronous access to the state of the\n            tasks\n\n        Examples:\n\n            Define a task\n\n            >>> from prefect import task\n            >>> @task\n            >>> def my_task(x):\n            >>>     return x + 1\n\n            Create mapped tasks\n\n            >>> from prefect import flow\n            >>> @flow\n            >>> def my_flow():\n            >>>     my_task.map([1, 2, 3])\n\n            Wait for all mapped tasks to finish\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     futures = my_task.map([1, 2, 3])\n            >>>     for future in futures:\n            >>>         future.wait()\n            >>>     # Now all of the mapped tasks have finished\n            >>>     my_task(10)\n\n            Use the result from mapped tasks in a flow\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     futures = my_task.map([1, 2, 3])\n            >>>     for future in futures:\n            >>>         print(future.result())\n            >>> my_flow()\n            2\n            3\n            4\n\n            Enforce ordering between tasks that do not exchange data\n            >>> @task\n            >>> def task_1(x):\n            >>>     pass\n            >>>\n            >>> @task\n            >>> def task_2(y):\n            >>>     pass\n            >>>\n            >>> @flow\n            >>> def my_flow():\n            >>>     x = task_1.submit()\n            >>>\n            >>>     # task 2 will wait for task_1 to complete\n            >>>     y = task_2.map([1, 2, 3], wait_for=[x])\n\n            Use a non-iterable input as a constant across mapped tasks\n            >>> @task\n            >>> def display(prefix, item):\n            >>>    print(prefix, item)\n            >>>\n            >>> @flow\n            >>> def my_flow():\n            >>>     display.map(\"Check it out: \", [1, 2, 3])\n            >>>\n            >>> my_flow()\n            Check it out: 1\n            Check it out: 2\n            Check it out: 3\n\n            Use `unmapped` to treat an iterable argument as a constant\n            >>> from prefect import unmapped\n            >>>\n            >>> @task\n            >>> def add_n_to_items(items, n):\n            >>>     return [item + n for item in items]\n            >>>\n            >>> @flow\n            >>> def my_flow():\n            >>>     return add_n_to_items.map(unmapped([10, 20]), n=[1, 2, 3])\n            >>>\n            >>> my_flow()\n            [[11, 21], [12, 22], [13, 23]]\n        \"\"\"\n\n        from prefect.engine import enter_task_run_engine\n\n        # Convert the call args/kwargs to a parameter dict; do not apply defaults\n        # since they should not be mapped over\n        parameters = get_call_parameters(self.fn, args, kwargs, apply_defaults=False)\n        return_type = \"state\" if return_state else \"future\"\n\n        task_viz_tracker = get_task_viz_tracker()\n        if task_viz_tracker:\n            raise VisualizationUnsupportedError(\n                \"`task.map()` is not currently supported by `flow.visualize()`\"\n            )\n\n        return enter_task_run_engine(\n            self,\n            parameters=parameters,\n            wait_for=wait_for,\n            return_type=return_type,\n            task_runner=None,\n            mapped=True,\n        )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.Task.map","title":"map","text":"

    Submit a mapped run of the task to a worker.

    Must be called within a flow function. If writing an async task, this call must be awaited.

    Must be called with at least one iterable and all iterables must be the same length. Any arguments that are not iterable will be treated as a static value and each task run will receive the same value.

    Will create as many task runs as the length of the iterable(s) in the backing API and submit the task runs to the flow's task runner. This call blocks if given a future as input while the future is resolved. It also blocks while the tasks are being submitted, once they are submitted, the flow function will continue executing. However, note that the SequentialTaskRunner does not implement parallel execution for sync tasks and they are fully resolved on submission.

    Parameters:

    Name Type Description Default *args Any

    Iterable and static arguments to run the tasks with

    () return_state bool

    Return a list of Prefect States that wrap the results of each task run.

    False wait_for Optional[Iterable[PrefectFuture]]

    Upstream task futures to wait for before starting the task

    None **kwargs Any

    Keyword iterable arguments to run the task with

    {}

    Returns:

    Type Description Any

    A list of futures allowing asynchronous access to the state of the

    Any

    tasks

    Define a task\n\n>>> from prefect import task\n>>> @task\n>>> def my_task(x):\n>>>     return x + 1\n\nCreate mapped tasks\n\n>>> from prefect import flow\n>>> @flow\n>>> def my_flow():\n>>>     my_task.map([1, 2, 3])\n\nWait for all mapped tasks to finish\n\n>>> @flow\n>>> def my_flow():\n>>>     futures = my_task.map([1, 2, 3])\n>>>     for future in futures:\n>>>         future.wait()\n>>>     # Now all of the mapped tasks have finished\n>>>     my_task(10)\n\nUse the result from mapped tasks in a flow\n\n>>> @flow\n>>> def my_flow():\n>>>     futures = my_task.map([1, 2, 3])\n>>>     for future in futures:\n>>>         print(future.result())\n>>> my_flow()\n2\n3\n4\n\nEnforce ordering between tasks that do not exchange data\n>>> @task\n>>> def task_1(x):\n>>>     pass\n>>>\n>>> @task\n>>> def task_2(y):\n>>>     pass\n>>>\n>>> @flow\n>>> def my_flow():\n>>>     x = task_1.submit()\n>>>\n>>>     # task 2 will wait for task_1 to complete\n>>>     y = task_2.map([1, 2, 3], wait_for=[x])\n\nUse a non-iterable input as a constant across mapped tasks\n>>> @task\n>>> def display(prefix, item):\n>>>    print(prefix, item)\n>>>\n>>> @flow\n>>> def my_flow():\n>>>     display.map(\"Check it out: \", [1, 2, 3])\n>>>\n>>> my_flow()\nCheck it out: 1\nCheck it out: 2\nCheck it out: 3\n\nUse `unmapped` to treat an iterable argument as a constant\n>>> from prefect import unmapped\n>>>\n>>> @task\n>>> def add_n_to_items(items, n):\n>>>     return [item + n for item in items]\n>>>\n>>> @flow\n>>> def my_flow():\n>>>     return add_n_to_items.map(unmapped([10, 20]), n=[1, 2, 3])\n>>>\n>>> my_flow()\n[[11, 21], [12, 22], [13, 23]]\n
    Source code in prefect/tasks.py
    def map(\n    self,\n    *args: Any,\n    return_state: bool = False,\n    wait_for: Optional[Iterable[PrefectFuture]] = None,\n    **kwargs: Any,\n) -> Any:\n    \"\"\"\n    Submit a mapped run of the task to a worker.\n\n    Must be called within a flow function. If writing an async task, this\n    call must be awaited.\n\n    Must be called with at least one iterable and all iterables must be\n    the same length. Any arguments that are not iterable will be treated as\n    a static value and each task run will receive the same value.\n\n    Will create as many task runs as the length of the iterable(s) in the\n    backing API and submit the task runs to the flow's task runner. This\n    call blocks if given a future as input while the future is resolved. It\n    also blocks while the tasks are being submitted, once they are\n    submitted, the flow function will continue executing. However, note\n    that the `SequentialTaskRunner` does not implement parallel execution\n    for sync tasks and they are fully resolved on submission.\n\n    Args:\n        *args: Iterable and static arguments to run the tasks with\n        return_state: Return a list of Prefect States that wrap the results\n            of each task run.\n        wait_for: Upstream task futures to wait for before starting the\n            task\n        **kwargs: Keyword iterable arguments to run the task with\n\n    Returns:\n        A list of futures allowing asynchronous access to the state of the\n        tasks\n\n    Examples:\n\n        Define a task\n\n        >>> from prefect import task\n        >>> @task\n        >>> def my_task(x):\n        >>>     return x + 1\n\n        Create mapped tasks\n\n        >>> from prefect import flow\n        >>> @flow\n        >>> def my_flow():\n        >>>     my_task.map([1, 2, 3])\n\n        Wait for all mapped tasks to finish\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     futures = my_task.map([1, 2, 3])\n        >>>     for future in futures:\n        >>>         future.wait()\n        >>>     # Now all of the mapped tasks have finished\n        >>>     my_task(10)\n\n        Use the result from mapped tasks in a flow\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     futures = my_task.map([1, 2, 3])\n        >>>     for future in futures:\n        >>>         print(future.result())\n        >>> my_flow()\n        2\n        3\n        4\n\n        Enforce ordering between tasks that do not exchange data\n        >>> @task\n        >>> def task_1(x):\n        >>>     pass\n        >>>\n        >>> @task\n        >>> def task_2(y):\n        >>>     pass\n        >>>\n        >>> @flow\n        >>> def my_flow():\n        >>>     x = task_1.submit()\n        >>>\n        >>>     # task 2 will wait for task_1 to complete\n        >>>     y = task_2.map([1, 2, 3], wait_for=[x])\n\n        Use a non-iterable input as a constant across mapped tasks\n        >>> @task\n        >>> def display(prefix, item):\n        >>>    print(prefix, item)\n        >>>\n        >>> @flow\n        >>> def my_flow():\n        >>>     display.map(\"Check it out: \", [1, 2, 3])\n        >>>\n        >>> my_flow()\n        Check it out: 1\n        Check it out: 2\n        Check it out: 3\n\n        Use `unmapped` to treat an iterable argument as a constant\n        >>> from prefect import unmapped\n        >>>\n        >>> @task\n        >>> def add_n_to_items(items, n):\n        >>>     return [item + n for item in items]\n        >>>\n        >>> @flow\n        >>> def my_flow():\n        >>>     return add_n_to_items.map(unmapped([10, 20]), n=[1, 2, 3])\n        >>>\n        >>> my_flow()\n        [[11, 21], [12, 22], [13, 23]]\n    \"\"\"\n\n    from prefect.engine import enter_task_run_engine\n\n    # Convert the call args/kwargs to a parameter dict; do not apply defaults\n    # since they should not be mapped over\n    parameters = get_call_parameters(self.fn, args, kwargs, apply_defaults=False)\n    return_type = \"state\" if return_state else \"future\"\n\n    task_viz_tracker = get_task_viz_tracker()\n    if task_viz_tracker:\n        raise VisualizationUnsupportedError(\n            \"`task.map()` is not currently supported by `flow.visualize()`\"\n        )\n\n    return enter_task_run_engine(\n        self,\n        parameters=parameters,\n        wait_for=wait_for,\n        return_type=return_type,\n        task_runner=None,\n        mapped=True,\n    )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.Task.submit","title":"submit","text":"

    Submit a run of the task to a worker.

    Must be called within a flow function. If writing an async task, this call must be awaited.

    Will create a new task run in the backing API and submit the task to the flow's task runner. This call only blocks execution while the task is being submitted, once it is submitted, the flow function will continue executing. However, note that the SequentialTaskRunner does not implement parallel execution for sync tasks and they are fully resolved on submission.

    Parameters:

    Name Type Description Default *args Any

    Arguments to run the task with

    () return_state bool

    Return the result of the flow run wrapped in a Prefect State.

    False wait_for Optional[Iterable[PrefectFuture]]

    Upstream task futures to wait for before starting the task

    None **kwargs Any

    Keyword arguments to run the task with

    {}

    Returns:

    Type Description Union[PrefectFuture, Awaitable[PrefectFuture]]

    If return_state is False a future allowing asynchronous access to the state of the task

    Union[PrefectFuture, Awaitable[PrefectFuture]]

    If return_state is True a future wrapped in a Prefect State allowing asynchronous access to the state of the task

    Define a task\n\n>>> from prefect import task\n>>> @task\n>>> def my_task():\n>>>     return \"hello\"\n\nRun a task in a flow\n\n>>> from prefect import flow\n>>> @flow\n>>> def my_flow():\n>>>     my_task.submit()\n\nWait for a task to finish\n\n>>> @flow\n>>> def my_flow():\n>>>     my_task.submit().wait()\n\nUse the result from a task in a flow\n\n>>> @flow\n>>> def my_flow():\n>>>     print(my_task.submit().result())\n>>>\n>>> my_flow()\nhello\n\nRun an async task in an async flow\n\n>>> @task\n>>> async def my_async_task():\n>>>     pass\n>>>\n>>> @flow\n>>> async def my_flow():\n>>>     await my_async_task.submit()\n\nRun a sync task in an async flow\n\n>>> @flow\n>>> async def my_flow():\n>>>     my_task.submit()\n\nEnforce ordering between tasks that do not exchange data\n>>> @task\n>>> def task_1():\n>>>     pass\n>>>\n>>> @task\n>>> def task_2():\n>>>     pass\n>>>\n>>> @flow\n>>> def my_flow():\n>>>     x = task_1.submit()\n>>>\n>>>     # task 2 will wait for task_1 to complete\n>>>     y = task_2.submit(wait_for=[x])\n
    Source code in prefect/tasks.py
    def submit(\n    self,\n    *args: Any,\n    return_state: bool = False,\n    wait_for: Optional[Iterable[PrefectFuture]] = None,\n    **kwargs: Any,\n) -> Union[PrefectFuture, Awaitable[PrefectFuture]]:\n    \"\"\"\n    Submit a run of the task to a worker.\n\n    Must be called within a flow function. If writing an async task, this call must\n    be awaited.\n\n    Will create a new task run in the backing API and submit the task to the flow's\n    task runner. This call only blocks execution while the task is being submitted,\n    once it is submitted, the flow function will continue executing. However, note\n    that the `SequentialTaskRunner` does not implement parallel execution for sync tasks\n    and they are fully resolved on submission.\n\n    Args:\n        *args: Arguments to run the task with\n        return_state: Return the result of the flow run wrapped in a\n            Prefect State.\n        wait_for: Upstream task futures to wait for before starting the task\n        **kwargs: Keyword arguments to run the task with\n\n    Returns:\n        If `return_state` is False a future allowing asynchronous access to\n            the state of the task\n        If `return_state` is True a future wrapped in a Prefect State allowing asynchronous access to\n            the state of the task\n\n    Examples:\n\n        Define a task\n\n        >>> from prefect import task\n        >>> @task\n        >>> def my_task():\n        >>>     return \"hello\"\n\n        Run a task in a flow\n\n        >>> from prefect import flow\n        >>> @flow\n        >>> def my_flow():\n        >>>     my_task.submit()\n\n        Wait for a task to finish\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     my_task.submit().wait()\n\n        Use the result from a task in a flow\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     print(my_task.submit().result())\n        >>>\n        >>> my_flow()\n        hello\n\n        Run an async task in an async flow\n\n        >>> @task\n        >>> async def my_async_task():\n        >>>     pass\n        >>>\n        >>> @flow\n        >>> async def my_flow():\n        >>>     await my_async_task.submit()\n\n        Run a sync task in an async flow\n\n        >>> @flow\n        >>> async def my_flow():\n        >>>     my_task.submit()\n\n        Enforce ordering between tasks that do not exchange data\n        >>> @task\n        >>> def task_1():\n        >>>     pass\n        >>>\n        >>> @task\n        >>> def task_2():\n        >>>     pass\n        >>>\n        >>> @flow\n        >>> def my_flow():\n        >>>     x = task_1.submit()\n        >>>\n        >>>     # task 2 will wait for task_1 to complete\n        >>>     y = task_2.submit(wait_for=[x])\n\n    \"\"\"\n\n    from prefect.engine import enter_task_run_engine\n\n    # Convert the call args/kwargs to a parameter dict\n    parameters = get_call_parameters(self.fn, args, kwargs)\n    return_type = \"state\" if return_state else \"future\"\n\n    task_viz_tracker = get_task_viz_tracker()\n    if task_viz_tracker:\n        raise VisualizationUnsupportedError(\n            \"`task.submit()` is not currently supported by `flow.visualize()`\"\n        )\n\n    return enter_task_run_engine(\n        self,\n        parameters=parameters,\n        wait_for=wait_for,\n        return_type=return_type,\n        task_runner=None,  # Use the flow's task runner\n        mapped=False,\n    )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.Task.with_options","title":"with_options","text":"

    Create a new task from the current object, updating provided options.

    Parameters:

    Name Type Description Default name str

    A new name for the task.

    None description str

    A new description for the task.

    None tags Iterable[str]

    A new set of tags for the task. If given, existing tags are ignored, not merged.

    None cache_key_fn Callable[[TaskRunContext, Dict[str, Any]], Optional[str]]

    A new cache key function for the task.

    None cache_expiration timedelta

    A new cache expiration time for the task.

    None task_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this task; this name can be provided as a string template with the task's keyword arguments as variables, or a function that returns a string.

    None retries Optional[int]

    A new number of times to retry on task run failure.

    NotSet retry_delay_seconds Union[float, int, List[float], Callable[[int], List[float]]]

    Optionally configures how long to wait before retrying the task after failure. This is only applicable if retries is nonzero. This setting can either be a number of seconds, a list of retry delays, or a callable that, given the total number of retries, generates a list of retry delays. If a number of seconds, that delay will be applied to all retries. If a list, each retry will wait for the corresponding delay before retrying. When passing a callable or a list, the number of configured retry delays cannot exceed 50.

    NotSet retry_jitter_factor Optional[float]

    An optional factor that defines the factor to which a retry can be jittered in order to avoid a \"thundering herd\".

    NotSet persist_result Optional[bool]

    A new option for enabling or disabling result persistence.

    NotSet result_storage Optional[ResultStorage]

    A new storage type to use for results.

    NotSet result_serializer Optional[ResultSerializer]

    A new serializer to use for results.

    NotSet result_storage_key Optional[str]

    A new key for the persisted result to be stored at.

    NotSet timeout_seconds Union[int, float]

    A new maximum time for the task to complete in seconds.

    None log_prints Optional[bool]

    A new option for enabling or disabling redirection of print statements.

    NotSet refresh_cache Optional[bool]

    A new option for enabling or disabling cache refresh.

    NotSet on_completion Optional[List[Callable[[Task, TaskRun, State], None]]]

    A new list of callables to run when the task enters a completed state.

    None on_failure Optional[List[Callable[[Task, TaskRun, State], None]]]

    A new list of callables to run when the task enters a failed state.

    None retry_condition_fn Optional[Callable[[Task, TaskRun, State], bool]]

    An optional callable run when a task run returns a Failed state. Should return True if the task should continue to its retry policy, and False if the task should end as failed. Defaults to None, indicating the task should always continue to its retry policy.

    None viz_return_value Optional[Any]

    An optional value to return when the task dependency tree is visualized.

    None

    Returns:

    Type Description

    A new Task instance.

    Create a new task from an existing task and update the name\n\n>>> @task(name=\"My task\")\n>>> def my_task():\n>>>     return 1\n>>>\n>>> new_task = my_task.with_options(name=\"My new task\")\n\nCreate a new task from an existing task and update the retry settings\n\n>>> from random import randint\n>>>\n>>> @task(retries=1, retry_delay_seconds=5)\n>>> def my_task():\n>>>     x = randint(0, 5)\n>>>     if x >= 3:  # Make a task that fails sometimes\n>>>         raise ValueError(\"Retry me please!\")\n>>>     return x\n>>>\n>>> new_task = my_task.with_options(retries=5, retry_delay_seconds=2)\n\nUse a task with updated options within a flow\n\n>>> @task(name=\"My task\")\n>>> def my_task():\n>>>     return 1\n>>>\n>>> @flow\n>>> my_flow():\n>>>     new_task = my_task.with_options(name=\"My new task\")\n>>>     new_task()\n
    Source code in prefect/tasks.py
    def with_options(\n    self,\n    *,\n    name: str = None,\n    description: str = None,\n    tags: Iterable[str] = None,\n    cache_key_fn: Callable[\n        [\"TaskRunContext\", Dict[str, Any]], Optional[str]\n    ] = None,\n    task_run_name: Optional[Union[Callable[[], str], str]] = None,\n    cache_expiration: datetime.timedelta = None,\n    retries: Optional[int] = NotSet,\n    retry_delay_seconds: Union[\n        float,\n        int,\n        List[float],\n        Callable[[int], List[float]],\n    ] = NotSet,\n    retry_jitter_factor: Optional[float] = NotSet,\n    persist_result: Optional[bool] = NotSet,\n    result_storage: Optional[ResultStorage] = NotSet,\n    result_serializer: Optional[ResultSerializer] = NotSet,\n    result_storage_key: Optional[str] = NotSet,\n    cache_result_in_memory: Optional[bool] = None,\n    timeout_seconds: Union[int, float] = None,\n    log_prints: Optional[bool] = NotSet,\n    refresh_cache: Optional[bool] = NotSet,\n    on_completion: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n    on_failure: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n    retry_condition_fn: Optional[Callable[[\"Task\", TaskRun, State], bool]] = None,\n    viz_return_value: Optional[Any] = None,\n):\n    \"\"\"\n    Create a new task from the current object, updating provided options.\n\n    Args:\n        name: A new name for the task.\n        description: A new description for the task.\n        tags: A new set of tags for the task. If given, existing tags are ignored,\n            not merged.\n        cache_key_fn: A new cache key function for the task.\n        cache_expiration: A new cache expiration time for the task.\n        task_run_name: An optional name to distinguish runs of this task; this name can be provided\n            as a string template with the task's keyword arguments as variables,\n            or a function that returns a string.\n        retries: A new number of times to retry on task run failure.\n        retry_delay_seconds: Optionally configures how long to wait before retrying\n            the task after failure. This is only applicable if `retries` is nonzero.\n            This setting can either be a number of seconds, a list of retry delays,\n            or a callable that, given the total number of retries, generates a list\n            of retry delays. If a number of seconds, that delay will be applied to\n            all retries. If a list, each retry will wait for the corresponding delay\n            before retrying. When passing a callable or a list, the number of\n            configured retry delays cannot exceed 50.\n        retry_jitter_factor: An optional factor that defines the factor to which a\n            retry can be jittered in order to avoid a \"thundering herd\".\n        persist_result: A new option for enabling or disabling result persistence.\n        result_storage: A new storage type to use for results.\n        result_serializer: A new serializer to use for results.\n        result_storage_key: A new key for the persisted result to be stored at.\n        timeout_seconds: A new maximum time for the task to complete in seconds.\n        log_prints: A new option for enabling or disabling redirection of `print` statements.\n        refresh_cache: A new option for enabling or disabling cache refresh.\n        on_completion: A new list of callables to run when the task enters a completed state.\n        on_failure: A new list of callables to run when the task enters a failed state.\n        retry_condition_fn: An optional callable run when a task run returns a Failed state.\n            Should return `True` if the task should continue to its retry policy, and `False`\n            if the task should end as failed. Defaults to `None`, indicating the task should\n            always continue to its retry policy.\n        viz_return_value: An optional value to return when the task dependency tree is visualized.\n\n    Returns:\n        A new `Task` instance.\n\n    Examples:\n\n        Create a new task from an existing task and update the name\n\n        >>> @task(name=\"My task\")\n        >>> def my_task():\n        >>>     return 1\n        >>>\n        >>> new_task = my_task.with_options(name=\"My new task\")\n\n        Create a new task from an existing task and update the retry settings\n\n        >>> from random import randint\n        >>>\n        >>> @task(retries=1, retry_delay_seconds=5)\n        >>> def my_task():\n        >>>     x = randint(0, 5)\n        >>>     if x >= 3:  # Make a task that fails sometimes\n        >>>         raise ValueError(\"Retry me please!\")\n        >>>     return x\n        >>>\n        >>> new_task = my_task.with_options(retries=5, retry_delay_seconds=2)\n\n        Use a task with updated options within a flow\n\n        >>> @task(name=\"My task\")\n        >>> def my_task():\n        >>>     return 1\n        >>>\n        >>> @flow\n        >>> my_flow():\n        >>>     new_task = my_task.with_options(name=\"My new task\")\n        >>>     new_task()\n    \"\"\"\n    return Task(\n        fn=self.fn,\n        name=name or self.name,\n        description=description or self.description,\n        tags=tags or copy(self.tags),\n        cache_key_fn=cache_key_fn or self.cache_key_fn,\n        cache_expiration=cache_expiration or self.cache_expiration,\n        task_run_name=task_run_name,\n        retries=retries if retries is not NotSet else self.retries,\n        retry_delay_seconds=(\n            retry_delay_seconds\n            if retry_delay_seconds is not NotSet\n            else self.retry_delay_seconds\n        ),\n        retry_jitter_factor=(\n            retry_jitter_factor\n            if retry_jitter_factor is not NotSet\n            else self.retry_jitter_factor\n        ),\n        persist_result=(\n            persist_result if persist_result is not NotSet else self.persist_result\n        ),\n        result_storage=(\n            result_storage if result_storage is not NotSet else self.result_storage\n        ),\n        result_storage_key=(\n            result_storage_key\n            if result_storage_key is not NotSet\n            else self.result_storage_key\n        ),\n        result_serializer=(\n            result_serializer\n            if result_serializer is not NotSet\n            else self.result_serializer\n        ),\n        cache_result_in_memory=(\n            cache_result_in_memory\n            if cache_result_in_memory is not None\n            else self.cache_result_in_memory\n        ),\n        timeout_seconds=(\n            timeout_seconds if timeout_seconds is not None else self.timeout_seconds\n        ),\n        log_prints=(log_prints if log_prints is not NotSet else self.log_prints),\n        refresh_cache=(\n            refresh_cache if refresh_cache is not NotSet else self.refresh_cache\n        ),\n        on_completion=on_completion or self.on_completion,\n        on_failure=on_failure or self.on_failure,\n        retry_condition_fn=retry_condition_fn or self.retry_condition_fn,\n        viz_return_value=viz_return_value or self.viz_return_value,\n    )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.exponential_backoff","title":"exponential_backoff","text":"

    A task retry backoff utility that configures exponential backoff for task retries. The exponential backoff design matches the urllib3 implementation.

    Parameters:

    Name Type Description Default backoff_factor float

    the base delay for the first retry, subsequent retries will increase the delay time by powers of 2.

    required

    Returns:

    Type Description Callable[[int], List[float]]

    a callable that can be passed to the task constructor

    Source code in prefect/tasks.py
    def exponential_backoff(backoff_factor: float) -> Callable[[int], List[float]]:\n    \"\"\"\n    A task retry backoff utility that configures exponential backoff for task retries.\n    The exponential backoff design matches the urllib3 implementation.\n\n    Arguments:\n        backoff_factor: the base delay for the first retry, subsequent retries will\n            increase the delay time by powers of 2.\n\n    Returns:\n        a callable that can be passed to the task constructor\n    \"\"\"\n\n    def retry_backoff_callable(retries: int) -> List[float]:\n        # no more than 50 retry delays can be configured on a task\n        retries = min(retries, 50)\n\n        return [backoff_factor * max(0, 2**r) for r in range(retries)]\n\n    return retry_backoff_callable\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.task","title":"task","text":"

    Decorator to designate a function as a task in a Prefect workflow.

    This decorator may be used for asynchronous or synchronous functions.

    Parameters:

    Name Type Description Default name str

    An optional name for the task; if not provided, the name will be inferred from the given function.

    None description str

    An optional string description for the task.

    None tags Iterable[str]

    An optional set of tags to be associated with runs of this task. These tags are combined with any tags defined by a prefect.tags context at task runtime.

    None version str

    An optional string specifying the version of this task definition

    None cache_key_fn Callable[[TaskRunContext, Dict[str, Any]], Optional[str]]

    An optional callable that, given the task run context and call parameters, generates a string key; if the key matches a previous completed state, that state result will be restored instead of running the task again.

    None cache_expiration timedelta

    An optional amount of time indicating how long cached states for this task should be restorable; if not provided, cached states will never expire.

    None task_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this task; this name can be provided as a string template with the task's keyword arguments as variables, or a function that returns a string.

    None retries int

    An optional number of times to retry on task run failure

    None retry_delay_seconds Union[float, int, List[float], Callable[[int], List[float]]]

    Optionally configures how long to wait before retrying the task after failure. This is only applicable if retries is nonzero. This setting can either be a number of seconds, a list of retry delays, or a callable that, given the total number of retries, generates a list of retry delays. If a number of seconds, that delay will be applied to all retries. If a list, each retry will wait for the corresponding delay before retrying. When passing a callable or a list, the number of configured retry delays cannot exceed 50.

    None retry_jitter_factor Optional[float]

    An optional factor that defines the factor to which a retry can be jittered in order to avoid a \"thundering herd\".

    None persist_result Optional[bool]

    An optional toggle indicating whether the result of this task should be persisted to result storage. Defaults to None, which indicates that Prefect should choose whether the result should be persisted depending on the features being used.

    None result_storage Optional[ResultStorage]

    An optional block to use to persist the result of this task. Defaults to the value set in the flow the task is called in.

    None result_storage_key Optional[str]

    An optional key to store the result in storage at when persisted. Defaults to a unique identifier.

    None result_serializer Optional[ResultSerializer]

    An optional serializer to use to serialize the result of this task for persistence. Defaults to the value set in the flow the task is called in.

    None timeout_seconds Union[int, float]

    An optional number of seconds indicating a maximum runtime for the task. If the task exceeds this runtime, it will be marked as failed.

    None log_prints Optional[bool]

    If set, print statements in the task will be redirected to the Prefect logger for the task run. Defaults to None, which indicates that the value from the flow should be used.

    None refresh_cache Optional[bool]

    If set, cached results for the cache key are not used. Defaults to None, which indicates that a cached result from a previous execution with matching cache key is used.

    None on_failure Optional[List[Callable[[Task, TaskRun, State], None]]]

    An optional list of callables to run when the task enters a failed state.

    None on_completion Optional[List[Callable[[Task, TaskRun, State], None]]]

    An optional list of callables to run when the task enters a completed state.

    None retry_condition_fn Optional[Callable[[Task, TaskRun, State], bool]]

    An optional callable run when a task run returns a Failed state. Should return True if the task should continue to its retry policy (e.g. retries=3), and False if the task should end as failed. Defaults to None, indicating the task should always continue to its retry policy.

    None viz_return_value Any

    An optional value to return when the task dependency tree is visualized.

    None

    Returns:

    Type Description

    A callable Task object which, when called, will submit the task for execution.

    Examples:

    Define a simple task

    >>> @task\n>>> def add(x, y):\n>>>     return x + y\n

    Define an async task

    >>> @task\n>>> async def add(x, y):\n>>>     return x + y\n

    Define a task with tags and a description

    >>> @task(tags={\"a\", \"b\"}, description=\"This task is empty but its my first!\")\n>>> def my_task():\n>>>     pass\n

    Define a task with a custom name

    >>> @task(name=\"The Ultimate Task\")\n>>> def my_task():\n>>>     pass\n

    Define a task that retries 3 times with a 5 second delay between attempts

    >>> from random import randint\n>>>\n>>> @task(retries=3, retry_delay_seconds=5)\n>>> def my_task():\n>>>     x = randint(0, 5)\n>>>     if x >= 3:  # Make a task that fails sometimes\n>>>         raise ValueError(\"Retry me please!\")\n>>>     return x\n

    Define a task that is cached for a day based on its inputs

    >>> from prefect.tasks import task_input_hash\n>>> from datetime import timedelta\n>>>\n>>> @task(cache_key_fn=task_input_hash, cache_expiration=timedelta(days=1))\n>>> def my_task():\n>>>     return \"hello\"\n
    Source code in prefect/tasks.py
    def task(\n    __fn=None,\n    *,\n    name: str = None,\n    description: str = None,\n    tags: Iterable[str] = None,\n    version: str = None,\n    cache_key_fn: Callable[[\"TaskRunContext\", Dict[str, Any]], Optional[str]] = None,\n    cache_expiration: datetime.timedelta = None,\n    task_run_name: Optional[Union[Callable[[], str], str]] = None,\n    retries: int = None,\n    retry_delay_seconds: Union[\n        float,\n        int,\n        List[float],\n        Callable[[int], List[float]],\n    ] = None,\n    retry_jitter_factor: Optional[float] = None,\n    persist_result: Optional[bool] = None,\n    result_storage: Optional[ResultStorage] = None,\n    result_storage_key: Optional[str] = None,\n    result_serializer: Optional[ResultSerializer] = None,\n    cache_result_in_memory: bool = True,\n    timeout_seconds: Union[int, float] = None,\n    log_prints: Optional[bool] = None,\n    refresh_cache: Optional[bool] = None,\n    on_completion: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n    on_failure: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n    retry_condition_fn: Optional[Callable[[\"Task\", TaskRun, State], bool]] = None,\n    viz_return_value: Any = None,\n):\n    \"\"\"\n    Decorator to designate a function as a task in a Prefect workflow.\n\n    This decorator may be used for asynchronous or synchronous functions.\n\n    Args:\n        name: An optional name for the task; if not provided, the name will be inferred\n            from the given function.\n        description: An optional string description for the task.\n        tags: An optional set of tags to be associated with runs of this task. These\n            tags are combined with any tags defined by a `prefect.tags` context at\n            task runtime.\n        version: An optional string specifying the version of this task definition\n        cache_key_fn: An optional callable that, given the task run context and call\n            parameters, generates a string key; if the key matches a previous completed\n            state, that state result will be restored instead of running the task again.\n        cache_expiration: An optional amount of time indicating how long cached states\n            for this task should be restorable; if not provided, cached states will\n            never expire.\n        task_run_name: An optional name to distinguish runs of this task; this name can be provided\n            as a string template with the task's keyword arguments as variables,\n            or a function that returns a string.\n        retries: An optional number of times to retry on task run failure\n        retry_delay_seconds: Optionally configures how long to wait before retrying the\n            task after failure. This is only applicable if `retries` is nonzero. This\n            setting can either be a number of seconds, a list of retry delays, or a\n            callable that, given the total number of retries, generates a list of retry\n            delays. If a number of seconds, that delay will be applied to all retries.\n            If a list, each retry will wait for the corresponding delay before retrying.\n            When passing a callable or a list, the number of configured retry delays\n            cannot exceed 50.\n        retry_jitter_factor: An optional factor that defines the factor to which a retry\n            can be jittered in order to avoid a \"thundering herd\".\n        persist_result: An optional toggle indicating whether the result of this task\n            should be persisted to result storage. Defaults to `None`, which indicates\n            that Prefect should choose whether the result should be persisted depending on\n            the features being used.\n        result_storage: An optional block to use to persist the result of this task.\n            Defaults to the value set in the flow the task is called in.\n        result_storage_key: An optional key to store the result in storage at when persisted.\n            Defaults to a unique identifier.\n        result_serializer: An optional serializer to use to serialize the result of this\n            task for persistence. Defaults to the value set in the flow the task is\n            called in.\n        timeout_seconds: An optional number of seconds indicating a maximum runtime for\n            the task. If the task exceeds this runtime, it will be marked as failed.\n        log_prints: If set, `print` statements in the task will be redirected to the\n            Prefect logger for the task run. Defaults to `None`, which indicates\n            that the value from the flow should be used.\n        refresh_cache: If set, cached results for the cache key are not used.\n            Defaults to `None`, which indicates that a cached result from a previous\n            execution with matching cache key is used.\n        on_failure: An optional list of callables to run when the task enters a failed state.\n        on_completion: An optional list of callables to run when the task enters a completed state.\n        retry_condition_fn: An optional callable run when a task run returns a Failed state. Should\n            return `True` if the task should continue to its retry policy (e.g. `retries=3`), and `False` if the task\n            should end as failed. Defaults to `None`, indicating the task should always continue\n            to its retry policy.\n        viz_return_value: An optional value to return when the task dependency tree is visualized.\n\n    Returns:\n        A callable `Task` object which, when called, will submit the task for execution.\n\n    Examples:\n        Define a simple task\n\n        >>> @task\n        >>> def add(x, y):\n        >>>     return x + y\n\n        Define an async task\n\n        >>> @task\n        >>> async def add(x, y):\n        >>>     return x + y\n\n        Define a task with tags and a description\n\n        >>> @task(tags={\"a\", \"b\"}, description=\"This task is empty but its my first!\")\n        >>> def my_task():\n        >>>     pass\n\n        Define a task with a custom name\n\n        >>> @task(name=\"The Ultimate Task\")\n        >>> def my_task():\n        >>>     pass\n\n        Define a task that retries 3 times with a 5 second delay between attempts\n\n        >>> from random import randint\n        >>>\n        >>> @task(retries=3, retry_delay_seconds=5)\n        >>> def my_task():\n        >>>     x = randint(0, 5)\n        >>>     if x >= 3:  # Make a task that fails sometimes\n        >>>         raise ValueError(\"Retry me please!\")\n        >>>     return x\n\n        Define a task that is cached for a day based on its inputs\n\n        >>> from prefect.tasks import task_input_hash\n        >>> from datetime import timedelta\n        >>>\n        >>> @task(cache_key_fn=task_input_hash, cache_expiration=timedelta(days=1))\n        >>> def my_task():\n        >>>     return \"hello\"\n    \"\"\"\n\n    if __fn:\n        return cast(\n            Task[P, R],\n            Task(\n                fn=__fn,\n                name=name,\n                description=description,\n                tags=tags,\n                version=version,\n                cache_key_fn=cache_key_fn,\n                cache_expiration=cache_expiration,\n                task_run_name=task_run_name,\n                retries=retries,\n                retry_delay_seconds=retry_delay_seconds,\n                retry_jitter_factor=retry_jitter_factor,\n                persist_result=persist_result,\n                result_storage=result_storage,\n                result_storage_key=result_storage_key,\n                result_serializer=result_serializer,\n                cache_result_in_memory=cache_result_in_memory,\n                timeout_seconds=timeout_seconds,\n                log_prints=log_prints,\n                refresh_cache=refresh_cache,\n                on_completion=on_completion,\n                on_failure=on_failure,\n                retry_condition_fn=retry_condition_fn,\n                viz_return_value=viz_return_value,\n            ),\n        )\n    else:\n        return cast(\n            Callable[[Callable[P, R]], Task[P, R]],\n            partial(\n                task,\n                name=name,\n                description=description,\n                tags=tags,\n                version=version,\n                cache_key_fn=cache_key_fn,\n                cache_expiration=cache_expiration,\n                task_run_name=task_run_name,\n                retries=retries,\n                retry_delay_seconds=retry_delay_seconds,\n                retry_jitter_factor=retry_jitter_factor,\n                persist_result=persist_result,\n                result_storage=result_storage,\n                result_storage_key=result_storage_key,\n                result_serializer=result_serializer,\n                cache_result_in_memory=cache_result_in_memory,\n                timeout_seconds=timeout_seconds,\n                log_prints=log_prints,\n                refresh_cache=refresh_cache,\n                on_completion=on_completion,\n                on_failure=on_failure,\n                retry_condition_fn=retry_condition_fn,\n                viz_return_value=viz_return_value,\n            ),\n        )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.task_input_hash","title":"task_input_hash","text":"

    A task cache key implementation which hashes all inputs to the task using a JSON or cloudpickle serializer. If any arguments are not JSON serializable, the pickle serializer is used as a fallback. If cloudpickle fails, this will return a null key indicating that a cache key could not be generated for the given inputs.

    Parameters:

    Name Type Description Default context TaskRunContext

    the active TaskRunContext

    required arguments Dict[str, Any]

    a dictionary of arguments to be passed to the underlying task

    required

    Returns:

    Type Description Optional[str]

    a string hash if hashing succeeded, else None

    Source code in prefect/tasks.py
    def task_input_hash(\n    context: \"TaskRunContext\", arguments: Dict[str, Any]\n) -> Optional[str]:\n    \"\"\"\n    A task cache key implementation which hashes all inputs to the task using a JSON or\n    cloudpickle serializer. If any arguments are not JSON serializable, the pickle\n    serializer is used as a fallback. If cloudpickle fails, this will return a null key\n    indicating that a cache key could not be generated for the given inputs.\n\n    Arguments:\n        context: the active `TaskRunContext`\n        arguments: a dictionary of arguments to be passed to the underlying task\n\n    Returns:\n        a string hash if hashing succeeded, else `None`\n    \"\"\"\n    return hash_objects(\n        # We use the task key to get the qualified name for the task and include the\n        # task functions `co_code` bytes to avoid caching when the underlying function\n        # changes\n        context.task.task_key,\n        context.task.fn.__code__.co_code.hex(),\n        arguments,\n    )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/testing/","title":"prefect.testing","text":"","tags":["Python API","testing"]},{"location":"api-ref/prefect/testing/#prefect.testing","title":"prefect.testing","text":"","tags":["Python API","testing"]},{"location":"api-ref/prefect/variables/","title":"prefect.variables","text":"","tags":["Python API","variables"]},{"location":"api-ref/prefect/variables/#prefect.variables","title":"prefect.variables","text":"","tags":["Python API","variables"]},{"location":"api-ref/prefect/variables/#prefect.variables.get","title":"get async","text":"

    Get a variable by name. If doesn't exist return the default.

        from prefect import variables\n\n    @flow\n    def my_flow():\n        var = variables.get(\"my_var\")\n
    or
        from prefect import variables\n\n    @flow\n    async def my_flow():\n        var = await variables.get(\"my_var\")\n

    Source code in prefect/variables.py
    @sync_compatible\nasync def get(name: str, default: str = None) -> Optional[str]:\n    \"\"\"\n    Get a variable by name. If doesn't exist return the default.\n    ```\n        from prefect import variables\n\n        @flow\n        def my_flow():\n            var = variables.get(\"my_var\")\n    ```\n    or\n    ```\n        from prefect import variables\n\n        @flow\n        async def my_flow():\n            var = await variables.get(\"my_var\")\n    ```\n    \"\"\"\n    variable = await _get_variable_by_name(name)\n    return variable.value if variable else default\n
    ","tags":["Python API","variables"]},{"location":"api-ref/prefect/blocks/core/","title":"core","text":"","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core","title":"prefect.blocks.core","text":"","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block","title":"Block","text":"

    Bases: BaseModel, ABC

    A base class for implementing a block that wraps an external service.

    This class can be defined with an arbitrary set of fields and methods, and couples business logic with data contained in an block document. _block_document_name, _block_document_id, _block_schema_id, and _block_type_id are reserved by Prefect as Block metadata fields, but otherwise a Block can implement arbitrary logic. Blocks can be instantiated without populating these metadata fields, but can only be used interactively, not with the Prefect API.

    Instead of the init method, a block implementation allows the definition of a block_initialization method that is called after initialization.

    Source code in prefect/blocks/core.py
    @register_base_type\n@instrument_method_calls_on_class_instances\nclass Block(BaseModel, ABC):\n    \"\"\"\n    A base class for implementing a block that wraps an external service.\n\n    This class can be defined with an arbitrary set of fields and methods, and\n    couples business logic with data contained in an block document.\n    `_block_document_name`, `_block_document_id`, `_block_schema_id`, and\n    `_block_type_id` are reserved by Prefect as Block metadata fields, but\n    otherwise a Block can implement arbitrary logic. Blocks can be instantiated\n    without populating these metadata fields, but can only be used interactively,\n    not with the Prefect API.\n\n    Instead of the __init__ method, a block implementation allows the\n    definition of a `block_initialization` method that is called after\n    initialization.\n    \"\"\"\n\n    class Config:\n        extra = \"allow\"\n\n        json_encoders = {SecretDict: lambda v: v.dict()}\n\n        @staticmethod\n        def schema_extra(schema: Dict[str, Any], model: Type[\"Block\"]):\n            \"\"\"\n            Customizes Pydantic's schema generation feature to add blocks related information.\n            \"\"\"\n            schema[\"block_type_slug\"] = model.get_block_type_slug()\n            # Ensures args and code examples aren't included in the schema\n            description = model.get_description()\n            if description:\n                schema[\"description\"] = description\n            else:\n                # Prevent the description of the base class from being included in the schema\n                schema.pop(\"description\", None)\n\n            # create a list of secret field names\n            # secret fields include both top-level keys and dot-delimited nested secret keys\n            # A wildcard (*) means that all fields under a given key are secret.\n            # for example: [\"x\", \"y\", \"z.*\", \"child.a\"]\n            # means the top-level keys \"x\" and \"y\", all keys under \"z\", and the key \"a\" of a block\n            # nested under the \"child\" key are all secret. There is no limit to nesting.\n            secrets = schema[\"secret_fields\"] = []\n            for field in model.__fields__.values():\n                _collect_secret_fields(field.name, field.type_, secrets)\n\n            # create block schema references\n            refs = schema[\"block_schema_references\"] = {}\n            for field in model.__fields__.values():\n                if Block.is_block_class(field.type_):\n                    refs[field.name] = field.type_._to_block_schema_reference_dict()\n                if get_origin(field.type_) is Union:\n                    for type_ in get_args(field.type_):\n                        if Block.is_block_class(type_):\n                            if isinstance(refs.get(field.name), list):\n                                refs[field.name].append(\n                                    type_._to_block_schema_reference_dict()\n                                )\n                            elif isinstance(refs.get(field.name), dict):\n                                refs[field.name] = [\n                                    refs[field.name],\n                                    type_._to_block_schema_reference_dict(),\n                                ]\n                            else:\n                                refs[field.name] = (\n                                    type_._to_block_schema_reference_dict()\n                                )\n\n    def __init__(self, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.block_initialization()\n\n    def __str__(self) -> str:\n        return self.__repr__()\n\n    def __repr_args__(self):\n        repr_args = super().__repr_args__()\n        data_keys = self.schema()[\"properties\"].keys()\n        return [\n            (key, value) for key, value in repr_args if key is None or key in data_keys\n        ]\n\n    def block_initialization(self) -> None:\n        pass\n\n    # -- private class variables\n    # set by the class itself\n\n    # Attribute to customize the name of the block type created\n    # when the block is registered with the API. If not set, block\n    # type name will default to the class name.\n    _block_type_name: Optional[str] = None\n    _block_type_slug: Optional[str] = None\n\n    # Attributes used to set properties on a block type when registered\n    # with the API.\n    _logo_url: Optional[HttpUrl] = None\n    _documentation_url: Optional[HttpUrl] = None\n    _description: Optional[str] = None\n    _code_example: Optional[str] = None\n\n    # -- private instance variables\n    # these are set when blocks are loaded from the API\n    _block_type_id: Optional[UUID] = None\n    _block_schema_id: Optional[UUID] = None\n    _block_schema_capabilities: Optional[List[str]] = None\n    _block_schema_version: Optional[str] = None\n    _block_document_id: Optional[UUID] = None\n    _block_document_name: Optional[str] = None\n    _is_anonymous: Optional[bool] = None\n\n    # Exclude `save` as it uses the `sync_compatible` decorator and needs to be\n    # decorated directly.\n    _events_excluded_methods = [\"block_initialization\", \"save\", \"dict\"]\n\n    @classmethod\n    def __dispatch_key__(cls):\n        if cls.__name__ == \"Block\":\n            return None  # The base class is abstract\n        return block_schema_to_key(cls._to_block_schema())\n\n    @classmethod\n    def get_block_type_name(cls):\n        return cls._block_type_name or cls.__name__\n\n    @classmethod\n    def get_block_type_slug(cls):\n        return slugify(cls._block_type_slug or cls.get_block_type_name())\n\n    @classmethod\n    def get_block_capabilities(cls) -> FrozenSet[str]:\n        \"\"\"\n        Returns the block capabilities for this Block. Recursively collects all block\n        capabilities of all parent classes into a single frozenset.\n        \"\"\"\n        return frozenset(\n            {\n                c\n                for base in (cls,) + cls.__mro__\n                for c in getattr(base, \"_block_schema_capabilities\", []) or []\n            }\n        )\n\n    @classmethod\n    def _get_current_package_version(cls):\n        current_module = inspect.getmodule(cls)\n        if current_module:\n            top_level_module = sys.modules[\n                current_module.__name__.split(\".\")[0] or \"__main__\"\n            ]\n            try:\n                version = Version(top_level_module.__version__)\n                # Strips off any local version information\n                return version.base_version\n            except (AttributeError, InvalidVersion):\n                # Module does not have a __version__ attribute or is not a parsable format\n                pass\n        return DEFAULT_BLOCK_SCHEMA_VERSION\n\n    @classmethod\n    def get_block_schema_version(cls) -> str:\n        return cls._block_schema_version or cls._get_current_package_version()\n\n    @classmethod\n    def _to_block_schema_reference_dict(cls):\n        return dict(\n            block_type_slug=cls.get_block_type_slug(),\n            block_schema_checksum=cls._calculate_schema_checksum(),\n        )\n\n    @classmethod\n    def _calculate_schema_checksum(\n        cls, block_schema_fields: Optional[Dict[str, Any]] = None\n    ):\n        \"\"\"\n        Generates a unique hash for the underlying schema of block.\n\n        Args:\n            block_schema_fields: Dictionary detailing block schema fields to generate a\n                checksum for. The fields of the current class is used if this parameter\n                is not provided.\n\n        Returns:\n            str: The calculated checksum prefixed with the hashing algorithm used.\n        \"\"\"\n        block_schema_fields = (\n            cls.schema() if block_schema_fields is None else block_schema_fields\n        )\n        fields_for_checksum = remove_nested_keys([\"secret_fields\"], block_schema_fields)\n        if fields_for_checksum.get(\"definitions\"):\n            non_block_definitions = _get_non_block_reference_definitions(\n                fields_for_checksum, fields_for_checksum[\"definitions\"]\n            )\n            if non_block_definitions:\n                fields_for_checksum[\"definitions\"] = non_block_definitions\n            else:\n                # Pop off definitions entirely instead of empty dict for consistency\n                # with the OpenAPI specification\n                fields_for_checksum.pop(\"definitions\")\n        checksum = hash_objects(fields_for_checksum, hash_algo=hashlib.sha256)\n        if checksum is None:\n            raise ValueError(\"Unable to compute checksum for block schema\")\n        else:\n            return f\"sha256:{checksum}\"\n\n    def _to_block_document(\n        self,\n        name: Optional[str] = None,\n        block_schema_id: Optional[UUID] = None,\n        block_type_id: Optional[UUID] = None,\n        is_anonymous: Optional[bool] = None,\n    ) -> BlockDocument:\n        \"\"\"\n        Creates the corresponding block document based on the data stored in a block.\n        The corresponding block document name, block type ID, and block schema ID must\n        either be passed into the method or configured on the block.\n\n        Args:\n            name: The name of the created block document. Not required if anonymous.\n            block_schema_id: UUID of the corresponding block schema.\n            block_type_id: UUID of the corresponding block type.\n            is_anonymous: if True, an anonymous block is created. Anonymous\n                blocks are not displayed in the UI and used primarily for system\n                operations and features that need to automatically generate blocks.\n\n        Returns:\n            BlockDocument: Corresponding block document\n                populated with the block's configured data.\n        \"\"\"\n        if is_anonymous is None:\n            is_anonymous = self._is_anonymous or False\n\n        # name must be present if not anonymous\n        if not is_anonymous and not name and not self._block_document_name:\n            raise ValueError(\"No name provided, either as an argument or on the block.\")\n\n        if not block_schema_id and not self._block_schema_id:\n            raise ValueError(\n                \"No block schema ID provided, either as an argument or on the block.\"\n            )\n        if not block_type_id and not self._block_type_id:\n            raise ValueError(\n                \"No block type ID provided, either as an argument or on the block.\"\n            )\n\n        # The keys passed to `include` must NOT be aliases, else some items will be missed\n        # i.e. must do `self.schema_` vs `self.schema` to get a `schema_ = Field(alias=\"schema\")`\n        # reported from https://github.com/PrefectHQ/prefect-dbt/issues/54\n        data_keys = self.schema(by_alias=False)[\"properties\"].keys()\n\n        # `block_document_data`` must return the aliased version for it to show in the UI\n        block_document_data = self.dict(by_alias=True, include=data_keys)\n\n        # Iterate through and find blocks that already have saved block documents to\n        # create references to those saved block documents.\n        for key in data_keys:\n            field_value = getattr(self, key)\n            if (\n                isinstance(field_value, Block)\n                and field_value._block_document_id is not None\n            ):\n                block_document_data[key] = {\n                    \"$ref\": {\"block_document_id\": field_value._block_document_id}\n                }\n\n        return BlockDocument(\n            id=self._block_document_id or uuid4(),\n            name=(name or self._block_document_name) if not is_anonymous else None,\n            block_schema_id=block_schema_id or self._block_schema_id,\n            block_type_id=block_type_id or self._block_type_id,\n            data=block_document_data,\n            block_schema=self._to_block_schema(\n                block_type_id=block_type_id or self._block_type_id,\n            ),\n            block_type=self._to_block_type(),\n            is_anonymous=is_anonymous,\n        )\n\n    @classmethod\n    def _to_block_schema(cls, block_type_id: Optional[UUID] = None) -> BlockSchema:\n        \"\"\"\n        Creates the corresponding block schema of the block.\n        The corresponding block_type_id must either be passed into\n        the method or configured on the block.\n\n        Args:\n            block_type_id: UUID of the corresponding block type.\n\n        Returns:\n            BlockSchema: The corresponding block schema.\n        \"\"\"\n        fields = cls.schema()\n        return BlockSchema(\n            id=cls._block_schema_id if cls._block_schema_id is not None else uuid4(),\n            checksum=cls._calculate_schema_checksum(),\n            fields=fields,\n            block_type_id=block_type_id or cls._block_type_id,\n            block_type=cls._to_block_type(),\n            capabilities=list(cls.get_block_capabilities()),\n            version=cls.get_block_schema_version(),\n        )\n\n    @classmethod\n    def _parse_docstring(cls) -> List[DocstringSection]:\n        \"\"\"\n        Parses the docstring into list of DocstringSection objects.\n        Helper method used primarily to suppress irrelevant logs, e.g.\n        `<module>:11: No type or annotation for parameter 'write_json'`\n        because griffe is unable to parse the types from pydantic.BaseModel.\n        \"\"\"\n        with disable_logger(\"griffe.docstrings.google\"):\n            with disable_logger(\"griffe.agents.nodes\"):\n                docstring = Docstring(cls.__doc__)\n                parsed = parse(docstring, Parser.google)\n        return parsed\n\n    @classmethod\n    def get_description(cls) -> Optional[str]:\n        \"\"\"\n        Returns the description for the current block. Attempts to parse\n        description from class docstring if an override is not defined.\n        \"\"\"\n        description = cls._description\n        # If no description override has been provided, find the first text section\n        # and use that as the description\n        if description is None and cls.__doc__ is not None:\n            parsed = cls._parse_docstring()\n            parsed_description = next(\n                (\n                    section.as_dict().get(\"value\")\n                    for section in parsed\n                    if section.kind == DocstringSectionKind.text\n                ),\n                None,\n            )\n            if isinstance(parsed_description, str):\n                description = parsed_description.strip()\n        return description\n\n    @classmethod\n    def get_code_example(cls) -> Optional[str]:\n        \"\"\"\n        Returns the code example for the given block. Attempts to parse\n        code example from the class docstring if an override is not provided.\n        \"\"\"\n        code_example = (\n            dedent(cls._code_example) if cls._code_example is not None else None\n        )\n        # If no code example override has been provided, attempt to find a examples\n        # section or an admonition with the annotation \"example\" and use that as the\n        # code example\n        if code_example is None and cls.__doc__ is not None:\n            parsed = cls._parse_docstring()\n            for section in parsed:\n                # Section kind will be \"examples\" if Examples section heading is used.\n                if section.kind == DocstringSectionKind.examples:\n                    # Examples sections are made up of smaller sections that need to be\n                    # joined with newlines. Smaller sections are represented as tuples\n                    # with shape (DocstringSectionKind, str)\n                    code_example = \"\\n\".join(\n                        (part[1] for part in section.as_dict().get(\"value\", []))\n                    )\n                    break\n                # Section kind will be \"admonition\" if Example section heading is used.\n                if section.kind == DocstringSectionKind.admonition:\n                    value = section.as_dict().get(\"value\", {})\n                    if value.get(\"annotation\") == \"example\":\n                        code_example = value.get(\"description\")\n                        break\n\n        if code_example is None:\n            # If no code example has been specified or extracted from the class\n            # docstring, generate a sensible default\n            code_example = cls._generate_code_example()\n\n        return code_example\n\n    @classmethod\n    def _generate_code_example(cls) -> str:\n        \"\"\"Generates a default code example for the current class\"\"\"\n        qualified_name = to_qualified_name(cls)\n        module_str = \".\".join(qualified_name.split(\".\")[:-1])\n        class_name = cls.__name__\n        block_variable_name = f'{cls.get_block_type_slug().replace(\"-\", \"_\")}_block'\n\n        return dedent(\n            f\"\"\"\\\n        ```python\n        from {module_str} import {class_name}\n\n        {block_variable_name} = {class_name}.load(\"BLOCK_NAME\")\n        ```\"\"\"\n        )\n\n    @classmethod\n    def _to_block_type(cls) -> BlockType:\n        \"\"\"\n        Creates the corresponding block type of the block.\n\n        Returns:\n            BlockType: The corresponding block type.\n        \"\"\"\n        return BlockType(\n            id=cls._block_type_id or uuid4(),\n            slug=cls.get_block_type_slug(),\n            name=cls.get_block_type_name(),\n            logo_url=cls._logo_url,\n            documentation_url=cls._documentation_url,\n            description=cls.get_description(),\n            code_example=cls.get_code_example(),\n        )\n\n    @classmethod\n    def _from_block_document(cls, block_document: BlockDocument):\n        \"\"\"\n        Instantiates a block from a given block document. The corresponding block class\n        will be looked up in the block registry based on the corresponding block schema\n        of the provided block document.\n\n        Args:\n            block_document: The block document used to instantiate a block.\n\n        Raises:\n            ValueError: If the provided block document doesn't have a corresponding block\n                schema.\n\n        Returns:\n            Block: Hydrated block with data from block document.\n        \"\"\"\n        if block_document.block_schema is None:\n            raise ValueError(\n                \"Unable to determine block schema for provided block document\"\n            )\n\n        block_cls = (\n            cls\n            if cls.__name__ != \"Block\"\n            # Look up the block class by dispatch\n            else cls.get_block_class_from_schema(block_document.block_schema)\n        )\n\n        block_cls = instrument_method_calls_on_class_instances(block_cls)\n\n        block = block_cls.parse_obj(block_document.data)\n        block._block_document_id = block_document.id\n        block.__class__._block_schema_id = block_document.block_schema_id\n        block.__class__._block_type_id = block_document.block_type_id\n        block._block_document_name = block_document.name\n        block._is_anonymous = block_document.is_anonymous\n        block._define_metadata_on_nested_blocks(\n            block_document.block_document_references\n        )\n\n        # Due to the way blocks are loaded we can't directly instrument the\n        # `load` method and have the data be about the block document. Instead\n        # this will emit a proxy event for the load method so that block\n        # document data can be included instead of the event being about an\n        # 'anonymous' block.\n\n        emit_instance_method_called_event(block, \"load\", successful=True)\n\n        return block\n\n    def _event_kind(self) -> str:\n        return f\"prefect.block.{self.get_block_type_slug()}\"\n\n    def _event_method_called_resources(self) -> Optional[ResourceTuple]:\n        if not (self._block_document_id and self._block_document_name):\n            return None\n\n        return (\n            {\n                \"prefect.resource.id\": (\n                    f\"prefect.block-document.{self._block_document_id}\"\n                ),\n                \"prefect.resource.name\": self._block_document_name,\n            },\n            [\n                {\n                    \"prefect.resource.id\": (\n                        f\"prefect.block-type.{self.get_block_type_slug()}\"\n                    ),\n                    \"prefect.resource.role\": \"block-type\",\n                }\n            ],\n        )\n\n    @classmethod\n    def get_block_class_from_schema(cls: Type[Self], schema: BlockSchema) -> Type[Self]:\n        \"\"\"\n        Retrieve the block class implementation given a schema.\n        \"\"\"\n        return cls.get_block_class_from_key(block_schema_to_key(schema))\n\n    @classmethod\n    def get_block_class_from_key(cls: Type[Self], key: str) -> Type[Self]:\n        \"\"\"\n        Retrieve the block class implementation given a key.\n        \"\"\"\n        # Ensure collections are imported and have the opportunity to register types\n        # before looking up the block class\n        prefect.plugins.load_prefect_collections()\n\n        return lookup_type(cls, key)\n\n    def _define_metadata_on_nested_blocks(\n        self, block_document_references: Dict[str, Dict[str, Any]]\n    ):\n        \"\"\"\n        Recursively populates metadata fields on nested blocks based on the\n        provided block document references.\n        \"\"\"\n        for item in block_document_references.items():\n            field_name, block_document_reference = item\n            nested_block = getattr(self, field_name)\n            if isinstance(nested_block, Block):\n                nested_block_document_info = block_document_reference.get(\n                    \"block_document\", {}\n                )\n                nested_block._define_metadata_on_nested_blocks(\n                    nested_block_document_info.get(\"block_document_references\", {})\n                )\n                nested_block_document_id = nested_block_document_info.get(\"id\")\n                nested_block._block_document_id = (\n                    UUID(nested_block_document_id) if nested_block_document_id else None\n                )\n                nested_block._block_document_name = nested_block_document_info.get(\n                    \"name\"\n                )\n                nested_block._is_anonymous = nested_block_document_info.get(\n                    \"is_anonymous\"\n                )\n\n    @classmethod\n    @inject_client\n    async def _get_block_document(\n        cls,\n        name: str,\n        client: \"PrefectClient\" = None,\n    ):\n        if cls.__name__ == \"Block\":\n            block_type_slug, block_document_name = name.split(\"/\", 1)\n        else:\n            block_type_slug = cls.get_block_type_slug()\n            block_document_name = name\n\n        try:\n            block_document = await client.read_block_document_by_name(\n                name=block_document_name, block_type_slug=block_type_slug\n            )\n        except prefect.exceptions.ObjectNotFound as e:\n            raise ValueError(\n                f\"Unable to find block document named {block_document_name} for block\"\n                f\" type {block_type_slug}\"\n            ) from e\n\n        return block_document, block_document_name\n\n    @classmethod\n    @sync_compatible\n    @inject_client\n    async def load(\n        cls,\n        name: str,\n        validate: bool = True,\n        client: \"PrefectClient\" = None,\n    ):\n        \"\"\"\n        Retrieves data from the block document with the given name for the block type\n        that corresponds with the current class and returns an instantiated version of\n        the current class with the data stored in the block document.\n\n        If a block document for a given block type is saved with a different schema\n        than the current class calling `load`, a warning will be raised.\n\n        If the current class schema is a subset of the block document schema, the block\n        can be loaded as normal using the default `validate = True`.\n\n        If the current class schema is a superset of the block document schema, `load`\n        must be called with `validate` set to False to prevent a validation error. In\n        this case, the block attributes will default to `None` and must be set manually\n        and saved to a new block document before the block can be used as expected.\n\n        Args:\n            name: The name or slug of the block document. A block document slug is a\n                string with the format <block_type_slug>/<block_document_name>\n            validate: If False, the block document will be loaded without Pydantic\n                validating the block schema. This is useful if the block schema has\n                changed client-side since the block document referred to by `name` was saved.\n            client: The client to use to load the block document. If not provided, the\n                default client will be injected.\n\n        Raises:\n            ValueError: If the requested block document is not found.\n\n        Returns:\n            An instance of the current class hydrated with the data stored in the\n            block document with the specified name.\n\n        Examples:\n            Load from a Block subclass with a block document name:\n            ```python\n            class Custom(Block):\n                message: str\n\n            Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n            loaded_block = Custom.load(\"my-custom-message\")\n            ```\n\n            Load from Block with a block document slug:\n            ```python\n            class Custom(Block):\n                message: str\n\n            Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n            loaded_block = Block.load(\"custom/my-custom-message\")\n            ```\n\n            Migrate a block document to a new schema:\n            ```python\n            # original class\n            class Custom(Block):\n                message: str\n\n            Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n            # Updated class with new required field\n            class Custom(Block):\n                message: str\n                number_of_ducks: int\n\n            loaded_block = Custom.load(\"my-custom-message\", validate=False)\n\n            # Prints UserWarning about schema mismatch\n\n            loaded_block.number_of_ducks = 42\n\n            loaded_block.save(\"my-custom-message\", overwrite=True)\n            ```\n        \"\"\"\n        block_document, block_document_name = await cls._get_block_document(name)\n\n        try:\n            return cls._from_block_document(block_document)\n        except ValidationError as e:\n            if not validate:\n                missing_fields = tuple(err[\"loc\"][0] for err in e.errors())\n                missing_block_data = {field: None for field in missing_fields}\n                warnings.warn(\n                    f\"Could not fully load {block_document_name!r} of block type\"\n                    f\" {cls._block_type_slug!r} - this is likely because one or more\"\n                    \" required fields were added to the schema for\"\n                    f\" {cls.__name__!r} that did not exist on the class when this block\"\n                    \" was last saved. Please specify values for new field(s):\"\n                    f\" {listrepr(missing_fields)}, then run\"\n                    f' `{cls.__name__}.save(\"{block_document_name}\", overwrite=True)`,'\n                    \" and load this block again before attempting to use it.\"\n                )\n                return cls.construct(**block_document.data, **missing_block_data)\n            raise RuntimeError(\n                f\"Unable to load {block_document_name!r} of block type\"\n                f\" {cls._block_type_slug!r} due to failed validation. To load without\"\n                \" validation, try loading again with `validate=False`.\"\n            ) from e\n\n    @staticmethod\n    def is_block_class(block) -> bool:\n        return _is_subclass(block, Block)\n\n    @classmethod\n    @sync_compatible\n    @inject_client\n    async def register_type_and_schema(cls, client: \"PrefectClient\" = None):\n        \"\"\"\n        Makes block available for configuration with current Prefect API.\n        Recursively registers all nested blocks. Registration is idempotent.\n\n        Args:\n            client: Optional client to use for registering type and schema with the\n                Prefect API. A new client will be created and used if one is not\n                provided.\n        \"\"\"\n        if cls.__name__ == \"Block\":\n            raise InvalidBlockRegistration(\n                \"`register_type_and_schema` should be called on a Block \"\n                \"subclass and not on the Block class directly.\"\n            )\n        if ABC in getattr(cls, \"__bases__\", []):\n            raise InvalidBlockRegistration(\n                \"`register_type_and_schema` should be called on a Block \"\n                \"subclass and not on a Block interface class directly.\"\n            )\n\n        for field in cls.__fields__.values():\n            if Block.is_block_class(field.type_):\n                await field.type_.register_type_and_schema(client=client)\n            if get_origin(field.type_) is Union:\n                for type_ in get_args(field.type_):\n                    if Block.is_block_class(type_):\n                        await type_.register_type_and_schema(client=client)\n\n        try:\n            block_type = await client.read_block_type_by_slug(\n                slug=cls.get_block_type_slug()\n            )\n            cls._block_type_id = block_type.id\n            local_block_type = cls._to_block_type()\n            if _should_update_block_type(\n                local_block_type=local_block_type, server_block_type=block_type\n            ):\n                await client.update_block_type(\n                    block_type_id=block_type.id, block_type=local_block_type\n                )\n        except prefect.exceptions.ObjectNotFound:\n            block_type = await client.create_block_type(block_type=cls._to_block_type())\n            cls._block_type_id = block_type.id\n\n        try:\n            block_schema = await client.read_block_schema_by_checksum(\n                checksum=cls._calculate_schema_checksum(),\n                version=cls.get_block_schema_version(),\n            )\n        except prefect.exceptions.ObjectNotFound:\n            block_schema = await client.create_block_schema(\n                block_schema=cls._to_block_schema(block_type_id=block_type.id)\n            )\n\n        cls._block_schema_id = block_schema.id\n\n    @inject_client\n    async def _save(\n        self,\n        name: Optional[str] = None,\n        is_anonymous: bool = False,\n        overwrite: bool = False,\n        client: \"PrefectClient\" = None,\n    ):\n        \"\"\"\n        Saves the values of a block as a block document with an option to save as an\n        anonymous block document.\n\n        Args:\n            name: User specified name to give saved block document which can later be used to load the\n                block document.\n            is_anonymous: Boolean value specifying whether the block document is anonymous. Anonymous\n                blocks are intended for system use and are not shown in the UI. Anonymous blocks do not\n                require a user-supplied name.\n            overwrite: Boolean value specifying if values should be overwritten if a block document with\n                the specified name already exists.\n\n        Raises:\n            ValueError: If a name is not given and `is_anonymous` is `False` or a name is given and\n                `is_anonymous` is `True`.\n        \"\"\"\n        if name is None and not is_anonymous:\n            raise ValueError(\n                \"You're attempting to save a block document without a name. \"\n                \"Please either save a block document with a name or set \"\n                \"is_anonymous to True.\"\n            )\n\n        self._is_anonymous = is_anonymous\n\n        # Ensure block type and schema are registered before saving block document.\n        await self.register_type_and_schema(client=client)\n\n        try:\n            block_document = await client.create_block_document(\n                block_document=self._to_block_document(name=name)\n            )\n        except prefect.exceptions.ObjectAlreadyExists as err:\n            if overwrite:\n                block_document_id = self._block_document_id\n                if block_document_id is None:\n                    existing_block_document = await client.read_block_document_by_name(\n                        name=name, block_type_slug=self.get_block_type_slug()\n                    )\n                    block_document_id = existing_block_document.id\n                await client.update_block_document(\n                    block_document_id=block_document_id,\n                    block_document=self._to_block_document(name=name),\n                )\n                block_document = await client.read_block_document(\n                    block_document_id=block_document_id\n                )\n            else:\n                raise ValueError(\n                    \"You are attempting to save values with a name that is already in\"\n                    \" use for this block type. If you would like to overwrite the\"\n                    \" values that are saved, then save with `overwrite=True`.\"\n                ) from err\n\n        # Update metadata on block instance for later use.\n        self._block_document_name = block_document.name\n        self._block_document_id = block_document.id\n        return self._block_document_id\n\n    @sync_compatible\n    @instrument_instance_method_call()\n    async def save(\n        self, name: str, overwrite: bool = False, client: \"PrefectClient\" = None\n    ):\n        \"\"\"\n        Saves the values of a block as a block document.\n\n        Args:\n            name: User specified name to give saved block document which can later be used to load the\n                block document.\n            overwrite: Boolean value specifying if values should be overwritten if a block document with\n                the specified name already exists.\n\n        \"\"\"\n        document_id = await self._save(name=name, overwrite=overwrite, client=client)\n\n        return document_id\n\n    @classmethod\n    @sync_compatible\n    @inject_client\n    async def delete(\n        cls,\n        name: str,\n        client: \"PrefectClient\" = None,\n    ):\n        block_document, block_document_name = await cls._get_block_document(name)\n\n        await client.delete_block_document(block_document.id)\n\n    def _iter(self, *, include=None, exclude=None, **kwargs):\n        # Injects the `block_type_slug` into serialized payloads for dispatch\n        for key_value in super()._iter(include=include, exclude=exclude, **kwargs):\n            yield key_value\n\n        # Respect inclusion and exclusion still\n        if include and \"block_type_slug\" not in include:\n            return\n        if exclude and \"block_type_slug\" in exclude:\n            return\n\n        yield \"block_type_slug\", self.get_block_type_slug()\n\n    def __new__(cls: Type[Self], **kwargs) -> Self:\n        \"\"\"\n        Create an instance of the Block subclass type if a `block_type_slug` is\n        present in the data payload.\n        \"\"\"\n        block_type_slug = kwargs.pop(\"block_type_slug\", None)\n        if block_type_slug:\n            subcls = lookup_type(cls, dispatch_key=block_type_slug)\n            m = super().__new__(subcls)\n            # NOTE: This is a workaround for an obscure issue where copied models were\n            #       missing attributes. This pattern is from Pydantic's\n            #       `BaseModel._copy_and_set_values`.\n            #       The issue this fixes could not be reproduced in unit tests that\n            #       directly targeted dispatch handling and was only observed when\n            #       copying then saving infrastructure blocks on deployment models.\n            object.__setattr__(m, \"__dict__\", kwargs)\n            object.__setattr__(m, \"__fields_set__\", set(kwargs.keys()))\n            return m\n        else:\n            m = super().__new__(cls)\n            object.__setattr__(m, \"__dict__\", kwargs)\n            object.__setattr__(m, \"__fields_set__\", set(kwargs.keys()))\n            return m\n\n    def get_block_placeholder(self) -> str:\n        \"\"\"\n        Returns the block placeholder for the current block which can be used for\n        templating.\n\n        Returns:\n            str: The block placeholder for the current block in the format\n                `prefect.blocks.{block_type_name}.{block_document_name}`\n\n        Raises:\n            BlockNotSavedError: Raised if the block has not been saved.\n\n        If a block has not been saved, the return value will be `None`.\n        \"\"\"\n        block_document_name = self._block_document_name\n        if not block_document_name:\n            raise BlockNotSavedError(\n                \"Could not generate block placeholder for unsaved block.\"\n            )\n\n        return f\"prefect.blocks.{self.get_block_type_slug()}.{block_document_name}\"\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.Config","title":"Config","text":"Source code in prefect/blocks/core.py
    class Config:\n    extra = \"allow\"\n\n    json_encoders = {SecretDict: lambda v: v.dict()}\n\n    @staticmethod\n    def schema_extra(schema: Dict[str, Any], model: Type[\"Block\"]):\n        \"\"\"\n        Customizes Pydantic's schema generation feature to add blocks related information.\n        \"\"\"\n        schema[\"block_type_slug\"] = model.get_block_type_slug()\n        # Ensures args and code examples aren't included in the schema\n        description = model.get_description()\n        if description:\n            schema[\"description\"] = description\n        else:\n            # Prevent the description of the base class from being included in the schema\n            schema.pop(\"description\", None)\n\n        # create a list of secret field names\n        # secret fields include both top-level keys and dot-delimited nested secret keys\n        # A wildcard (*) means that all fields under a given key are secret.\n        # for example: [\"x\", \"y\", \"z.*\", \"child.a\"]\n        # means the top-level keys \"x\" and \"y\", all keys under \"z\", and the key \"a\" of a block\n        # nested under the \"child\" key are all secret. There is no limit to nesting.\n        secrets = schema[\"secret_fields\"] = []\n        for field in model.__fields__.values():\n            _collect_secret_fields(field.name, field.type_, secrets)\n\n        # create block schema references\n        refs = schema[\"block_schema_references\"] = {}\n        for field in model.__fields__.values():\n            if Block.is_block_class(field.type_):\n                refs[field.name] = field.type_._to_block_schema_reference_dict()\n            if get_origin(field.type_) is Union:\n                for type_ in get_args(field.type_):\n                    if Block.is_block_class(type_):\n                        if isinstance(refs.get(field.name), list):\n                            refs[field.name].append(\n                                type_._to_block_schema_reference_dict()\n                            )\n                        elif isinstance(refs.get(field.name), dict):\n                            refs[field.name] = [\n                                refs[field.name],\n                                type_._to_block_schema_reference_dict(),\n                            ]\n                        else:\n                            refs[field.name] = (\n                                type_._to_block_schema_reference_dict()\n                            )\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.Config.schema_extra","title":"schema_extra staticmethod","text":"

    Customizes Pydantic's schema generation feature to add blocks related information.

    Source code in prefect/blocks/core.py
    @staticmethod\ndef schema_extra(schema: Dict[str, Any], model: Type[\"Block\"]):\n    \"\"\"\n    Customizes Pydantic's schema generation feature to add blocks related information.\n    \"\"\"\n    schema[\"block_type_slug\"] = model.get_block_type_slug()\n    # Ensures args and code examples aren't included in the schema\n    description = model.get_description()\n    if description:\n        schema[\"description\"] = description\n    else:\n        # Prevent the description of the base class from being included in the schema\n        schema.pop(\"description\", None)\n\n    # create a list of secret field names\n    # secret fields include both top-level keys and dot-delimited nested secret keys\n    # A wildcard (*) means that all fields under a given key are secret.\n    # for example: [\"x\", \"y\", \"z.*\", \"child.a\"]\n    # means the top-level keys \"x\" and \"y\", all keys under \"z\", and the key \"a\" of a block\n    # nested under the \"child\" key are all secret. There is no limit to nesting.\n    secrets = schema[\"secret_fields\"] = []\n    for field in model.__fields__.values():\n        _collect_secret_fields(field.name, field.type_, secrets)\n\n    # create block schema references\n    refs = schema[\"block_schema_references\"] = {}\n    for field in model.__fields__.values():\n        if Block.is_block_class(field.type_):\n            refs[field.name] = field.type_._to_block_schema_reference_dict()\n        if get_origin(field.type_) is Union:\n            for type_ in get_args(field.type_):\n                if Block.is_block_class(type_):\n                    if isinstance(refs.get(field.name), list):\n                        refs[field.name].append(\n                            type_._to_block_schema_reference_dict()\n                        )\n                    elif isinstance(refs.get(field.name), dict):\n                        refs[field.name] = [\n                            refs[field.name],\n                            type_._to_block_schema_reference_dict(),\n                        ]\n                    else:\n                        refs[field.name] = (\n                            type_._to_block_schema_reference_dict()\n                        )\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_block_capabilities","title":"get_block_capabilities classmethod","text":"

    Returns the block capabilities for this Block. Recursively collects all block capabilities of all parent classes into a single frozenset.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_block_capabilities(cls) -> FrozenSet[str]:\n    \"\"\"\n    Returns the block capabilities for this Block. Recursively collects all block\n    capabilities of all parent classes into a single frozenset.\n    \"\"\"\n    return frozenset(\n        {\n            c\n            for base in (cls,) + cls.__mro__\n            for c in getattr(base, \"_block_schema_capabilities\", []) or []\n        }\n    )\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_block_class_from_key","title":"get_block_class_from_key classmethod","text":"

    Retrieve the block class implementation given a key.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_block_class_from_key(cls: Type[Self], key: str) -> Type[Self]:\n    \"\"\"\n    Retrieve the block class implementation given a key.\n    \"\"\"\n    # Ensure collections are imported and have the opportunity to register types\n    # before looking up the block class\n    prefect.plugins.load_prefect_collections()\n\n    return lookup_type(cls, key)\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_block_class_from_schema","title":"get_block_class_from_schema classmethod","text":"

    Retrieve the block class implementation given a schema.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_block_class_from_schema(cls: Type[Self], schema: BlockSchema) -> Type[Self]:\n    \"\"\"\n    Retrieve the block class implementation given a schema.\n    \"\"\"\n    return cls.get_block_class_from_key(block_schema_to_key(schema))\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_block_placeholder","title":"get_block_placeholder","text":"

    Returns the block placeholder for the current block which can be used for templating.

    Returns:

    Name Type Description str str

    The block placeholder for the current block in the format prefect.blocks.{block_type_name}.{block_document_name}

    Raises:

    Type Description BlockNotSavedError

    Raised if the block has not been saved.

    If a block has not been saved, the return value will be None.

    Source code in prefect/blocks/core.py
    def get_block_placeholder(self) -> str:\n    \"\"\"\n    Returns the block placeholder for the current block which can be used for\n    templating.\n\n    Returns:\n        str: The block placeholder for the current block in the format\n            `prefect.blocks.{block_type_name}.{block_document_name}`\n\n    Raises:\n        BlockNotSavedError: Raised if the block has not been saved.\n\n    If a block has not been saved, the return value will be `None`.\n    \"\"\"\n    block_document_name = self._block_document_name\n    if not block_document_name:\n        raise BlockNotSavedError(\n            \"Could not generate block placeholder for unsaved block.\"\n        )\n\n    return f\"prefect.blocks.{self.get_block_type_slug()}.{block_document_name}\"\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_code_example","title":"get_code_example classmethod","text":"

    Returns the code example for the given block. Attempts to parse code example from the class docstring if an override is not provided.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_code_example(cls) -> Optional[str]:\n    \"\"\"\n    Returns the code example for the given block. Attempts to parse\n    code example from the class docstring if an override is not provided.\n    \"\"\"\n    code_example = (\n        dedent(cls._code_example) if cls._code_example is not None else None\n    )\n    # If no code example override has been provided, attempt to find a examples\n    # section or an admonition with the annotation \"example\" and use that as the\n    # code example\n    if code_example is None and cls.__doc__ is not None:\n        parsed = cls._parse_docstring()\n        for section in parsed:\n            # Section kind will be \"examples\" if Examples section heading is used.\n            if section.kind == DocstringSectionKind.examples:\n                # Examples sections are made up of smaller sections that need to be\n                # joined with newlines. Smaller sections are represented as tuples\n                # with shape (DocstringSectionKind, str)\n                code_example = \"\\n\".join(\n                    (part[1] for part in section.as_dict().get(\"value\", []))\n                )\n                break\n            # Section kind will be \"admonition\" if Example section heading is used.\n            if section.kind == DocstringSectionKind.admonition:\n                value = section.as_dict().get(\"value\", {})\n                if value.get(\"annotation\") == \"example\":\n                    code_example = value.get(\"description\")\n                    break\n\n    if code_example is None:\n        # If no code example has been specified or extracted from the class\n        # docstring, generate a sensible default\n        code_example = cls._generate_code_example()\n\n    return code_example\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_description","title":"get_description classmethod","text":"

    Returns the description for the current block. Attempts to parse description from class docstring if an override is not defined.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_description(cls) -> Optional[str]:\n    \"\"\"\n    Returns the description for the current block. Attempts to parse\n    description from class docstring if an override is not defined.\n    \"\"\"\n    description = cls._description\n    # If no description override has been provided, find the first text section\n    # and use that as the description\n    if description is None and cls.__doc__ is not None:\n        parsed = cls._parse_docstring()\n        parsed_description = next(\n            (\n                section.as_dict().get(\"value\")\n                for section in parsed\n                if section.kind == DocstringSectionKind.text\n            ),\n            None,\n        )\n        if isinstance(parsed_description, str):\n            description = parsed_description.strip()\n    return description\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.load","title":"load async classmethod","text":"

    Retrieves data from the block document with the given name for the block type that corresponds with the current class and returns an instantiated version of the current class with the data stored in the block document.

    If a block document for a given block type is saved with a different schema than the current class calling load, a warning will be raised.

    If the current class schema is a subset of the block document schema, the block can be loaded as normal using the default validate = True.

    If the current class schema is a superset of the block document schema, load must be called with validate set to False to prevent a validation error. In this case, the block attributes will default to None and must be set manually and saved to a new block document before the block can be used as expected.

    Parameters:

    Name Type Description Default name str

    The name or slug of the block document. A block document slug is a string with the format / required validate bool

    If False, the block document will be loaded without Pydantic validating the block schema. This is useful if the block schema has changed client-side since the block document referred to by name was saved.

    True client PrefectClient

    The client to use to load the block document. If not provided, the default client will be injected.

    None

    Raises:

    Type Description ValueError

    If the requested block document is not found.

    Returns:

    Type Description

    An instance of the current class hydrated with the data stored in the

    block document with the specified name.

    Examples:

    Load from a Block subclass with a block document name:

    class Custom(Block):\n    message: str\n\nCustom(message=\"Hello!\").save(\"my-custom-message\")\n\nloaded_block = Custom.load(\"my-custom-message\")\n

    Load from Block with a block document slug:

    class Custom(Block):\n    message: str\n\nCustom(message=\"Hello!\").save(\"my-custom-message\")\n\nloaded_block = Block.load(\"custom/my-custom-message\")\n

    Migrate a block document to a new schema:

    # original class\nclass Custom(Block):\n    message: str\n\nCustom(message=\"Hello!\").save(\"my-custom-message\")\n\n# Updated class with new required field\nclass Custom(Block):\n    message: str\n    number_of_ducks: int\n\nloaded_block = Custom.load(\"my-custom-message\", validate=False)\n\n# Prints UserWarning about schema mismatch\n\nloaded_block.number_of_ducks = 42\n\nloaded_block.save(\"my-custom-message\", overwrite=True)\n

    Source code in prefect/blocks/core.py
    @classmethod\n@sync_compatible\n@inject_client\nasync def load(\n    cls,\n    name: str,\n    validate: bool = True,\n    client: \"PrefectClient\" = None,\n):\n    \"\"\"\n    Retrieves data from the block document with the given name for the block type\n    that corresponds with the current class and returns an instantiated version of\n    the current class with the data stored in the block document.\n\n    If a block document for a given block type is saved with a different schema\n    than the current class calling `load`, a warning will be raised.\n\n    If the current class schema is a subset of the block document schema, the block\n    can be loaded as normal using the default `validate = True`.\n\n    If the current class schema is a superset of the block document schema, `load`\n    must be called with `validate` set to False to prevent a validation error. In\n    this case, the block attributes will default to `None` and must be set manually\n    and saved to a new block document before the block can be used as expected.\n\n    Args:\n        name: The name or slug of the block document. A block document slug is a\n            string with the format <block_type_slug>/<block_document_name>\n        validate: If False, the block document will be loaded without Pydantic\n            validating the block schema. This is useful if the block schema has\n            changed client-side since the block document referred to by `name` was saved.\n        client: The client to use to load the block document. If not provided, the\n            default client will be injected.\n\n    Raises:\n        ValueError: If the requested block document is not found.\n\n    Returns:\n        An instance of the current class hydrated with the data stored in the\n        block document with the specified name.\n\n    Examples:\n        Load from a Block subclass with a block document name:\n        ```python\n        class Custom(Block):\n            message: str\n\n        Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n        loaded_block = Custom.load(\"my-custom-message\")\n        ```\n\n        Load from Block with a block document slug:\n        ```python\n        class Custom(Block):\n            message: str\n\n        Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n        loaded_block = Block.load(\"custom/my-custom-message\")\n        ```\n\n        Migrate a block document to a new schema:\n        ```python\n        # original class\n        class Custom(Block):\n            message: str\n\n        Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n        # Updated class with new required field\n        class Custom(Block):\n            message: str\n            number_of_ducks: int\n\n        loaded_block = Custom.load(\"my-custom-message\", validate=False)\n\n        # Prints UserWarning about schema mismatch\n\n        loaded_block.number_of_ducks = 42\n\n        loaded_block.save(\"my-custom-message\", overwrite=True)\n        ```\n    \"\"\"\n    block_document, block_document_name = await cls._get_block_document(name)\n\n    try:\n        return cls._from_block_document(block_document)\n    except ValidationError as e:\n        if not validate:\n            missing_fields = tuple(err[\"loc\"][0] for err in e.errors())\n            missing_block_data = {field: None for field in missing_fields}\n            warnings.warn(\n                f\"Could not fully load {block_document_name!r} of block type\"\n                f\" {cls._block_type_slug!r} - this is likely because one or more\"\n                \" required fields were added to the schema for\"\n                f\" {cls.__name__!r} that did not exist on the class when this block\"\n                \" was last saved. Please specify values for new field(s):\"\n                f\" {listrepr(missing_fields)}, then run\"\n                f' `{cls.__name__}.save(\"{block_document_name}\", overwrite=True)`,'\n                \" and load this block again before attempting to use it.\"\n            )\n            return cls.construct(**block_document.data, **missing_block_data)\n        raise RuntimeError(\n            f\"Unable to load {block_document_name!r} of block type\"\n            f\" {cls._block_type_slug!r} due to failed validation. To load without\"\n            \" validation, try loading again with `validate=False`.\"\n        ) from e\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.register_type_and_schema","title":"register_type_and_schema async classmethod","text":"

    Makes block available for configuration with current Prefect API. Recursively registers all nested blocks. Registration is idempotent.

    Parameters:

    Name Type Description Default client PrefectClient

    Optional client to use for registering type and schema with the Prefect API. A new client will be created and used if one is not provided.

    None Source code in prefect/blocks/core.py
    @classmethod\n@sync_compatible\n@inject_client\nasync def register_type_and_schema(cls, client: \"PrefectClient\" = None):\n    \"\"\"\n    Makes block available for configuration with current Prefect API.\n    Recursively registers all nested blocks. Registration is idempotent.\n\n    Args:\n        client: Optional client to use for registering type and schema with the\n            Prefect API. A new client will be created and used if one is not\n            provided.\n    \"\"\"\n    if cls.__name__ == \"Block\":\n        raise InvalidBlockRegistration(\n            \"`register_type_and_schema` should be called on a Block \"\n            \"subclass and not on the Block class directly.\"\n        )\n    if ABC in getattr(cls, \"__bases__\", []):\n        raise InvalidBlockRegistration(\n            \"`register_type_and_schema` should be called on a Block \"\n            \"subclass and not on a Block interface class directly.\"\n        )\n\n    for field in cls.__fields__.values():\n        if Block.is_block_class(field.type_):\n            await field.type_.register_type_and_schema(client=client)\n        if get_origin(field.type_) is Union:\n            for type_ in get_args(field.type_):\n                if Block.is_block_class(type_):\n                    await type_.register_type_and_schema(client=client)\n\n    try:\n        block_type = await client.read_block_type_by_slug(\n            slug=cls.get_block_type_slug()\n        )\n        cls._block_type_id = block_type.id\n        local_block_type = cls._to_block_type()\n        if _should_update_block_type(\n            local_block_type=local_block_type, server_block_type=block_type\n        ):\n            await client.update_block_type(\n                block_type_id=block_type.id, block_type=local_block_type\n            )\n    except prefect.exceptions.ObjectNotFound:\n        block_type = await client.create_block_type(block_type=cls._to_block_type())\n        cls._block_type_id = block_type.id\n\n    try:\n        block_schema = await client.read_block_schema_by_checksum(\n            checksum=cls._calculate_schema_checksum(),\n            version=cls.get_block_schema_version(),\n        )\n    except prefect.exceptions.ObjectNotFound:\n        block_schema = await client.create_block_schema(\n            block_schema=cls._to_block_schema(block_type_id=block_type.id)\n        )\n\n    cls._block_schema_id = block_schema.id\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.save","title":"save async","text":"

    Saves the values of a block as a block document.

    Parameters:

    Name Type Description Default name str

    User specified name to give saved block document which can later be used to load the block document.

    required overwrite bool

    Boolean value specifying if values should be overwritten if a block document with the specified name already exists.

    False Source code in prefect/blocks/core.py
    @sync_compatible\n@instrument_instance_method_call()\nasync def save(\n    self, name: str, overwrite: bool = False, client: \"PrefectClient\" = None\n):\n    \"\"\"\n    Saves the values of a block as a block document.\n\n    Args:\n        name: User specified name to give saved block document which can later be used to load the\n            block document.\n        overwrite: Boolean value specifying if values should be overwritten if a block document with\n            the specified name already exists.\n\n    \"\"\"\n    document_id = await self._save(name=name, overwrite=overwrite, client=client)\n\n    return document_id\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.BlockNotSavedError","title":"BlockNotSavedError","text":"

    Bases: RuntimeError

    Raised when a given block is not saved and an operation that requires the block to be saved is attempted.

    Source code in prefect/blocks/core.py
    class BlockNotSavedError(RuntimeError):\n    \"\"\"\n    Raised when a given block is not saved and an operation that requires\n    the block to be saved is attempted.\n    \"\"\"\n\n    pass\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.InvalidBlockRegistration","title":"InvalidBlockRegistration","text":"

    Bases: Exception

    Raised on attempted registration of the base Block class or a Block interface class

    Source code in prefect/blocks/core.py
    class InvalidBlockRegistration(Exception):\n    \"\"\"\n    Raised on attempted registration of the base Block\n    class or a Block interface class\n    \"\"\"\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.block_schema_to_key","title":"block_schema_to_key","text":"

    Defines the unique key used to lookup the Block class for a given schema.

    Source code in prefect/blocks/core.py
    def block_schema_to_key(schema: BlockSchema) -> str:\n    \"\"\"\n    Defines the unique key used to lookup the Block class for a given schema.\n    \"\"\"\n    return f\"{schema.block_type.slug}\"\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/fields/","title":"fields","text":"","tags":["Python API","fields"]},{"location":"api-ref/prefect/blocks/fields/#prefect.blocks.fields","title":"prefect.blocks.fields","text":"","tags":["Python API","fields"]},{"location":"api-ref/prefect/blocks/kubernetes/","title":"kubernetes","text":"","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes","title":"prefect.blocks.kubernetes","text":"","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig","title":"KubernetesClusterConfig","text":"

    Bases: Block

    Stores configuration for interaction with Kubernetes clusters.

    See from_file for creation.

    Attributes:

    Name Type Description config Dict

    The entire loaded YAML contents of a kubectl config file

    context_name str

    The name of the kubectl context to use

    Example

    Load a saved Kubernetes cluster config:

    from prefect.blocks.kubernetes import KubernetesClusterConfig\n\ncluster_config_block = KubernetesClusterConfig.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/kubernetes.py
    class KubernetesClusterConfig(Block):\n    \"\"\"\n    Stores configuration for interaction with Kubernetes clusters.\n\n    See `from_file` for creation.\n\n    Attributes:\n        config: The entire loaded YAML contents of a kubectl config file\n        context_name: The name of the kubectl context to use\n\n    Example:\n        Load a saved Kubernetes cluster config:\n        ```python\n        from prefect.blocks.kubernetes import KubernetesClusterConfig\n\n        cluster_config_block = KubernetesClusterConfig.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Kubernetes Cluster Config\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/2d0b896006ad463b49c28aaac14f31e00e32cfab-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig\"\n\n    config: Dict = Field(\n        default=..., description=\"The entire contents of a kubectl config file.\"\n    )\n    context_name: str = Field(\n        default=..., description=\"The name of the kubectl context to use.\"\n    )\n\n    @validator(\"config\", pre=True)\n    def parse_yaml_config(cls, value):\n        if isinstance(value, str):\n            return yaml.safe_load(value)\n        return value\n\n    @classmethod\n    def from_file(cls: Type[Self], path: Path = None, context_name: str = None) -> Self:\n        \"\"\"\n        Create a cluster config from the a Kubernetes config file.\n\n        By default, the current context in the default Kubernetes config file will be\n        used.\n\n        An alternative file or context may be specified.\n\n        The entire config file will be loaded and stored.\n        \"\"\"\n        kube_config = kubernetes.config.kube_config\n\n        path = Path(path or kube_config.KUBE_CONFIG_DEFAULT_LOCATION)\n        path = path.expanduser().resolve()\n\n        # Determine the context\n        existing_contexts, current_context = kube_config.list_kube_config_contexts(\n            config_file=str(path)\n        )\n        context_names = {ctx[\"name\"] for ctx in existing_contexts}\n        if context_name:\n            if context_name not in context_names:\n                raise ValueError(\n                    f\"Context {context_name!r} not found. \"\n                    f\"Specify one of: {listrepr(context_names, sep=', ')}.\"\n                )\n        else:\n            context_name = current_context[\"name\"]\n\n        # Load the entire config file\n        config_file_contents = path.read_text()\n        config_dict = yaml.safe_load(config_file_contents)\n\n        return cls(config=config_dict, context_name=context_name)\n\n    def get_api_client(self) -> \"ApiClient\":\n        \"\"\"\n        Returns a Kubernetes API client for this cluster config.\n        \"\"\"\n        return kubernetes.config.kube_config.new_client_from_config_dict(\n            config_dict=self.config, context=self.context_name\n        )\n\n    def configure_client(self) -> None:\n        \"\"\"\n        Activates this cluster configuration by loading the configuration into the\n        Kubernetes Python client. After calling this, Kubernetes API clients can use\n        this config's context.\n        \"\"\"\n        kubernetes.config.kube_config.load_kube_config_from_dict(\n            config_dict=self.config, context=self.context_name\n        )\n
    ","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig.configure_client","title":"configure_client","text":"

    Activates this cluster configuration by loading the configuration into the Kubernetes Python client. After calling this, Kubernetes API clients can use this config's context.

    Source code in prefect/blocks/kubernetes.py
    def configure_client(self) -> None:\n    \"\"\"\n    Activates this cluster configuration by loading the configuration into the\n    Kubernetes Python client. After calling this, Kubernetes API clients can use\n    this config's context.\n    \"\"\"\n    kubernetes.config.kube_config.load_kube_config_from_dict(\n        config_dict=self.config, context=self.context_name\n    )\n
    ","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig.from_file","title":"from_file classmethod","text":"

    Create a cluster config from the a Kubernetes config file.

    By default, the current context in the default Kubernetes config file will be used.

    An alternative file or context may be specified.

    The entire config file will be loaded and stored.

    Source code in prefect/blocks/kubernetes.py
    @classmethod\ndef from_file(cls: Type[Self], path: Path = None, context_name: str = None) -> Self:\n    \"\"\"\n    Create a cluster config from the a Kubernetes config file.\n\n    By default, the current context in the default Kubernetes config file will be\n    used.\n\n    An alternative file or context may be specified.\n\n    The entire config file will be loaded and stored.\n    \"\"\"\n    kube_config = kubernetes.config.kube_config\n\n    path = Path(path or kube_config.KUBE_CONFIG_DEFAULT_LOCATION)\n    path = path.expanduser().resolve()\n\n    # Determine the context\n    existing_contexts, current_context = kube_config.list_kube_config_contexts(\n        config_file=str(path)\n    )\n    context_names = {ctx[\"name\"] for ctx in existing_contexts}\n    if context_name:\n        if context_name not in context_names:\n            raise ValueError(\n                f\"Context {context_name!r} not found. \"\n                f\"Specify one of: {listrepr(context_names, sep=', ')}.\"\n            )\n    else:\n        context_name = current_context[\"name\"]\n\n    # Load the entire config file\n    config_file_contents = path.read_text()\n    config_dict = yaml.safe_load(config_file_contents)\n\n    return cls(config=config_dict, context_name=context_name)\n
    ","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig.get_api_client","title":"get_api_client","text":"

    Returns a Kubernetes API client for this cluster config.

    Source code in prefect/blocks/kubernetes.py
    def get_api_client(self) -> \"ApiClient\":\n    \"\"\"\n    Returns a Kubernetes API client for this cluster config.\n    \"\"\"\n    return kubernetes.config.kube_config.new_client_from_config_dict(\n        config_dict=self.config, context=self.context_name\n    )\n
    ","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/notifications/","title":"notifications","text":"","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications","title":"prefect.blocks.notifications","text":"","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.AbstractAppriseNotificationBlock","title":"AbstractAppriseNotificationBlock","text":"

    Bases: NotificationBlock, ABC

    An abstract class for sending notifications using Apprise.

    Source code in prefect/blocks/notifications.py
    class AbstractAppriseNotificationBlock(NotificationBlock, ABC):\n    \"\"\"\n    An abstract class for sending notifications using Apprise.\n    \"\"\"\n\n    notify_type: Literal[\"prefect_default\", \"info\", \"success\", \"warning\", \"failure\"] = (\n        Field(\n            default=PREFECT_NOTIFY_TYPE_DEFAULT,\n            description=(\n                \"The type of notification being performed; the prefect_default \"\n                \"is a plain notification that does not attach an image.\"\n            ),\n        )\n    )\n\n    def __init__(self, *args, **kwargs):\n        import apprise\n\n        if PREFECT_NOTIFY_TYPE_DEFAULT not in apprise.NOTIFY_TYPES:\n            apprise.NOTIFY_TYPES += (PREFECT_NOTIFY_TYPE_DEFAULT,)\n\n        super().__init__(*args, **kwargs)\n\n    def _start_apprise_client(self, url: SecretStr):\n        from apprise import Apprise, AppriseAsset\n\n        # A custom `AppriseAsset` that ensures Prefect Notifications\n        # appear correctly across multiple messaging platforms\n        prefect_app_data = AppriseAsset(\n            app_id=\"Prefect Notifications\",\n            app_desc=\"Prefect Notifications\",\n            app_url=\"https://prefect.io\",\n        )\n\n        self._apprise_client = Apprise(asset=prefect_app_data)\n        self._apprise_client.add(url.get_secret_value())\n\n    def block_initialization(self) -> None:\n        self._start_apprise_client(self.url)\n\n    @sync_compatible\n    @instrument_instance_method_call()\n    async def notify(self, body: str, subject: Optional[str] = None):\n        await self._apprise_client.async_notify(\n            body=body, title=subject, notify_type=self.notify_type\n        )\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.AppriseNotificationBlock","title":"AppriseNotificationBlock","text":"

    Bases: AbstractAppriseNotificationBlock, ABC

    A base class for sending notifications using Apprise, through webhook URLs.

    Source code in prefect/blocks/notifications.py
    class AppriseNotificationBlock(AbstractAppriseNotificationBlock, ABC):\n    \"\"\"\n    A base class for sending notifications using Apprise, through webhook URLs.\n    \"\"\"\n\n    _documentation_url = \"https://docs.prefect.io/ui/notifications/\"\n    url: SecretStr = Field(\n        default=...,\n        title=\"Webhook URL\",\n        description=\"Incoming webhook URL used to send notifications.\",\n        example=\"https://hooks.example.com/XXX\",\n    )\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.CustomWebhookNotificationBlock","title":"CustomWebhookNotificationBlock","text":"

    Bases: NotificationBlock

    Enables sending notifications via any custom webhook.

    All nested string param contains {{key}} will be substituted with value from context/secrets.

    Context values include: subject, body and name.

    Examples:

    Load a saved custom webhook and send a message:

    from prefect.blocks.notifications import CustomWebhookNotificationBlock\n\ncustom_webhook_block = CustomWebhookNotificationBlock.load(\"BLOCK_NAME\")\n\ncustom_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class CustomWebhookNotificationBlock(NotificationBlock):\n    \"\"\"\n    Enables sending notifications via any custom webhook.\n\n    All nested string param contains `{{key}}` will be substituted with value from context/secrets.\n\n    Context values include: `subject`, `body` and `name`.\n\n    Examples:\n        Load a saved custom webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import CustomWebhookNotificationBlock\n\n        custom_webhook_block = CustomWebhookNotificationBlock.load(\"BLOCK_NAME\")\n\n        custom_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Custom Webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c7247cb359eb6cf276734d4b1fbf00fb8930e89e-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.CustomWebhookNotificationBlock\"\n\n    name: str = Field(title=\"Name\", description=\"Name of the webhook.\")\n\n    url: str = Field(\n        title=\"Webhook URL\",\n        description=\"The webhook URL.\",\n        example=\"https://hooks.slack.com/XXX\",\n    )\n\n    method: Literal[\"GET\", \"POST\", \"PUT\", \"PATCH\", \"DELETE\"] = Field(\n        default=\"POST\", description=\"The webhook request method. Defaults to `POST`.\"\n    )\n\n    params: Optional[Dict[str, str]] = Field(\n        default=None, title=\"Query Params\", description=\"Custom query params.\"\n    )\n    json_data: Optional[dict] = Field(\n        default=None,\n        title=\"JSON Data\",\n        description=\"Send json data as payload.\",\n        example=(\n            '{\"text\": \"{{subject}}\\\\n{{body}}\", \"title\": \"{{name}}\", \"token\":'\n            ' \"{{tokenFromSecrets}}\"}'\n        ),\n    )\n    form_data: Optional[Dict[str, str]] = Field(\n        default=None,\n        title=\"Form Data\",\n        description=(\n            \"Send form data as payload. Should not be used together with _JSON Data_.\"\n        ),\n        example=(\n            '{\"text\": \"{{subject}}\\\\n{{body}}\", \"title\": \"{{name}}\", \"token\":'\n            ' \"{{tokenFromSecrets}}\"}'\n        ),\n    )\n\n    headers: Optional[Dict[str, str]] = Field(None, description=\"Custom headers.\")\n    cookies: Optional[Dict[str, str]] = Field(None, description=\"Custom cookies.\")\n\n    timeout: float = Field(\n        default=10, description=\"Request timeout in seconds. Defaults to 10.\"\n    )\n\n    secrets: SecretDict = Field(\n        default_factory=lambda: SecretDict(dict()),\n        title=\"Custom Secret Values\",\n        description=\"A dictionary of secret values to be substituted in other configs.\",\n        example='{\"tokenFromSecrets\":\"SomeSecretToken\"}',\n    )\n\n    def _build_request_args(self, body: str, subject: Optional[str]):\n        \"\"\"Build kwargs for httpx.AsyncClient.request\"\"\"\n        # prepare values\n        values = self.secrets.get_secret_value()\n        # use 'null' when subject is None\n        values.update(\n            {\n                \"subject\": \"null\" if subject is None else subject,\n                \"body\": body,\n                \"name\": self.name,\n            }\n        )\n        # do substution\n        return apply_values(\n            {\n                \"method\": self.method,\n                \"url\": self.url,\n                \"params\": self.params,\n                \"data\": self.form_data,\n                \"json\": self.json_data,\n                \"headers\": self.headers,\n                \"cookies\": self.cookies,\n                \"timeout\": self.timeout,\n            },\n            values,\n        )\n\n    def block_initialization(self) -> None:\n        # check form_data and json_data\n        if self.form_data is not None and self.json_data is not None:\n            raise ValueError(\"both `Form Data` and `JSON Data` provided\")\n        allowed_keys = {\"subject\", \"body\", \"name\"}.union(\n            self.secrets.get_secret_value().keys()\n        )\n        # test template to raise a error early\n        for name in [\"url\", \"params\", \"form_data\", \"json_data\", \"headers\", \"cookies\"]:\n            template = getattr(self, name)\n            if template is None:\n                continue\n            # check for placeholders not in predefined keys and secrets\n            placeholders = find_placeholders(template)\n            for placeholder in placeholders:\n                if placeholder.name not in allowed_keys:\n                    raise KeyError(f\"{name}/{placeholder}\")\n\n    @sync_compatible\n    @instrument_instance_method_call()\n    async def notify(self, body: str, subject: Optional[str] = None):\n        import httpx\n\n        # make request with httpx\n        client = httpx.AsyncClient(headers={\"user-agent\": \"Prefect Notifications\"})\n        resp = await client.request(**self._build_request_args(body, subject))\n        resp.raise_for_status()\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.DiscordWebhook","title":"DiscordWebhook","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via a provided Discord webhook. See Apprise notify_Discord docs # noqa

    Examples:

    Load a saved Discord webhook and send a message:

    from prefect.blocks.notifications import DiscordWebhook\n\ndiscord_webhook_block = DiscordWebhook.load(\"BLOCK_NAME\")\n\ndiscord_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class DiscordWebhook(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Discord webhook.\n    See [Apprise notify_Discord docs](https://github.com/caronc/apprise/wiki/Notify_Discord) # noqa\n\n    Examples:\n        Load a saved Discord webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import DiscordWebhook\n\n        discord_webhook_block = DiscordWebhook.load(\"BLOCK_NAME\")\n\n        discord_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via a provided Discord webhook.\"\n    _block_type_name = \"Discord Webhook\"\n    _block_type_slug = \"discord-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/9e94976c80ef925b66d24e5d14f0d47baa6b8f88-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.DiscordWebhook\"\n\n    webhook_id: SecretStr = Field(\n        default=...,\n        description=(\n            \"The first part of 2 tokens provided to you after creating a\"\n            \" incoming-webhook.\"\n        ),\n    )\n\n    webhook_token: SecretStr = Field(\n        default=...,\n        description=(\n            \"The second part of 2 tokens provided to you after creating a\"\n            \" incoming-webhook.\"\n        ),\n    )\n\n    botname: Optional[str] = Field(\n        title=\"Bot name\",\n        default=None,\n        description=(\n            \"Identify the name of the bot that should issue the message. If one isn't\"\n            \" specified then the default is to just use your account (associated with\"\n            \" the incoming-webhook).\"\n        ),\n    )\n\n    tts: bool = Field(\n        default=False,\n        description=\"Whether to enable Text-To-Speech.\",\n    )\n\n    include_image: bool = Field(\n        default=False,\n        description=(\n            \"Whether to include an image in-line with the message describing the\"\n            \" notification type.\"\n        ),\n    )\n\n    avatar: bool = Field(\n        default=False,\n        description=\"Whether to override the default discord avatar icon.\",\n    )\n\n    avatar_url: Optional[str] = Field(\n        title=\"Avatar URL\",\n        default=False,\n        description=(\n            \"Over-ride the default discord avatar icon URL. By default this is not set\"\n            \" and Apprise chooses the URL dynamically based on the type of message\"\n            \" (info, success, warning, or error).\"\n        ),\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyDiscord import NotifyDiscord\n\n        url = SecretStr(\n            NotifyDiscord(\n                webhook_id=self.webhook_id.get_secret_value(),\n                webhook_token=self.webhook_token.get_secret_value(),\n                botname=self.botname,\n                tts=self.tts,\n                include_image=self.include_image,\n                avatar=self.avatar,\n                avatar_url=self.avatar_url,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.MattermostWebhook","title":"MattermostWebhook","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via a provided Mattermost webhook. See Apprise notify_Mattermost docs # noqa

    Examples:

    Load a saved Mattermost webhook and send a message:

    from prefect.blocks.notifications import MattermostWebhook\n\nmattermost_webhook_block = MattermostWebhook.load(\"BLOCK_NAME\")\n\nmattermost_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class MattermostWebhook(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Mattermost webhook.\n    See [Apprise notify_Mattermost docs](https://github.com/caronc/apprise/wiki/Notify_Mattermost) # noqa\n\n\n    Examples:\n        Load a saved Mattermost webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import MattermostWebhook\n\n        mattermost_webhook_block = MattermostWebhook.load(\"BLOCK_NAME\")\n\n        mattermost_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via a provided Mattermost webhook.\"\n    _block_type_name = \"Mattermost Webhook\"\n    _block_type_slug = \"mattermost-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/1350a147130bf82cbc799a5f868d2c0116207736-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.MattermostWebhook\"\n\n    hostname: str = Field(\n        default=...,\n        description=\"The hostname of your Mattermost server.\",\n        example=\"Mattermost.example.com\",\n    )\n\n    token: SecretStr = Field(\n        default=...,\n        description=\"The token associated with your Mattermost webhook.\",\n    )\n\n    botname: Optional[str] = Field(\n        title=\"Bot name\",\n        default=None,\n        description=\"The name of the bot that will send the message.\",\n    )\n\n    channels: Optional[List[str]] = Field(\n        default=None,\n        description=\"The channel(s) you wish to notify.\",\n    )\n\n    include_image: bool = Field(\n        default=False,\n        description=\"Whether to include the Apprise status image in the message.\",\n    )\n\n    path: Optional[str] = Field(\n        default=None,\n        description=\"An optional sub-path specification to append to the hostname.\",\n    )\n\n    port: int = Field(\n        default=8065,\n        description=\"The port of your Mattermost server.\",\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyMattermost import NotifyMattermost\n\n        url = SecretStr(\n            NotifyMattermost(\n                token=self.token.get_secret_value(),\n                fullpath=self.path,\n                host=self.hostname,\n                botname=self.botname,\n                channels=self.channels,\n                include_image=self.include_image,\n                port=self.port,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.MicrosoftTeamsWebhook","title":"MicrosoftTeamsWebhook","text":"

    Bases: AppriseNotificationBlock

    Enables sending notifications via a provided Microsoft Teams webhook.

    Examples:

    Load a saved Teams webhook and send a message:

    from prefect.blocks.notifications import MicrosoftTeamsWebhook\nteams_webhook_block = MicrosoftTeamsWebhook.load(\"BLOCK_NAME\")\nteams_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class MicrosoftTeamsWebhook(AppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Microsoft Teams webhook.\n\n    Examples:\n        Load a saved Teams webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import MicrosoftTeamsWebhook\n        teams_webhook_block = MicrosoftTeamsWebhook.load(\"BLOCK_NAME\")\n        teams_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Microsoft Teams Webhook\"\n    _block_type_slug = \"ms-teams-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/817efe008a57f0a24f3587414714b563e5e23658-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.MicrosoftTeamsWebhook\"\n\n    url: SecretStr = Field(\n        ...,\n        title=\"Webhook URL\",\n        description=\"The Teams incoming webhook URL used to send notifications.\",\n        example=(\n            \"https://your-org.webhook.office.com/webhookb2/XXX/IncomingWebhook/YYY/ZZZ\"\n        ),\n    )\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.OpsgenieWebhook","title":"OpsgenieWebhook","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via a provided Opsgenie webhook. See Apprise notify_opsgenie docs for more info on formatting the URL.

    Examples:

    Load a saved Opsgenie webhook and send a message:

    from prefect.blocks.notifications import OpsgenieWebhook\nopsgenie_webhook_block = OpsgenieWebhook.load(\"BLOCK_NAME\")\nopsgenie_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class OpsgenieWebhook(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Opsgenie webhook.\n    See [Apprise notify_opsgenie docs](https://github.com/caronc/apprise/wiki/Notify_opsgenie)\n    for more info on formatting the URL.\n\n    Examples:\n        Load a saved Opsgenie webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import OpsgenieWebhook\n        opsgenie_webhook_block = OpsgenieWebhook.load(\"BLOCK_NAME\")\n        opsgenie_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via a provided Opsgenie webhook.\"\n\n    _block_type_name = \"Opsgenie Webhook\"\n    _block_type_slug = \"opsgenie-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/d8b5bc6244ae6cd83b62ec42f10d96e14d6e9113-280x280.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.OpsgenieWebhook\"\n\n    apikey: SecretStr = Field(\n        default=...,\n        title=\"API Key\",\n        description=\"The API Key associated with your Opsgenie account.\",\n    )\n\n    target_user: Optional[List] = Field(\n        default=None, description=\"The user(s) you wish to notify.\"\n    )\n\n    target_team: Optional[List] = Field(\n        default=None, description=\"The team(s) you wish to notify.\"\n    )\n\n    target_schedule: Optional[List] = Field(\n        default=None, description=\"The schedule(s) you wish to notify.\"\n    )\n\n    target_escalation: Optional[List] = Field(\n        default=None, description=\"The escalation(s) you wish to notify.\"\n    )\n\n    region_name: Literal[\"us\", \"eu\"] = Field(\n        default=\"us\", description=\"The 2-character region code.\"\n    )\n\n    batch: bool = Field(\n        default=False,\n        description=\"Notify all targets in batches (instead of individually).\",\n    )\n\n    tags: Optional[List] = Field(\n        default=None,\n        description=(\n            \"A comma-separated list of tags you can associate with your Opsgenie\"\n            \" message.\"\n        ),\n        example='[\"tag1\", \"tag2\"]',\n    )\n\n    priority: Optional[str] = Field(\n        default=3,\n        description=(\n            \"The priority to associate with the message. It is on a scale between 1\"\n            \" (LOW) and 5 (EMERGENCY).\"\n        ),\n    )\n\n    alias: Optional[str] = Field(\n        default=None, description=\"The alias to associate with the message.\"\n    )\n\n    entity: Optional[str] = Field(\n        default=None, description=\"The entity to associate with the message.\"\n    )\n\n    details: Optional[Dict[str, str]] = Field(\n        default=None,\n        description=\"Additional details composed of key/values pairs.\",\n        example='{\"key1\": \"value1\", \"key2\": \"value2\"}',\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyOpsgenie import NotifyOpsgenie\n\n        targets = []\n        if self.target_user:\n            [targets.append(f\"@{x}\") for x in self.target_user]\n        if self.target_team:\n            [targets.append(f\"#{x}\") for x in self.target_team]\n        if self.target_schedule:\n            [targets.append(f\"*{x}\") for x in self.target_schedule]\n        if self.target_escalation:\n            [targets.append(f\"^{x}\") for x in self.target_escalation]\n        url = SecretStr(\n            NotifyOpsgenie(\n                apikey=self.apikey.get_secret_value(),\n                targets=targets,\n                region_name=self.region_name,\n                details=self.details,\n                priority=self.priority,\n                alias=self.alias,\n                entity=self.entity,\n                batch=self.batch,\n                tags=self.tags,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.PagerDutyWebHook","title":"PagerDutyWebHook","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via a provided PagerDuty webhook. See Apprise notify_pagerduty docs for more info on formatting the URL.

    Examples:

    Load a saved PagerDuty webhook and send a message:

    from prefect.blocks.notifications import PagerDutyWebHook\npagerduty_webhook_block = PagerDutyWebHook.load(\"BLOCK_NAME\")\npagerduty_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class PagerDutyWebHook(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided PagerDuty webhook.\n    See [Apprise notify_pagerduty docs](https://github.com/caronc/apprise/wiki/Notify_pagerduty)\n    for more info on formatting the URL.\n\n    Examples:\n        Load a saved PagerDuty webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import PagerDutyWebHook\n        pagerduty_webhook_block = PagerDutyWebHook.load(\"BLOCK_NAME\")\n        pagerduty_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via a provided PagerDuty webhook.\"\n\n    _block_type_name = \"Pager Duty Webhook\"\n    _block_type_slug = \"pager-duty-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/8dbf37d17089c1ce531708eac2e510801f7b3aee-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.PagerDutyWebHook\"\n\n    # The default cannot be prefect_default because NotifyPagerDuty's\n    # PAGERDUTY_SEVERITY_MAP only has these notify types defined as keys\n    notify_type: Literal[\"info\", \"success\", \"warning\", \"failure\"] = Field(\n        default=\"info\", description=\"The severity of the notification.\"\n    )\n\n    integration_key: SecretStr = Field(\n        default=...,\n        description=(\n            \"This can be found on the Events API V2 \"\n            \"integration's detail page, and is also referred to as a Routing Key. \"\n            \"This must be provided alongside `api_key`, but will error if provided \"\n            \"alongside `url`.\"\n        ),\n    )\n\n    api_key: SecretStr = Field(\n        default=...,\n        title=\"API Key\",\n        description=(\n            \"This can be found under Integrations. \"\n            \"This must be provided alongside `integration_key`, but will error if \"\n            \"provided alongside `url`.\"\n        ),\n    )\n\n    source: Optional[str] = Field(\n        default=\"Prefect\", description=\"The source string as part of the payload.\"\n    )\n\n    component: str = Field(\n        default=\"Notification\",\n        description=\"The component string as part of the payload.\",\n    )\n\n    group: Optional[str] = Field(\n        default=None, description=\"The group string as part of the payload.\"\n    )\n\n    class_id: Optional[str] = Field(\n        default=None,\n        title=\"Class ID\",\n        description=\"The class string as part of the payload.\",\n    )\n\n    region_name: Literal[\"us\", \"eu\"] = Field(\n        default=\"us\", description=\"The region name.\"\n    )\n\n    clickable_url: Optional[AnyHttpUrl] = Field(\n        default=None,\n        title=\"Clickable URL\",\n        description=\"A clickable URL to associate with the notice.\",\n    )\n\n    include_image: bool = Field(\n        default=True,\n        description=\"Associate the notification status via a represented icon.\",\n    )\n\n    custom_details: Optional[Dict[str, str]] = Field(\n        default=None,\n        description=\"Additional details to include as part of the payload.\",\n        example='{\"disk_space_left\": \"145GB\"}',\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyPagerDuty import NotifyPagerDuty\n\n        url = SecretStr(\n            NotifyPagerDuty(\n                apikey=self.api_key.get_secret_value(),\n                integrationkey=self.integration_key.get_secret_value(),\n                source=self.source,\n                component=self.component,\n                group=self.group,\n                class_id=self.class_id,\n                region_name=self.region_name,\n                click=self.clickable_url,\n                include_image=self.include_image,\n                details=self.custom_details,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.SendgridEmail","title":"SendgridEmail","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via any sendgrid account. See Apprise Notify_sendgrid docs

    Examples:

    Load a saved Sendgrid and send a email message: ```python from prefect.blocks.notifications import SendgridEmail

    sendgrid_block = SendgridEmail.load(\"BLOCK_NAME\")

    sendgrid_block.notify(\"Hello from Prefect!\")

    Source code in prefect/blocks/notifications.py
    class SendgridEmail(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via any sendgrid account.\n    See [Apprise Notify_sendgrid docs](https://github.com/caronc/apprise/wiki/Notify_Sendgrid)\n\n    Examples:\n        Load a saved Sendgrid and send a email message:\n        ```python\n        from prefect.blocks.notifications import SendgridEmail\n\n        sendgrid_block = SendgridEmail.load(\"BLOCK_NAME\")\n\n        sendgrid_block.notify(\"Hello from Prefect!\")\n    \"\"\"\n\n    _description = \"Enables sending notifications via Sendgrid email service.\"\n    _block_type_name = \"Sendgrid Email\"\n    _block_type_slug = \"sendgrid-email\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/82bc6ed16ca42a2252a5512c72233a253b8a58eb-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.SendgridEmail\"\n\n    api_key: SecretStr = Field(\n        default=...,\n        title=\"API Key\",\n        description=\"The API Key associated with your sendgrid account.\",\n    )\n\n    sender_email: str = Field(\n        title=\"Sender email id\",\n        description=\"The sender email id.\",\n        example=\"test-support@gmail.com\",\n    )\n\n    to_emails: List[str] = Field(\n        default=...,\n        title=\"Recipient emails\",\n        description=\"Email ids of all recipients.\",\n        example='\"recipient1@gmail.com\"',\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifySendGrid import NotifySendGrid\n\n        url = SecretStr(\n            NotifySendGrid(\n                apikey=self.api_key.get_secret_value(),\n                from_email=self.sender_email,\n                targets=self.to_emails,\n            ).url()\n        )\n\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.SlackWebhook","title":"SlackWebhook","text":"

    Bases: AppriseNotificationBlock

    Enables sending notifications via a provided Slack webhook.

    Examples:

    Load a saved Slack webhook and send a message:

    from prefect.blocks.notifications import SlackWebhook\n\nslack_webhook_block = SlackWebhook.load(\"BLOCK_NAME\")\nslack_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class SlackWebhook(AppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Slack webhook.\n\n    Examples:\n        Load a saved Slack webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import SlackWebhook\n\n        slack_webhook_block = SlackWebhook.load(\"BLOCK_NAME\")\n        slack_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Slack Webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c1965ecbf8704ee1ea20d77786de9a41ce1087d1-500x500.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.SlackWebhook\"\n\n    url: SecretStr = Field(\n        default=...,\n        title=\"Webhook URL\",\n        description=\"Slack incoming webhook URL used to send notifications.\",\n        example=\"https://hooks.slack.com/XXX\",\n    )\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.TwilioSMS","title":"TwilioSMS","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via Twilio SMS. Find more on sending Twilio SMS messages in the docs.

    Examples:

    Load a saved TwilioSMS block and send a message:

    from prefect.blocks.notifications import TwilioSMS\ntwilio_webhook_block = TwilioSMS.load(\"BLOCK_NAME\")\ntwilio_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class TwilioSMS(AbstractAppriseNotificationBlock):\n    \"\"\"Enables sending notifications via Twilio SMS.\n    Find more on sending Twilio SMS messages in the [docs](https://www.twilio.com/docs/sms).\n\n    Examples:\n        Load a saved `TwilioSMS` block and send a message:\n        ```python\n        from prefect.blocks.notifications import TwilioSMS\n        twilio_webhook_block = TwilioSMS.load(\"BLOCK_NAME\")\n        twilio_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via Twilio SMS.\"\n    _block_type_name = \"Twilio SMS\"\n    _block_type_slug = \"twilio-sms\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/8bd8777999f82112c09b9c8d57083ac75a4a0d65-250x250.png\"  # noqa\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.TwilioSMS\"\n\n    account_sid: str = Field(\n        default=...,\n        description=(\n            \"The Twilio Account SID - it can be found on the homepage \"\n            \"of the Twilio console.\"\n        ),\n    )\n\n    auth_token: SecretStr = Field(\n        default=...,\n        description=(\n            \"The Twilio Authentication Token - \"\n            \"it can be found on the homepage of the Twilio console.\"\n        ),\n    )\n\n    from_phone_number: str = Field(\n        default=...,\n        description=\"The valid Twilio phone number to send the message from.\",\n        example=\"18001234567\",\n    )\n\n    to_phone_numbers: List[str] = Field(\n        default=...,\n        description=\"A list of valid Twilio phone number(s) to send the message to.\",\n        # not wrapped in brackets because of the way UI displays examples; in code should be [\"18004242424\"]\n        example=\"18004242424\",\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyTwilio import NotifyTwilio\n\n        url = SecretStr(\n            NotifyTwilio(\n                account_sid=self.account_sid,\n                auth_token=self.auth_token.get_secret_value(),\n                source=self.from_phone_number,\n                targets=self.to_phone_numbers,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/system/","title":"system","text":"","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system","title":"prefect.blocks.system","text":"","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system.DateTime","title":"DateTime","text":"

    Bases: Block

    A block that represents a datetime

    Attributes:

    Name Type Description value DateTime

    An ISO 8601-compatible datetime value.

    Example

    Load a stored JSON value:

    from prefect.blocks.system import DateTime\n\ndata_time_block = DateTime.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/system.py
    class DateTime(Block):\n    \"\"\"\n    A block that represents a datetime\n\n    Attributes:\n        value: An ISO 8601-compatible datetime value.\n\n    Example:\n        Load a stored JSON value:\n        ```python\n        from prefect.blocks.system import DateTime\n\n        data_time_block = DateTime.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Date Time\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/8b3da9a6621e92108b8e6a75b82e15374e170ff7-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/system/#prefect.blocks.system.DateTime\"\n\n    value: pendulum.DateTime = Field(\n        default=...,\n        description=\"An ISO 8601-compatible datetime value.\",\n    )\n
    ","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system.JSON","title":"JSON","text":"

    Bases: Block

    A block that represents JSON

    Attributes:

    Name Type Description value Any

    A JSON-compatible value.

    Example

    Load a stored JSON value:

    from prefect.blocks.system import JSON\n\njson_block = JSON.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/system.py
    class JSON(Block):\n    \"\"\"\n    A block that represents JSON\n\n    Attributes:\n        value: A JSON-compatible value.\n\n    Example:\n        Load a stored JSON value:\n        ```python\n        from prefect.blocks.system import JSON\n\n        json_block = JSON.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/4fcef2294b6eeb423b1332d1ece5156bf296ff96-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/system/#prefect.blocks.system.JSON\"\n\n    value: Any = Field(default=..., description=\"A JSON-compatible value.\")\n
    ","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system.Secret","title":"Secret","text":"

    Bases: Block

    A block that represents a secret value. The value stored in this block will be obfuscated when this block is logged or shown in the UI.

    Attributes:

    Name Type Description value SecretStr

    A string value that should be kept secret.

    Example
    from prefect.blocks.system import Secret\n\nsecret_block = Secret.load(\"BLOCK_NAME\")\n\n# Access the stored secret\nsecret_block.get()\n
    Source code in prefect/blocks/system.py
    class Secret(Block):\n    \"\"\"\n    A block that represents a secret value. The value stored in this block will be obfuscated when\n    this block is logged or shown in the UI.\n\n    Attributes:\n        value: A string value that should be kept secret.\n\n    Example:\n        ```python\n        from prefect.blocks.system import Secret\n\n        secret_block = Secret.load(\"BLOCK_NAME\")\n\n        # Access the stored secret\n        secret_block.get()\n        ```\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c6f20e556dd16effda9df16551feecfb5822092b-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/system/#prefect.blocks.system.Secret\"\n\n    value: SecretStr = Field(\n        default=..., description=\"A string value that should be kept secret.\"\n    )\n\n    def get(self):\n        return self.value.get_secret_value()\n
    ","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system.String","title":"String","text":"

    Bases: Block

    A block that represents a string

    Attributes:

    Name Type Description value str

    A string value.

    Example

    Load a stored string value:

    from prefect.blocks.system import String\n\nstring_block = String.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/system.py
    class String(Block):\n    \"\"\"\n    A block that represents a string\n\n    Attributes:\n        value: A string value.\n\n    Example:\n        Load a stored string value:\n        ```python\n        from prefect.blocks.system import String\n\n        string_block = String.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c262ea2c80a2c043564e8763f3370c3db5a6b3e6-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/system/#prefect.blocks.system.String\"\n\n    value: str = Field(default=..., description=\"A string value.\")\n
    ","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/webhook/","title":"webhook","text":"","tags":["Python API","blocks","webhook"]},{"location":"api-ref/prefect/blocks/webhook/#prefect.blocks.webhook","title":"prefect.blocks.webhook","text":"","tags":["Python API","blocks","webhook"]},{"location":"api-ref/prefect/blocks/webhook/#prefect.blocks.webhook.Webhook","title":"Webhook","text":"

    Bases: Block

    Block that enables calling webhooks.

    Source code in prefect/blocks/webhook.py
    class Webhook(Block):\n    \"\"\"\n    Block that enables calling webhooks.\n    \"\"\"\n\n    _block_type_name = \"Webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c7247cb359eb6cf276734d4b1fbf00fb8930e89e-250x250.png\"  # type: ignore\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/webhook/#prefect.blocks.webhook.Webhook\"\n\n    method: Literal[\"GET\", \"POST\", \"PUT\", \"PATCH\", \"DELETE\"] = Field(\n        default=\"POST\", description=\"The webhook request method. Defaults to `POST`.\"\n    )\n\n    url: SecretStr = Field(\n        default=...,\n        title=\"Webhook URL\",\n        description=\"The webhook URL.\",\n        example=\"https://hooks.slack.com/XXX\",\n    )\n\n    headers: SecretDict = Field(\n        default_factory=lambda: SecretDict(dict()),\n        title=\"Webhook Headers\",\n        description=\"A dictionary of headers to send with the webhook request.\",\n    )\n\n    def block_initialization(self):\n        self._client = AsyncClient(transport=_http_transport)\n\n    async def call(self, payload: Optional[dict] = None) -> Response:\n        \"\"\"\n        Call the webhook.\n\n        Args:\n            payload: an optional payload to send when calling the webhook.\n        \"\"\"\n        async with self._client:\n            return await self._client.request(\n                method=self.method,\n                url=self.url.get_secret_value(),\n                headers=self.headers.get_secret_value(),\n                json=payload,\n            )\n
    ","tags":["Python API","blocks","webhook"]},{"location":"api-ref/prefect/blocks/webhook/#prefect.blocks.webhook.Webhook.call","title":"call async","text":"

    Call the webhook.

    Parameters:

    Name Type Description Default payload Optional[dict]

    an optional payload to send when calling the webhook.

    None Source code in prefect/blocks/webhook.py
    async def call(self, payload: Optional[dict] = None) -> Response:\n    \"\"\"\n    Call the webhook.\n\n    Args:\n        payload: an optional payload to send when calling the webhook.\n    \"\"\"\n    async with self._client:\n        return await self._client.request(\n            method=self.method,\n            url=self.url.get_secret_value(),\n            headers=self.headers.get_secret_value(),\n            json=payload,\n        )\n
    ","tags":["Python API","blocks","webhook"]},{"location":"api-ref/prefect/cli/agent/","title":"agent","text":"","tags":["Python API","agents","CLI"]},{"location":"api-ref/prefect/cli/agent/#prefect.cli.agent","title":"prefect.cli.agent","text":"

    Command line interface for working with agent services

    ","tags":["Python API","agents","CLI"]},{"location":"api-ref/prefect/cli/agent/#prefect.cli.agent.start","title":"start async","text":"

    Start an agent process to poll one or more work queues for flow runs.

    Source code in prefect/cli/agent.py
    @agent_app.command()\nasync def start(\n    # deprecated main argument\n    work_queue: str = typer.Argument(\n        None,\n        show_default=False,\n        help=\"DEPRECATED: A work queue name or ID\",\n    ),\n    work_queues: List[str] = typer.Option(\n        None,\n        \"-q\",\n        \"--work-queue\",\n        help=\"One or more work queue names for the agent to pull from.\",\n    ),\n    work_queue_prefix: List[str] = typer.Option(\n        None,\n        \"-m\",\n        \"--match\",\n        help=(\n            \"Dynamically matches work queue names with the specified prefix for the\"\n            \" agent to pull from,for example `dev-` will match all work queues with a\"\n            \" name that starts with `dev-`\"\n        ),\n    ),\n    work_pool_name: str = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"A work pool name for the agent to pull from.\",\n    ),\n    hide_welcome: bool = typer.Option(False, \"--hide-welcome\"),\n    api: str = SettingsOption(PREFECT_API_URL),\n    run_once: bool = typer.Option(\n        False, help=\"Run the agent loop once, instead of forever.\"\n    ),\n    prefetch_seconds: int = SettingsOption(PREFECT_AGENT_PREFETCH_SECONDS),\n    # deprecated tags\n    tags: List[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=(\n            \"DEPRECATED: One or more optional tags that will be used to create a work\"\n            \" queue. This option will be removed on 2023-02-23.\"\n        ),\n    ),\n    limit: int = typer.Option(\n        None,\n        \"-l\",\n        \"--limit\",\n        help=\"Maximum number of flow runs to start simultaneously.\",\n    ),\n):\n    \"\"\"\n    Start an agent process to poll one or more work queues for flow runs.\n    \"\"\"\n    work_queues = work_queues or []\n\n    if work_queue is not None:\n        # try to treat the work_queue as a UUID\n        try:\n            async with get_client() as client:\n                q = await client.read_work_queue(UUID(work_queue))\n                work_queue = q.name\n        # otherwise treat it as a string name\n        except (TypeError, ValueError):\n            pass\n        work_queues.append(work_queue)\n        app.console.print(\n            (\n                \"Agents now support multiple work queues. Instead of passing a single\"\n                \" argument, provide work queue names with the `-q` or `--work-queue`\"\n                f\" flag: `prefect agent start -q {work_queue}`\\n\"\n            ),\n            style=\"blue\",\n        )\n\n    if not work_queues and not tags and not work_queue_prefix and not work_pool_name:\n        exit_with_error(\"No work queues provided!\", style=\"red\")\n    elif bool(work_queues) + bool(tags) + bool(work_queue_prefix) > 1:\n        exit_with_error(\n            \"Only one of `work_queues`, `match`, or `tags` can be provided.\",\n            style=\"red\",\n        )\n    if work_pool_name and tags:\n        exit_with_error(\n            \"`tag` and `pool` options cannot be used together.\", style=\"red\"\n        )\n\n    if tags:\n        work_queue_name = f\"Agent queue {'-'.join(sorted(tags))}\"\n        app.console.print(\n            (\n                \"`tags` are deprecated. For backwards-compatibility with old versions\"\n                \" of Prefect, this agent will create a work queue named\"\n                f\" `{work_queue_name}` that uses legacy tag-based matching. This option\"\n                \" will be removed on 2023-02-23.\"\n            ),\n            style=\"red\",\n        )\n\n        async with get_client() as client:\n            try:\n                work_queue = await client.read_work_queue_by_name(work_queue_name)\n                if work_queue.filter is None:\n                    # ensure the work queue has legacy (deprecated) tag-based behavior\n                    await client.update_work_queue(filter=dict(tags=tags))\n            except ObjectNotFound:\n                # if the work queue doesn't already exist, we create it with tags\n                # to enable legacy (deprecated) tag-matching behavior\n                await client.create_work_queue(name=work_queue_name, tags=tags)\n\n        work_queues = [work_queue_name]\n\n    if not hide_welcome:\n        if api:\n            app.console.print(\n                f\"Starting v{prefect.__version__} agent connected to {api}...\"\n            )\n        else:\n            app.console.print(\n                f\"Starting v{prefect.__version__} agent with ephemeral API...\"\n            )\n\n    agent_process_id = os.getpid()\n    setup_signal_handlers_agent(\n        agent_process_id, \"the Prefect agent\", app.console.print\n    )\n\n    async with PrefectAgent(\n        work_queues=work_queues,\n        work_queue_prefix=work_queue_prefix,\n        work_pool_name=work_pool_name,\n        prefetch_seconds=prefetch_seconds,\n        limit=limit,\n    ) as agent:\n        if not hide_welcome:\n            app.console.print(ascii_name)\n            if work_pool_name:\n                app.console.print(\n                    \"Agent started! Looking for work from \"\n                    f\"work pool '{work_pool_name}'...\"\n                )\n            elif work_queue_prefix:\n                app.console.print(\n                    \"Agent started! Looking for work from \"\n                    f\"queue(s) that start with the prefix: {work_queue_prefix}...\"\n                )\n            else:\n                app.console.print(\n                    \"Agent started! Looking for work from \"\n                    f\"queue(s): {', '.join(work_queues)}...\"\n                )\n\n        async with anyio.create_task_group() as tg:\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    agent.get_and_submit_flow_runs,\n                    PREFECT_AGENT_QUERY_INTERVAL.value(),\n                    printer=app.console.print,\n                    run_once=run_once,\n                    jitter_range=0.3,\n                    backoff=4,  # Up to ~1 minute interval during backoff\n                )\n            )\n\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    agent.check_for_cancelled_flow_runs,\n                    PREFECT_AGENT_QUERY_INTERVAL.value() * 2,\n                    printer=app.console.print,\n                    run_once=run_once,\n                    jitter_range=0.3,\n                    backoff=4,\n                )\n            )\n\n    app.console.print(\"Agent stopped!\")\n
    ","tags":["Python API","agents","CLI"]},{"location":"api-ref/prefect/cli/artifact/","title":"artifact","text":"","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/artifact/#prefect.cli.artifact","title":"prefect.cli.artifact","text":"","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/artifact/#prefect.cli.artifact.delete","title":"delete async","text":"

    Delete an artifact.

    Parameters:

    Name Type Description Default key Optional[str]

    the key of the artifact to delete

    Argument(None, help='The key of the artifact to delete.')

    Examples:

    $ prefect artifact delete \"my-artifact\"

    Source code in prefect/cli/artifact.py
    @artifact_app.command(\"delete\")\nasync def delete(\n    key: Optional[str] = typer.Argument(\n        None, help=\"The key of the artifact to delete.\"\n    ),\n    artifact_id: Optional[str] = typer.Option(\n        None, \"--id\", help=\"The ID of the artifact to delete.\"\n    ),\n):\n    \"\"\"\n    Delete an artifact.\n\n    Arguments:\n        key: the key of the artifact to delete\n\n    Examples:\n        $ prefect artifact delete \"my-artifact\"\n    \"\"\"\n    if key and artifact_id:\n        exit_with_error(\"Please provide either a key or an artifact_id but not both.\")\n\n    async with get_client() as client:\n        if artifact_id is not None:\n            try:\n                confirm_delete = typer.confirm(\n                    (\n                        \"Are you sure you want to delete artifact with id\"\n                        f\" {artifact_id!r}?\"\n                    ),\n                    default=False,\n                )\n                if not confirm_delete:\n                    exit_with_error(\"Deletion aborted.\")\n\n                await client.delete_artifact(artifact_id)\n                exit_with_success(f\"Deleted artifact with id {artifact_id!r}.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Artifact with id {artifact_id!r} not found!\")\n\n        elif key is not None:\n            artifacts = await client.read_artifacts(\n                artifact_filter=ArtifactFilter(key=ArtifactFilterKey(any_=[key])),\n            )\n            if not artifacts:\n                exit_with_error(\n                    f\"Artifact with key {key!r} not found. You can also specify an\"\n                    \" artifact id with the --id flag.\"\n                )\n\n            confirm_delete = typer.confirm(\n                (\n                    f\"Are you sure you want to delete {len(artifacts)} artifact(s) with\"\n                    f\" key {key!r}?\"\n                ),\n                default=False,\n            )\n            if not confirm_delete:\n                exit_with_error(\"Deletion aborted.\")\n\n            for a in artifacts:\n                await client.delete_artifact(a.id)\n\n            exit_with_success(f\"Deleted {len(artifacts)} artifact(s) with key {key!r}.\")\n\n        else:\n            exit_with_error(\"Please provide a key or an artifact_id.\")\n
    ","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/artifact/#prefect.cli.artifact.inspect","title":"inspect async","text":"
    View details about an artifact.\n\nArguments:\n    key: the key of the artifact to inspect\n\nExamples:\n    $ prefect artifact inspect \"my-artifact\"\n   [\n    {\n        'id': 'ba1d67be-0bd7-452e-8110-247fe5e6d8cc',\n        'created': '2023-03-21T21:40:09.895910+00:00',\n        'updated': '2023-03-21T21:40:09.895910+00:00',\n        'key': 'my-artifact',\n        'type': 'markdown',\n        'description': None,\n        'data': 'my markdown',\n        'metadata_': None,\n        'flow_run_id': '8dc54b6f-6e24-4586-a05c-e98c6490cb98',\n        'task_run_id': None\n    },\n    {\n        'id': '57f235b5-2576-45a5-bd93-c829c2900966',\n        'created': '2023-03-27T23:16:15.536434+00:00',\n        'updated': '2023-03-27T23:16:15.536434+00:00',\n        'key': 'my-artifact',\n        'type': 'markdown',\n        'description': 'my-artifact-description',\n        'data': 'my markdown',\n        'metadata_': None,\n        'flow_run_id': 'ffa91051-f249-48c1-ae0f-4754fcb7eb29',\n        'task_run_id': None\n    }\n

    ]

    Source code in prefect/cli/artifact.py
    @artifact_app.command(\"inspect\")\nasync def inspect(\n    key: str,\n    limit: int = typer.Option(\n        10,\n        \"--limit\",\n        help=\"The maximum number of artifacts to return.\",\n    ),\n):\n    \"\"\"\n        View details about an artifact.\n\n        Arguments:\n            key: the key of the artifact to inspect\n\n        Examples:\n            $ prefect artifact inspect \"my-artifact\"\n           [\n            {\n                'id': 'ba1d67be-0bd7-452e-8110-247fe5e6d8cc',\n                'created': '2023-03-21T21:40:09.895910+00:00',\n                'updated': '2023-03-21T21:40:09.895910+00:00',\n                'key': 'my-artifact',\n                'type': 'markdown',\n                'description': None,\n                'data': 'my markdown',\n                'metadata_': None,\n                'flow_run_id': '8dc54b6f-6e24-4586-a05c-e98c6490cb98',\n                'task_run_id': None\n            },\n            {\n                'id': '57f235b5-2576-45a5-bd93-c829c2900966',\n                'created': '2023-03-27T23:16:15.536434+00:00',\n                'updated': '2023-03-27T23:16:15.536434+00:00',\n                'key': 'my-artifact',\n                'type': 'markdown',\n                'description': 'my-artifact-description',\n                'data': 'my markdown',\n                'metadata_': None,\n                'flow_run_id': 'ffa91051-f249-48c1-ae0f-4754fcb7eb29',\n                'task_run_id': None\n            }\n    ]\n    \"\"\"\n\n    async with get_client() as client:\n        artifacts = await client.read_artifacts(\n            limit=limit,\n            sort=ArtifactSort.UPDATED_DESC,\n            artifact_filter=ArtifactFilter(key=ArtifactFilterKey(any_=[key])),\n        )\n        if not artifacts:\n            exit_with_error(f\"Artifact {key!r} not found.\")\n\n        artifacts = [a.dict(json_compatible=True) for a in artifacts]\n\n        app.console.print(Pretty(artifacts))\n
    ","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/artifact/#prefect.cli.artifact.list_artifacts","title":"list_artifacts async","text":"

    List artifacts.

    Source code in prefect/cli/artifact.py
    @artifact_app.command(\"ls\")\nasync def list_artifacts(\n    limit: int = typer.Option(\n        100,\n        \"--limit\",\n        help=\"The maximum number of artifacts to return.\",\n    ),\n    all: bool = typer.Option(\n        False,\n        \"--all\",\n        \"-a\",\n        help=\"Whether or not to only return the latest version of each artifact.\",\n    ),\n):\n    \"\"\"\n    List artifacts.\n    \"\"\"\n    table = Table(\n        title=\"Artifacts\",\n        caption=\"List Artifacts using `prefect artifact ls`\",\n        show_header=True,\n    )\n\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Key\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Type\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Updated\", style=\"blue\", no_wrap=True)\n\n    async with get_client() as client:\n        if all:\n            artifacts = await client.read_artifacts(\n                sort=ArtifactSort.KEY_ASC,\n                limit=limit,\n            )\n\n            for artifact in sorted(artifacts, key=lambda x: f\"{x.key}\"):\n                table.add_row(\n                    str(artifact.id),\n                    artifact.key,\n                    artifact.type,\n                    pendulum.instance(artifact.updated).diff_for_humans(),\n                )\n\n        else:\n            artifacts = await client.read_latest_artifacts(\n                sort=ArtifactCollectionSort.KEY_ASC,\n                limit=limit,\n            )\n\n            for artifact in sorted(artifacts, key=lambda x: f\"{x.key}\"):\n                table.add_row(\n                    str(artifact.latest_id),\n                    artifact.key,\n                    artifact.type,\n                    pendulum.instance(artifact.updated).diff_for_humans(),\n                )\n\n        app.console.print(table)\n
    ","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/block/","title":"block","text":"","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block","title":"prefect.cli.block","text":"

    Command line interface for working with blocks.

    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.block_create","title":"block_create async","text":"

    Generate a link to the Prefect UI to create a block.

    Source code in prefect/cli/block.py
    @blocks_app.command(\"create\")\nasync def block_create(\n    block_type_slug: str = typer.Argument(\n        ...,\n        help=\"A block type slug. View available types with: prefect block type ls\",\n        show_default=False,\n    ),\n):\n    \"\"\"\n    Generate a link to the Prefect UI to create a block.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            block_type = await client.read_block_type_by_slug(block_type_slug)\n        except ObjectNotFound:\n            app.console.print(f\"[red]Block type {block_type_slug!r} not found![/red]\")\n            block_types = await client.read_block_types()\n            slugs = {block_type.slug for block_type in block_types}\n            app.console.print(f\"Available block types: {', '.join(slugs)}\")\n            raise typer.Exit(1)\n\n        if not PREFECT_UI_URL:\n            exit_with_error(\n                \"Prefect must be configured to use a hosted Prefect server or \"\n                \"Prefect Cloud to display the Prefect UI\"\n            )\n\n        block_link = f\"{PREFECT_UI_URL.value()}/blocks/catalog/{block_type.slug}/create\"\n        app.console.print(\n            f\"Create a {block_type_slug} block: {block_link}\",\n        )\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.block_delete","title":"block_delete async","text":"

    Delete a configured block.

    Source code in prefect/cli/block.py
    @blocks_app.command(\"delete\")\nasync def block_delete(\n    slug: Optional[str] = typer.Argument(\n        None, help=\"A block slug. Formatted as '<BLOCK_TYPE_SLUG>/<BLOCK_NAME>'\"\n    ),\n    block_id: Optional[str] = typer.Option(None, \"--id\", help=\"A block id.\"),\n):\n    \"\"\"\n    Delete a configured block.\n    \"\"\"\n    async with get_client() as client:\n        if slug is None and block_id is not None:\n            try:\n                await client.delete_block_document(block_id)\n                exit_with_success(f\"Deleted Block '{block_id}'.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {block_id!r} not found!\")\n        elif slug is not None:\n            block_type_slug, block_document_name = slug.split(\"/\")\n            try:\n                block_document = await client.read_block_document_by_name(\n                    block_document_name, block_type_slug, include_secrets=False\n                )\n                await client.delete_block_document(block_document.id)\n                exit_with_success(f\"Deleted Block '{slug}'.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Block {slug!r} not found!\")\n        else:\n            exit_with_error(\"Must provide a block slug or id\")\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.block_inspect","title":"block_inspect async","text":"

    Displays details about a configured block.

    Source code in prefect/cli/block.py
    @blocks_app.command(\"inspect\")\nasync def block_inspect(\n    slug: Optional[str] = typer.Argument(\n        None, help=\"A Block slug: <BLOCK_TYPE_SLUG>/<BLOCK_NAME>\"\n    ),\n    block_id: Optional[str] = typer.Option(\n        None, \"--id\", help=\"A Block id to search for if no slug is given\"\n    ),\n):\n    \"\"\"\n    Displays details about a configured block.\n    \"\"\"\n    async with get_client() as client:\n        if slug is None and block_id is not None:\n            try:\n                block_document = await client.read_block_document(\n                    block_id, include_secrets=False\n                )\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {block_id!r} not found!\")\n        elif slug is not None:\n            block_type_slug, block_document_name = slug.split(\"/\")\n            try:\n                block_document = await client.read_block_document_by_name(\n                    block_document_name, block_type_slug, include_secrets=False\n                )\n            except ObjectNotFound:\n                exit_with_error(f\"Block {slug!r} not found!\")\n        else:\n            exit_with_error(\"Must provide a block slug or id\")\n        app.console.print(display_block(block_document))\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.block_ls","title":"block_ls async","text":"

    View all configured blocks.

    Source code in prefect/cli/block.py
    @blocks_app.command(\"ls\")\nasync def block_ls():\n    \"\"\"\n    View all configured blocks.\n    \"\"\"\n    async with get_client() as client:\n        blocks = await client.read_block_documents()\n\n    table = Table(\n        title=\"Blocks\", caption=\"List Block Types using `prefect block type ls`\"\n    )\n    table.add_column(\"ID\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Type\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Name\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Slug\", style=\"blue\", no_wrap=True)\n\n    for block in sorted(blocks, key=lambda x: f\"{x.block_type.slug}/{x.name}\"):\n        table.add_row(\n            str(block.id),\n            block.block_type.name,\n            str(block.name),\n            f\"{block.block_type.slug}/{block.name}\",\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.blocktype_delete","title":"blocktype_delete async","text":"

    Delete an unprotected Block Type.

    Source code in prefect/cli/block.py
    @blocktypes_app.command(\"delete\")\nasync def blocktype_delete(\n    slug: str = typer.Argument(..., help=\"A Block type slug\"),\n):\n    \"\"\"\n    Delete an unprotected Block Type.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            block_type = await client.read_block_type_by_slug(slug)\n            await client.delete_block_type(block_type.id)\n            exit_with_success(f\"Deleted Block Type '{slug}'.\")\n        except ObjectNotFound:\n            exit_with_error(f\"Block Type {slug!r} not found!\")\n        except ProtectedBlockError:\n            exit_with_error(f\"Block Type {slug!r} is a protected block!\")\n        except PrefectHTTPStatusError:\n            exit_with_error(f\"Cannot delete Block Type {slug!r}!\")\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.blocktype_inspect","title":"blocktype_inspect async","text":"

    Display details about a block type.

    Source code in prefect/cli/block.py
    @blocktypes_app.command(\"inspect\")\nasync def blocktype_inspect(\n    slug: str = typer.Argument(..., help=\"A block type slug\"),\n):\n    \"\"\"\n    Display details about a block type.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            block_type = await client.read_block_type_by_slug(slug)\n        except ObjectNotFound:\n            exit_with_error(f\"Block type {slug!r} not found!\")\n\n        app.console.print(display_block_type(block_type))\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.list_types","title":"list_types async","text":"

    List all block types.

    Source code in prefect/cli/block.py
    @blocktypes_app.command(\"ls\")\nasync def list_types():\n    \"\"\"\n    List all block types.\n    \"\"\"\n    async with get_client() as client:\n        block_types = await client.read_block_types()\n\n    table = Table(\n        title=\"Block Types\",\n        show_lines=True,\n    )\n\n    table.add_column(\"Block Type Slug\", style=\"italic cyan\", no_wrap=True)\n    table.add_column(\"Description\", style=\"blue\", no_wrap=False, justify=\"left\")\n    table.add_column(\n        \"Generate creation link\", style=\"italic cyan\", no_wrap=False, justify=\"left\"\n    )\n\n    for blocktype in sorted(block_types, key=lambda x: x.name):\n        table.add_row(\n            str(blocktype.slug),\n            (\n                str(blocktype.description.splitlines()[0].partition(\".\")[0])\n                if blocktype.description is not None\n                else \"\"\n            ),\n            f\"prefect block create {blocktype.slug}\",\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.register","title":"register async","text":"

    Register blocks types within a module or file.

    This makes the blocks available for configuration via the UI. If a block type has already been registered, its registration will be updated to match the block's current definition.

    \b Examples: \b Register block types in a Python module: $ prefect block register -m prefect_aws.credentials \b Register block types in a .py file: $ prefect block register -f my_blocks.py

    Source code in prefect/cli/block.py
    @blocks_app.command()\nasync def register(\n    module_name: Optional[str] = typer.Option(\n        None,\n        \"--module\",\n        \"-m\",\n        help=\"Python module containing block types to be registered\",\n    ),\n    file_path: Optional[Path] = typer.Option(\n        None,\n        \"--file\",\n        \"-f\",\n        help=\"Path to .py file containing block types to be registered\",\n    ),\n):\n    \"\"\"\n    Register blocks types within a module or file.\n\n    This makes the blocks available for configuration via the UI.\n    If a block type has already been registered, its registration will be updated to\n    match the block's current definition.\n\n    \\b\n    Examples:\n        \\b\n        Register block types in a Python module:\n        $ prefect block register -m prefect_aws.credentials\n        \\b\n        Register block types in a .py file:\n        $ prefect block register -f my_blocks.py\n    \"\"\"\n    # Handles if both options are specified or if neither are specified\n    if not (bool(file_path) ^ bool(module_name)):\n        exit_with_error(\n            \"Please specify either a module or a file containing blocks to be\"\n            \" registered, but not both.\"\n        )\n\n    if module_name:\n        try:\n            imported_module = import_module(name=module_name)\n        except ModuleNotFoundError:\n            exit_with_error(\n                f\"Unable to load {module_name}. Please make sure the module is \"\n                \"installed in your current environment.\"\n            )\n\n    if file_path:\n        if file_path.suffix != \".py\":\n            exit_with_error(\n                f\"{file_path} is not a .py file. Please specify a \"\n                \".py that contains blocks to be registered.\"\n            )\n        try:\n            imported_module = await run_sync_in_worker_thread(\n                load_script_as_module, str(file_path)\n            )\n        except ScriptError as exc:\n            app.console.print(exc)\n            app.console.print(exception_traceback(exc.user_exc))\n            exit_with_error(\n                f\"Unable to load file at {file_path}. Please make sure the file path \"\n                \"is correct and the file contains valid Python.\"\n            )\n\n    registered_blocks = await _register_blocks_in_module(imported_module)\n    number_of_registered_blocks = len(registered_blocks)\n    block_text = \"block\" if 0 < number_of_registered_blocks < 2 else \"blocks\"\n    app.console.print(\n        f\"[green]Successfully registered {number_of_registered_blocks} {block_text}\\n\"\n    )\n    app.console.print(_build_registered_blocks_table(registered_blocks))\n    msg = (\n        \"\\n To configure the newly registered blocks, \"\n        \"go to the Blocks page in the Prefect UI.\\n\"\n    )\n\n    ui_url = PREFECT_UI_URL.value()\n    if ui_url is not None:\n        block_catalog_url = f\"{ui_url}/blocks/catalog\"\n        msg = f\"{msg.rstrip().rstrip('.')}: {block_catalog_url}\\n\"\n\n    app.console.print(msg)\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/cloud-webhook/","title":"Cloud webhook","text":"","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook","title":"prefect.cli.cloud.webhook","text":"

    Command line interface for working with webhooks

    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.create","title":"create async","text":"

    Create a new Cloud webhook

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def create(\n    webhook_name: str,\n    description: str = typer.Option(\n        \"\", \"--description\", \"-d\", help=\"Description of the webhook\"\n    ),\n    template: str = typer.Option(\n        None, \"--template\", \"-t\", help=\"Jinja2 template expression\"\n    ),\n):\n    \"\"\"\n    Create a new Cloud webhook\n    \"\"\"\n    if not template:\n        exit_with_error(\n            \"Please provide a Jinja2 template expression in the --template flag \\nwhich\"\n            ' should define (at minimum) the following attributes: \\n{ \"event\":'\n            ' \"your.event.name\", \"resource\": { \"prefect.resource.id\":'\n            ' \"your.resource.id\" } }'\n            \" \\nhttps://docs.prefect.io/latest/cloud/webhooks/#webhook-templates\"\n        )\n\n    confirm_logged_in()\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        response = await client.request(\n            \"POST\",\n            \"/webhooks/\",\n            json={\n                \"name\": webhook_name,\n                \"description\": description,\n                \"template\": template,\n            },\n        )\n        app.console.print(f'Successfully created webhook {response[\"name\"]}')\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.delete","title":"delete async","text":"

    Delete an existing Cloud webhook

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def delete(webhook_id: UUID):\n    \"\"\"\n    Delete an existing Cloud webhook\n    \"\"\"\n    confirm_logged_in()\n\n    confirm_delete = typer.confirm(\n        \"Are you sure you want to delete it? This cannot be undone.\"\n    )\n\n    if not confirm_delete:\n        return\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        await client.request(\"DELETE\", f\"/webhooks/{webhook_id}\")\n        app.console.print(f\"Successfully deleted webhook {webhook_id}\")\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.get","title":"get async","text":"

    Retrieve a webhook by ID.

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def get(webhook_id: UUID):\n    \"\"\"\n    Retrieve a webhook by ID.\n    \"\"\"\n    confirm_logged_in()\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        webhook = await client.request(\"GET\", f\"/webhooks/{webhook_id}\")\n        display_table = _render_webhooks_into_table([webhook])\n        app.console.print(display_table)\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.ls","title":"ls async","text":"

    Fetch and list all webhooks in your workspace

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def ls():\n    \"\"\"\n    Fetch and list all webhooks in your workspace\n    \"\"\"\n    confirm_logged_in()\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        retrieved_webhooks = await client.request(\"POST\", \"/webhooks/filter\")\n        display_table = _render_webhooks_into_table(retrieved_webhooks)\n        app.console.print(display_table)\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.rotate","title":"rotate async","text":"

    Rotate url for an existing Cloud webhook, in case it has been compromised

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def rotate(webhook_id: UUID):\n    \"\"\"\n    Rotate url for an existing Cloud webhook, in case it has been compromised\n    \"\"\"\n    confirm_logged_in()\n\n    confirm_rotate = typer.confirm(\n        \"Are you sure you want to rotate? This will invalidate the old URL.\"\n    )\n\n    if not confirm_rotate:\n        return\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        response = await client.request(\"POST\", f\"/webhooks/{webhook_id}/rotate\")\n        app.console.print(f'Successfully rotated webhook URL to {response[\"slug\"]}')\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.toggle","title":"toggle async","text":"

    Toggle the enabled status of an existing Cloud webhook

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def toggle(\n    webhook_id: UUID,\n):\n    \"\"\"\n    Toggle the enabled status of an existing Cloud webhook\n    \"\"\"\n    confirm_logged_in()\n\n    status_lookup = {True: \"enabled\", False: \"disabled\"}\n\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        response = await client.request(\"GET\", f\"/webhooks/{webhook_id}\")\n        current_status = response[\"enabled\"]\n        new_status = not current_status\n\n        await client.request(\n            \"PATCH\", f\"/webhooks/{webhook_id}\", json={\"enabled\": new_status}\n        )\n        app.console.print(f\"Webhook is now {status_lookup[new_status]}\")\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.update","title":"update async","text":"

    Partially update an existing Cloud webhook

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def update(\n    webhook_id: UUID,\n    webhook_name: str = typer.Option(None, \"--name\", \"-n\", help=\"Webhook name\"),\n    description: str = typer.Option(\n        None, \"--description\", \"-d\", help=\"Description of the webhook\"\n    ),\n    template: str = typer.Option(\n        None, \"--template\", \"-t\", help=\"Jinja2 template expression\"\n    ),\n):\n    \"\"\"\n    Partially update an existing Cloud webhook\n    \"\"\"\n    confirm_logged_in()\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        response = await client.request(\"GET\", f\"/webhooks/{webhook_id}\")\n        update_payload = {\n            \"name\": webhook_name or response[\"name\"],\n            \"description\": description or response[\"description\"],\n            \"template\": template or response[\"template\"],\n        }\n\n        await client.request(\"PUT\", f\"/webhooks/{webhook_id}\", json=update_payload)\n        app.console.print(f\"Successfully updated webhook {webhook_id}\")\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud/","title":"cloud","text":"","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud","title":"prefect.cli.cloud","text":"

    Command line interface for interacting with Prefect Cloud

    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.login_api","title":"login_api = FastAPI(lifespan=lifespan) module-attribute","text":"

    This small API server is used for data transmission for browser-based log in.

    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.check_key_is_valid_for_login","title":"check_key_is_valid_for_login async","text":"

    Attempt to use a key to see if it is valid

    Source code in prefect/cli/cloud/__init__.py
    async def check_key_is_valid_for_login(key: str):\n    \"\"\"\n    Attempt to use a key to see if it is valid\n    \"\"\"\n    async with get_cloud_client(api_key=key) as client:\n        try:\n            await client.read_workspaces()\n            return True\n        except CloudUnauthorizedError:\n            return False\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.login","title":"login async","text":"

    Log in to Prefect Cloud. Creates a new profile configured to use the specified PREFECT_API_KEY. Uses a previously configured profile if it exists.

    Source code in prefect/cli/cloud/__init__.py
    @cloud_app.command()\nasync def login(\n    key: Optional[str] = typer.Option(\n        None, \"--key\", \"-k\", help=\"API Key to authenticate with Prefect\"\n    ),\n    workspace_handle: Optional[str] = typer.Option(\n        None,\n        \"--workspace\",\n        \"-w\",\n        help=(\n            \"Full handle of workspace, in format '<account_handle>/<workspace_handle>'\"\n        ),\n    ),\n):\n    \"\"\"\n    Log in to Prefect Cloud.\n    Creates a new profile configured to use the specified PREFECT_API_KEY.\n    Uses a previously configured profile if it exists.\n    \"\"\"\n    if not is_interactive() and (not key or not workspace_handle):\n        exit_with_error(\n            \"When not using an interactive terminal, you must supply a `--key` and\"\n            \" `--workspace`.\"\n        )\n\n    profiles = load_profiles()\n    current_profile = get_settings_context().profile\n    env_var_api_key = PREFECT_API_KEY.value()\n\n    if env_var_api_key and key and env_var_api_key != key:\n        exit_with_error(\n            \"Cannot log in with a key when a different PREFECT_API_KEY is present as an\"\n            \" environment variable that will override it.\"\n        )\n\n    if env_var_api_key and env_var_api_key == key:\n        is_valid_key = await check_key_is_valid_for_login(key)\n        is_correct_key_format = key.startswith(\"pnu_\") or key.startswith(\"pnb_\")\n        if not is_valid_key:\n            help_message = \"Please ensure your credentials are correct and unexpired.\"\n            if not is_correct_key_format:\n                help_message = \"Your key is not in our expected format.\"\n            exit_with_error(\n                f\"Unable to authenticate with Prefect Cloud. {help_message}\"\n            )\n\n    already_logged_in_profiles = []\n    for name, profile in profiles.items():\n        profile_key = profile.settings.get(PREFECT_API_KEY)\n        if (\n            # If a key is provided, only show profiles with the same key\n            (key and profile_key == key)\n            # Otherwise, show all profiles with a key set\n            or (not key and profile_key is not None)\n            # Check that the key is usable to avoid suggesting unauthenticated profiles\n            and await check_key_is_valid_for_login(profile_key)\n        ):\n            already_logged_in_profiles.append(name)\n\n    current_profile_is_logged_in = current_profile.name in already_logged_in_profiles\n\n    if current_profile_is_logged_in:\n        app.console.print(\"It looks like you're already authenticated on this profile.\")\n        should_reauth = typer.confirm(\n            \"? Would you like to reauthenticate?\", default=False\n        )\n        if not should_reauth:\n            app.console.print(\"Using the existing authentication on this profile.\")\n            key = PREFECT_API_KEY.value()\n\n    elif already_logged_in_profiles:\n        app.console.print(\n            \"It looks like you're already authenticated with another profile.\"\n        )\n        if not typer.confirm(\n            \"? Would you like to reauthenticate with this profile?\", default=False\n        ):\n            if typer.confirm(\n                \"? Would you like to switch to an authenticated profile?\", default=True\n            ):\n                profile_name = prompt_select_from_list(\n                    app.console,\n                    \"Which authenticated profile would you like to switch to?\",\n                    already_logged_in_profiles,\n                )\n\n                profiles.set_active(profile_name)\n                save_profiles(profiles)\n                exit_with_success(\n                    f\"Switched to authenticated profile {profile_name!r}.\"\n                )\n            else:\n                return\n\n    if not key:\n        choice = prompt_select_from_list(\n            app.console,\n            \"How would you like to authenticate?\",\n            [\n                (\"browser\", \"Log in with a web browser\"),\n                (\"key\", \"Paste an API key\"),\n            ],\n        )\n\n        if choice == \"key\":\n            key = typer.prompt(\"Paste your API key\", hide_input=True)\n        elif choice == \"browser\":\n            key = await login_with_browser()\n\n    async with get_cloud_client(api_key=key) as client:\n        try:\n            workspaces = await client.read_workspaces()\n        except CloudUnauthorizedError:\n            if key.startswith(\"pcu\"):\n                help_message = (\n                    \"It looks like you're using API key from Cloud 1\"\n                    \" (https://cloud.prefect.io). Make sure that you generate API key\"\n                    \" using Cloud 2 (https://app.prefect.cloud)\"\n                )\n            elif not key.startswith(\"pnu_\") and not key.startswith(\"pnb_\"):\n                help_message = (\n                    \"Your key is not in our expected format: 'pnu_' or 'pnb_'.\"\n                )\n            else:\n                help_message = (\n                    \"Please ensure your credentials are correct and unexpired.\"\n                )\n            exit_with_error(\n                f\"Unable to authenticate with Prefect Cloud. {help_message}\"\n            )\n        except httpx.HTTPStatusError as exc:\n            exit_with_error(f\"Error connecting to Prefect Cloud: {exc!r}\")\n\n    if workspace_handle:\n        # Search for the given workspace\n        for workspace in workspaces:\n            if workspace.handle == workspace_handle:\n                break\n        else:\n            if workspaces:\n                hint = (\n                    \" Available workspaces:\"\n                    f\" {listrepr((w.handle for w in workspaces), ', ')}\"\n                )\n            else:\n                hint = \"\"\n\n            exit_with_error(f\"Workspace {workspace_handle!r} not found.\" + hint)\n    else:\n        # Prompt a switch if the number of workspaces is greater than one\n        prompt_switch_workspace = len(workspaces) > 1\n\n        current_workspace = get_current_workspace(workspaces)\n\n        # Confirm that we want to switch if the current profile is already logged in\n        if (\n            current_profile_is_logged_in and current_workspace is not None\n        ) and prompt_switch_workspace:\n            app.console.print(\n                f\"You are currently using workspace {current_workspace.handle!r}.\"\n            )\n            prompt_switch_workspace = typer.confirm(\n                \"? Would you like to switch workspaces?\", default=False\n            )\n\n        if prompt_switch_workspace:\n            workspace = prompt_select_from_list(\n                app.console,\n                \"Which workspace would you like to use?\",\n                [(workspace, workspace.handle) for workspace in workspaces],\n            )\n        else:\n            if current_workspace:\n                workspace = current_workspace\n            elif len(workspaces) > 0:\n                workspace = workspaces[0]\n            else:\n                exit_with_error(\n                    \"No workspaces found! Create a workspace at\"\n                    f\" {PREFECT_CLOUD_UI_URL.value()} and try again.\"\n                )\n\n    update_current_profile(\n        {\n            PREFECT_API_KEY: key,\n            PREFECT_API_URL: workspace.api_url(),\n        }\n    )\n\n    exit_with_success(\n        f\"Authenticated with Prefect Cloud! Using workspace {workspace.handle!r}.\"\n    )\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.login_with_browser","title":"login_with_browser async","text":"

    Perform login using the browser.

    On failure, this function will exit the process. On success, it will return an API key.

    Source code in prefect/cli/cloud/__init__.py
    async def login_with_browser() -> str:\n    \"\"\"\n    Perform login using the browser.\n\n    On failure, this function will exit the process.\n    On success, it will return an API key.\n    \"\"\"\n\n    # Set up an event that the login API will toggle on startup\n    ready_event = login_api.extra[\"ready-event\"] = anyio.Event()\n\n    # Set up an event that the login API will set when a response comes from the UI\n    result_event = login_api.extra[\"result-event\"] = anyio.Event()\n\n    timeout_scope = None\n    async with anyio.create_task_group() as tg:\n        # Run a server in the background to get payload from the browser\n        server = await tg.start(serve_login_api, tg.cancel_scope)\n\n        # Wait for the login server to be ready\n        with anyio.fail_after(10):\n            await ready_event.wait()\n\n            # The server may not actually be serving as the lifespan is started first\n            while not server.started:\n                await anyio.sleep(0)\n\n        # Get the port the server is using\n        server_port = server.servers[0].sockets[0].getsockname()[1]\n        callback = urllib.parse.quote(f\"http://localhost:{server_port}\")\n        ui_login_url = (\n            PREFECT_CLOUD_UI_URL.value() + f\"/auth/client?callback={callback}\"\n        )\n\n        # Then open the authorization page in a new browser tab\n        app.console.print(\"Opening browser...\")\n        await run_sync_in_worker_thread(webbrowser.open_new_tab, ui_login_url)\n\n        # Wait for the response from the browser,\n        with anyio.move_on_after(120) as timeout_scope:\n            app.console.print(\"Waiting for response...\")\n            await result_event.wait()\n\n        # Uvicorn installs signal handlers, this is the cleanest way to shutdown the\n        # login API\n        raise_signal(signal.SIGINT)\n\n    result = login_api.extra.get(\"result\")\n    if not result:\n        if timeout_scope and timeout_scope.cancel_called:\n            exit_with_error(\"Timed out while waiting for authorization.\")\n        else:\n            exit_with_error(\"Aborted.\")\n\n    if result.type == \"success\":\n        return result.content.api_key\n    elif result.type == \"failure\":\n        exit_with_error(f\"Failed to log in. {result.content.reason}\")\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.logout","title":"logout async","text":"

    Logout the current workspace. Reset PREFECT_API_KEY and PREFECT_API_URL to default.

    Source code in prefect/cli/cloud/__init__.py
    @cloud_app.command()\nasync def logout():\n    \"\"\"\n    Logout the current workspace.\n    Reset PREFECT_API_KEY and PREFECT_API_URL to default.\n    \"\"\"\n    current_profile = prefect.context.get_settings_context().profile\n    if current_profile is None:\n        exit_with_error(\"There is no current profile set.\")\n\n    if current_profile.settings.get(PREFECT_API_KEY) is None:\n        exit_with_error(\"Current profile is not logged into Prefect Cloud.\")\n\n    update_current_profile(\n        {\n            PREFECT_API_URL: None,\n            PREFECT_API_KEY: None,\n        },\n    )\n\n    exit_with_success(\"Logged out from Prefect Cloud.\")\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.ls","title":"ls async","text":"

    List available workspaces.

    Source code in prefect/cli/cloud/__init__.py
    @workspace_app.command()\nasync def ls():\n    \"\"\"List available workspaces.\"\"\"\n\n    confirm_logged_in()\n\n    async with get_cloud_client() as client:\n        try:\n            workspaces = await client.read_workspaces()\n        except CloudUnauthorizedError:\n            exit_with_error(\n                \"Unable to authenticate. Please ensure your credentials are correct.\"\n            )\n\n    current_workspace = get_current_workspace(workspaces)\n\n    table = Table(caption=\"* active workspace\")\n    table.add_column(\n        \"[#024dfd]Workspaces:\", justify=\"left\", style=\"#8ea0ae\", no_wrap=True\n    )\n\n    for workspace_handle in sorted(workspace.handle for workspace in workspaces):\n        if workspace_handle == current_workspace.handle:\n            table.add_row(f\"[green]* {workspace_handle}[/green]\")\n        else:\n            table.add_row(f\"  {workspace_handle}\")\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.open","title":"open async","text":"

    Open the Prefect Cloud UI in the browser.

    Source code in prefect/cli/cloud/__init__.py
    @cloud_app.command()\nasync def open():\n    \"\"\"\n    Open the Prefect Cloud UI in the browser.\n    \"\"\"\n    confirm_logged_in()\n\n    current_profile = prefect.context.get_settings_context().profile\n    if current_profile is None:\n        exit_with_error(\n            \"There is no current profile set - set one with `prefect profile create\"\n            \" <name>` and `prefect profile use <name>`.\"\n        )\n\n    current_workspace = get_current_workspace(\n        await prefect.get_cloud_client().read_workspaces()\n    )\n    if current_workspace is None:\n        exit_with_error(\n            \"There is no current workspace set - set one with `prefect cloud workspace\"\n            \" set --workspace <workspace>`.\"\n        )\n\n    ui_url = current_workspace.ui_url()\n\n    await run_sync_in_worker_thread(webbrowser.open_new_tab, ui_url)\n\n    exit_with_success(f\"Opened {current_workspace.handle!r} in browser.\")\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.prompt_select_from_list","title":"prompt_select_from_list","text":"

    Given a list of options, display the values to user in a table and prompt them to select one.

    Parameters:

    Name Type Description Default options Union[List[str], List[Tuple[Hashable, str]]]

    A list of options to present to the user. A list of tuples can be passed as key value pairs. If a value is chosen, the key will be returned.

    required

    Returns:

    Name Type Description str str

    the selected option

    Source code in prefect/cli/cloud/__init__.py
    def prompt_select_from_list(\n    console, prompt: str, options: Union[List[str], List[Tuple[Hashable, str]]]\n) -> str:\n    \"\"\"\n    Given a list of options, display the values to user in a table and prompt them\n    to select one.\n\n    Args:\n        options: A list of options to present to the user.\n            A list of tuples can be passed as key value pairs. If a value is chosen, the\n            key will be returned.\n\n    Returns:\n        str: the selected option\n    \"\"\"\n\n    current_idx = 0\n    selected_option = None\n\n    def build_table() -> Table:\n        \"\"\"\n        Generate a table of options. The `current_idx` will be highlighted.\n        \"\"\"\n\n        table = Table(box=False, header_style=None, padding=(0, 0))\n        table.add_column(\n            f\"? [bold]{prompt}[/] [bright_blue][Use arrows to move; enter to select]\",\n            justify=\"left\",\n            no_wrap=True,\n        )\n\n        for i, option in enumerate(options):\n            if isinstance(option, tuple):\n                option = option[1]\n\n            if i == current_idx:\n                # Use blue for selected options\n                table.add_row(\"[bold][blue]> \" + option)\n            else:\n                table.add_row(\"  \" + option)\n        return table\n\n    with Live(build_table(), auto_refresh=False, console=console) as live:\n        while selected_option is None:\n            key = readchar.readkey()\n\n            if key == readchar.key.UP:\n                current_idx = current_idx - 1\n                # wrap to bottom if at the top\n                if current_idx < 0:\n                    current_idx = len(options) - 1\n            elif key == readchar.key.DOWN:\n                current_idx = current_idx + 1\n                # wrap to top if at the bottom\n                if current_idx >= len(options):\n                    current_idx = 0\n            elif key == readchar.key.CTRL_C:\n                # gracefully exit with no message\n                exit_with_error(\"\")\n            elif key == readchar.key.ENTER or key == readchar.key.CR:\n                selected_option = options[current_idx]\n                if isinstance(selected_option, tuple):\n                    selected_option = selected_option[0]\n\n            live.update(build_table(), refresh=True)\n\n        return selected_option\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.set","title":"set async","text":"

    Set current workspace. Shows a workspace picker if no workspace is specified.

    Source code in prefect/cli/cloud/__init__.py
    @workspace_app.command()\nasync def set(\n    workspace_handle: str = typer.Option(\n        None,\n        \"--workspace\",\n        \"-w\",\n        help=(\n            \"Full handle of workspace, in format '<account_handle>/<workspace_handle>'\"\n        ),\n    ),\n):\n    \"\"\"Set current workspace. Shows a workspace picker if no workspace is specified.\"\"\"\n    confirm_logged_in()\n\n    async with get_cloud_client() as client:\n        try:\n            workspaces = await client.read_workspaces()\n        except CloudUnauthorizedError:\n            exit_with_error(\n                \"Unable to authenticate. Please ensure your credentials are correct.\"\n            )\n\n    if workspace_handle:\n        # Search for the given workspace\n        for workspace in workspaces:\n            if workspace.handle == workspace_handle:\n                break\n        else:\n            exit_with_error(f\"Workspace {workspace_handle!r} not found.\")\n    else:\n        workspace = prompt_select_from_list(\n            app.console,\n            \"Which workspace would you like to use?\",\n            [(workspace, workspace.handle) for workspace in workspaces],\n        )\n\n    profile = update_current_profile({PREFECT_API_URL: workspace.api_url()})\n\n    exit_with_success(\n        f\"Successfully set workspace to {workspace.handle!r} in profile\"\n        f\" {profile.name!r}.\"\n    )\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/concurrency_limit/","title":"concurrency_limit","text":"","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit","title":"prefect.cli.concurrency_limit","text":"

    Command line interface for working with concurrency limits.

    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.create","title":"create async","text":"

    Create a concurrency limit against a tag.

    This limit controls how many task runs with that tag may simultaneously be in a Running state.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def create(tag: str, concurrency_limit: int):\n    \"\"\"\n    Create a concurrency limit against a tag.\n\n    This limit controls how many task runs with that tag may simultaneously be in a\n    Running state.\n    \"\"\"\n\n    async with get_client() as client:\n        await client.create_concurrency_limit(\n            tag=tag, concurrency_limit=concurrency_limit\n        )\n        await client.read_concurrency_limit_by_tag(tag)\n\n    app.console.print(\n        textwrap.dedent(\n            f\"\"\"\n            Created concurrency limit with properties:\n                tag - {tag!r}\n                concurrency_limit - {concurrency_limit}\n\n            Delete the concurrency limit:\n                prefect concurrency-limit delete {tag!r}\n\n            Inspect the concurrency limit:\n                prefect concurrency-limit inspect {tag!r}\n        \"\"\"\n        )\n    )\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.delete","title":"delete async","text":"

    Delete the concurrency limit set on the specified tag.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def delete(tag: str):\n    \"\"\"\n    Delete the concurrency limit set on the specified tag.\n    \"\"\"\n\n    async with get_client() as client:\n        try:\n            await client.delete_concurrency_limit_by_tag(tag=tag)\n        except ObjectNotFound:\n            exit_with_error(f\"No concurrency limit found for the tag: {tag}\")\n\n    exit_with_success(f\"Deleted concurrency limit set on the tag: {tag}\")\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.inspect","title":"inspect async","text":"

    View details about a concurrency limit. active_slots shows a list of TaskRun IDs which are currently using a concurrency slot.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def inspect(tag: str):\n    \"\"\"\n    View details about a concurrency limit. `active_slots` shows a list of TaskRun IDs\n    which are currently using a concurrency slot.\n    \"\"\"\n\n    async with get_client() as client:\n        try:\n            result = await client.read_concurrency_limit_by_tag(tag=tag)\n        except ObjectNotFound:\n            exit_with_error(f\"No concurrency limit found for the tag: {tag}\")\n\n    trid_table = Table()\n    trid_table.add_column(\"Active Task Run IDs\", style=\"cyan\", no_wrap=True)\n\n    cl_table = Table(title=f\"Concurrency Limit ID: [red]{str(result.id)}\")\n    cl_table.add_column(\"Tag\", style=\"green\", no_wrap=True)\n    cl_table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n    cl_table.add_column(\"Created\", style=\"magenta\", no_wrap=True)\n    cl_table.add_column(\"Updated\", style=\"magenta\", no_wrap=True)\n\n    for trid in sorted(result.active_slots):\n        trid_table.add_row(str(trid))\n\n    cl_table.add_row(\n        str(result.tag),\n        str(result.concurrency_limit),\n        Pretty(pendulum.instance(result.created).diff_for_humans()),\n        Pretty(pendulum.instance(result.updated).diff_for_humans()),\n    )\n\n    group = Group(\n        cl_table,\n        trid_table,\n    )\n    app.console.print(Panel(group, expand=False))\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.ls","title":"ls async","text":"

    View all concurrency limits.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def ls(limit: int = 15, offset: int = 0):\n    \"\"\"\n    View all concurrency limits.\n    \"\"\"\n    table = Table(\n        title=\"Concurrency Limits\",\n        caption=\"inspect a concurrency limit to show active task run IDs\",\n    )\n    table.add_column(\"Tag\", style=\"green\", no_wrap=True)\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Active Task Runs\", style=\"magenta\", no_wrap=True)\n\n    async with get_client() as client:\n        concurrency_limits = await client.read_concurrency_limits(\n            limit=limit, offset=offset\n        )\n\n    for cl in sorted(concurrency_limits, key=lambda c: c.updated, reverse=True):\n        table.add_row(\n            str(cl.tag),\n            str(cl.id),\n            str(cl.concurrency_limit),\n            str(len(cl.active_slots)),\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.reset","title":"reset async","text":"

    Resets the concurrency limit slots set on the specified tag.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def reset(tag: str):\n    \"\"\"\n    Resets the concurrency limit slots set on the specified tag.\n    \"\"\"\n\n    async with get_client() as client:\n        try:\n            await client.reset_concurrency_limit_by_tag(tag=tag)\n        except ObjectNotFound:\n            exit_with_error(f\"No concurrency limit found for the tag: {tag}\")\n\n    exit_with_success(f\"Reset concurrency limit set on the tag: {tag}\")\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/config/","title":"config","text":"","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config","title":"prefect.cli.config","text":"

    Command line interface for working with profiles

    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config.set_","title":"set_","text":"

    Change the value for a setting by setting the value in the current profile.

    Source code in prefect/cli/config.py
    @config_app.command(\"set\")\ndef set_(settings: List[str]):\n    \"\"\"\n    Change the value for a setting by setting the value in the current profile.\n    \"\"\"\n    parsed_settings = {}\n    for item in settings:\n        try:\n            setting, value = item.split(\"=\", maxsplit=1)\n        except ValueError:\n            exit_with_error(\n                f\"Failed to parse argument {item!r}. Use the format 'VAR=VAL'.\"\n            )\n\n        if setting not in prefect.settings.SETTING_VARIABLES:\n            exit_with_error(f\"Unknown setting name {setting!r}.\")\n\n        # Guard against changing settings that tweak config locations\n        if setting in {\"PREFECT_HOME\", \"PREFECT_PROFILES_PATH\"}:\n            exit_with_error(\n                f\"Setting {setting!r} cannot be changed with this command. \"\n                \"Use an environment variable instead.\"\n            )\n\n        parsed_settings[setting] = value\n\n    try:\n        new_profile = prefect.settings.update_current_profile(parsed_settings)\n    except pydantic.ValidationError as exc:\n        for error in exc.errors():\n            setting = error[\"loc\"][0]\n            message = error[\"msg\"]\n            app.console.print(f\"Validation error for setting {setting!r}: {message}\")\n        exit_with_error(\"Invalid setting value.\")\n\n    for setting, value in parsed_settings.items():\n        app.console.print(f\"Set {setting!r} to {value!r}.\")\n        if setting in os.environ:\n            app.console.print(\n                f\"[yellow]{setting} is also set by an environment variable which will \"\n                f\"override your config value. Run `unset {setting}` to clear it.\"\n            )\n\n        if prefect.settings.SETTING_VARIABLES[setting].deprecated:\n            app.console.print(\n                f\"[yellow]{prefect.settings.SETTING_VARIABLES[setting].deprecated_message}.\"\n            )\n\n    exit_with_success(f\"Updated profile {new_profile.name!r}.\")\n
    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config.unset","title":"unset","text":"

    Restore the default value for a setting.

    Removes the setting from the current profile.

    Source code in prefect/cli/config.py
    @config_app.command()\ndef unset(settings: List[str]):\n    \"\"\"\n    Restore the default value for a setting.\n\n    Removes the setting from the current profile.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    profile = profiles[prefect.context.get_settings_context().profile.name]\n    parsed = set()\n\n    for setting in settings:\n        if setting not in prefect.settings.SETTING_VARIABLES:\n            exit_with_error(f\"Unknown setting name {setting!r}.\")\n        # Cast to settings objects\n        parsed.add(prefect.settings.SETTING_VARIABLES[setting])\n\n    for setting in parsed:\n        if setting not in profile.settings:\n            exit_with_error(f\"{setting.name!r} is not set in profile {profile.name!r}.\")\n\n    profiles.update_profile(\n        name=profile.name, settings={setting: None for setting in parsed}\n    )\n\n    for setting in settings:\n        app.console.print(f\"Unset {setting!r}.\")\n\n        if setting in os.environ:\n            app.console.print(\n                f\"[yellow]{setting!r} is also set by an environment variable. \"\n                f\"Use `unset {setting}` to clear it.\"\n            )\n\n    prefect.settings.save_profiles(profiles)\n    exit_with_success(f\"Updated profile {profile.name!r}.\")\n
    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config.validate","title":"validate","text":"

    Read and validate the current profile.

    Deprecated settings will be automatically converted to new names unless both are set.

    Source code in prefect/cli/config.py
    @config_app.command()\ndef validate():\n    \"\"\"\n    Read and validate the current profile.\n\n    Deprecated settings will be automatically converted to new names unless both are\n    set.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    profile = profiles[prefect.context.get_settings_context().profile.name]\n    changed = profile.convert_deprecated_renamed_settings()\n    for old, new in changed:\n        app.console.print(f\"Updated {old.name!r} to {new.name!r}.\")\n\n    for setting in profile.settings.keys():\n        if setting.deprecated:\n            app.console.print(f\"Found deprecated setting {setting.name!r}.\")\n\n    profile.validate_settings()\n\n    prefect.settings.save_profiles(profiles)\n    exit_with_success(\"Configuration valid!\")\n
    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config.view","title":"view","text":"

    Display the current settings.

    Source code in prefect/cli/config.py
    @config_app.command()\ndef view(\n    show_defaults: Optional[bool] = typer.Option(\n        False, \"--show-defaults/--hide-defaults\", help=(show_defaults_help)\n    ),\n    show_sources: Optional[bool] = typer.Option(\n        True,\n        \"--show-sources/--hide-sources\",\n        help=(show_sources_help),\n    ),\n    show_secrets: Optional[bool] = typer.Option(\n        False,\n        \"--show-secrets/--hide-secrets\",\n        help=\"Toggle display of secrets setting values.\",\n    ),\n):\n    \"\"\"\n    Display the current settings.\n    \"\"\"\n    context = prefect.context.get_settings_context()\n\n    # Get settings at each level, converted to a flat dictionary for easy comparison\n    default_settings = prefect.settings.get_default_settings()\n    env_settings = prefect.settings.get_settings_from_env()\n    current_profile_settings = context.settings\n\n    # Obfuscate secrets\n    if not show_secrets:\n        default_settings = default_settings.with_obfuscated_secrets()\n        env_settings = env_settings.with_obfuscated_secrets()\n        current_profile_settings = current_profile_settings.with_obfuscated_secrets()\n\n    # Display the profile first\n    app.console.print(f\"PREFECT_PROFILE={context.profile.name!r}\")\n\n    settings_output = []\n\n    # The combination of environment variables and profile settings that are in use\n    profile_overrides = current_profile_settings.dict(exclude_unset=True)\n\n    # Used to see which settings in current_profile_settings came from env vars\n    env_overrides = env_settings.dict(exclude_unset=True)\n\n    for key, value in profile_overrides.items():\n        source = \"env\" if env_overrides.get(key) is not None else \"profile\"\n        source_blurb = f\" (from {source})\" if show_sources else \"\"\n        settings_output.append(f\"{key}='{value}'{source_blurb}\")\n\n    if show_defaults:\n        for key, value in default_settings.dict().items():\n            if key not in profile_overrides:\n                source_blurb = \" (from defaults)\" if show_sources else \"\"\n                settings_output.append(f\"{key}='{value}'{source_blurb}\")\n\n    app.console.print(\"\\n\".join(sorted(settings_output)))\n
    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/deploy/","title":"deploy","text":"","tags":["Python API","deploy","deployment","CLI"]},{"location":"api-ref/prefect/cli/deploy/#prefect.cli.deploy","title":"prefect.cli.deploy","text":"

    Module containing implementation for deploying projects.

    ","tags":["Python API","deploy","deployment","CLI"]},{"location":"api-ref/prefect/cli/deploy/#prefect.cli.deploy.deploy","title":"deploy async","text":"

    Deploy a flow from this project by creating a deployment.

    Should be run from a project root directory.

    Source code in prefect/cli/deploy.py
    @app.command()\nasync def deploy(\n    entrypoint: str = typer.Argument(\n        None,\n        help=(\n            \"The path to a flow entrypoint within a project, in the form of\"\n            \" `./path/to/file.py:flow_func_name`\"\n        ),\n    ),\n    flow_name: str = typer.Option(\n        None,\n        \"--flow\",\n        \"-f\",\n        help=\"DEPRECATED: The name of a registered flow to create a deployment for.\",\n    ),\n    names: List[str] = typer.Option(\n        None,\n        \"--name\",\n        \"-n\",\n        help=(\n            \"The name to give the deployment. Can be a pattern. Examples:\"\n            \" 'my-deployment', 'my-flow/my-deployment', 'my-deployment-*',\"\n            \" '*-flow-name/deployment*'\"\n        ),\n    ),\n    description: str = typer.Option(\n        None,\n        \"--description\",\n        \"-d\",\n        help=(\n            \"The description to give the deployment. If not provided, the description\"\n            \" will be populated from the flow's description.\"\n        ),\n    ),\n    version: str = typer.Option(\n        None, \"--version\", help=\"A version to give the deployment.\"\n    ),\n    tags: List[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=(\n            \"One or more optional tags to apply to the deployment. Note: tags are used\"\n            \" only for organizational purposes. For delegating work to agents, use the\"\n            \" --work-queue flag.\"\n        ),\n    ),\n    work_pool_name: str = SettingsOption(\n        PREFECT_DEFAULT_WORK_POOL_NAME,\n        \"-p\",\n        \"--pool\",\n        help=\"The work pool that will handle this deployment's runs.\",\n    ),\n    work_queue_name: str = typer.Option(\n        None,\n        \"-q\",\n        \"--work-queue\",\n        help=(\n            \"The work queue that will handle this deployment's runs. \"\n            \"It will be created if it doesn't already exist. Defaults to `None`.\"\n        ),\n    ),\n    variables: List[str] = typer.Option(\n        None,\n        \"-v\",\n        \"--variable\",\n        help=(\n            \"One or more job variable overrides for the work pool provided in the\"\n            \" format of key=value string or a JSON object\"\n        ),\n    ),\n    cron: str = typer.Option(\n        None,\n        \"--cron\",\n        help=\"A cron string that will be used to set a CronSchedule on the deployment.\",\n    ),\n    interval: int = typer.Option(\n        None,\n        \"--interval\",\n        help=(\n            \"An integer specifying an interval (in seconds) that will be used to set an\"\n            \" IntervalSchedule on the deployment.\"\n        ),\n    ),\n    interval_anchor: Optional[str] = typer.Option(\n        None, \"--anchor-date\", help=\"The anchor date for an interval schedule\"\n    ),\n    rrule: str = typer.Option(\n        None,\n        \"--rrule\",\n        help=\"An RRule that will be used to set an RRuleSchedule on the deployment.\",\n    ),\n    timezone: str = typer.Option(\n        None,\n        \"--timezone\",\n        help=\"Deployment schedule timezone string e.g. 'America/New_York'\",\n    ),\n    trigger: List[str] = typer.Option(\n        None,\n        \"--trigger\",\n        help=(\n            \"Specifies a trigger for the deployment. The value can be a\"\n            \" json string or path to `.yaml`/`.json` file. This flag can be used\"\n            \" multiple times.\"\n        ),\n    ),\n    param: List[str] = typer.Option(\n        None,\n        \"--param\",\n        help=(\n            \"An optional parameter override, values are parsed as JSON strings e.g.\"\n            \" --param question=ultimate --param answer=42\"\n        ),\n    ),\n    params: str = typer.Option(\n        None,\n        \"--params\",\n        help=(\n            \"An optional parameter override in a JSON string format e.g.\"\n            ' --params=\\'{\"question\": \"ultimate\", \"answer\": 42}\\''\n        ),\n    ),\n    enforce_parameter_schema: bool = typer.Option(\n        False,\n        \"--enforce-parameter-schema\",\n        help=(\n            \"Whether to enforce the parameter schema on this deployment. If set to\"\n            \" True, any parameters passed to this deployment must match the signature\"\n            \" of the flow.\"\n        ),\n    ),\n    deploy_all: bool = typer.Option(\n        False,\n        \"--all\",\n        help=(\n            \"Deploy all flows in the project. If a flow name or entrypoint is also\"\n            \" provided, this flag will be ignored.\"\n        ),\n    ),\n    ci: bool = typer.Option(\n        False,\n        \"--ci\",\n        help=(\n            \"DEPRECATED: Please use the global '--no-prompt' flag instead: 'prefect\"\n            \" --no-prompt deploy'.\\n\\nRun this command in CI mode. This will disable\"\n            \" interactive prompts and will error if any required arguments are not\"\n            \" provided.\"\n        ),\n    ),\n):\n    \"\"\"\n    Deploy a flow from this project by creating a deployment.\n\n    Should be run from a project root directory.\n    \"\"\"\n    if ci:\n        app.console.print(\n            generate_deprecation_message(\n                name=\"The `--ci` flag\",\n                start_date=\"Jun 2023\",\n                help=(\n                    \"Please use the global `--no-prompt` flag instead: `prefect\"\n                    \" --no-prompt deploy`.\"\n                ),\n            ),\n            style=\"yellow\",\n        )\n\n    options = {\n        \"entrypoint\": entrypoint,\n        \"flow_name\": flow_name,\n        \"description\": description,\n        \"version\": version,\n        \"tags\": tags,\n        \"work_pool_name\": work_pool_name,\n        \"work_queue_name\": work_queue_name,\n        \"variables\": variables,\n        \"cron\": cron,\n        \"interval\": interval,\n        \"anchor_date\": interval_anchor,\n        \"rrule\": rrule,\n        \"timezone\": timezone,\n        \"triggers\": trigger,\n        \"param\": param,\n        \"params\": params,\n        \"enforce_parameter_schema\": enforce_parameter_schema,\n    }\n    try:\n        deploy_configs, actions = _load_deploy_configs_and_actions(ci=ci)\n\n        parsed_names = []\n        for name in names:\n            if \"*\" in name:\n                parsed_names.extend(_parse_name_from_pattern(deploy_configs, name))\n            else:\n                parsed_names.append(name)\n        deploy_configs = _pick_deploy_configs(\n            deploy_configs, parsed_names, deploy_all, ci\n        )\n\n        if len(deploy_configs) > 1:\n            if any(options.values()):\n                app.console.print(\n                    (\n                        \"You have passed options to the deploy command, but you are\"\n                        \" creating or updating multiple deployments. These options\"\n                        \" will be ignored.\"\n                    ),\n                    style=\"yellow\",\n                )\n            await _run_multi_deploy(\n                deploy_configs=deploy_configs,\n                actions=actions,\n                deploy_all=deploy_all,\n                ci=ci,\n            )\n        else:\n            # Accommodate passing in -n flow-name/deployment-name as well as -n deployment-name\n            options[\"names\"] = [\n                name.split(\"/\", 1)[-1] if \"/\" in name else name for name in parsed_names\n            ]\n\n            await _run_single_deploy(\n                deploy_config=deploy_configs[0] if deploy_configs else {},\n                actions=actions,\n                options=options,\n                ci=ci,\n            )\n    except ValueError as exc:\n        exit_with_error(str(exc))\n
    ","tags":["Python API","deploy","deployment","CLI"]},{"location":"api-ref/prefect/cli/deployment/","title":"deployment","text":"","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment","title":"prefect.cli.deployment","text":"

    Command line interface for working with deployments.

    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.apply","title":"apply async","text":"

    Create or update a deployment from a YAML file.

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def apply(\n    paths: List[str] = typer.Argument(\n        ...,\n        help=\"One or more paths to deployment YAML files.\",\n    ),\n    upload: bool = typer.Option(\n        False,\n        \"--upload\",\n        help=(\n            \"A flag that, when provided, uploads this deployment's files to remote\"\n            \" storage.\"\n        ),\n    ),\n    work_queue_concurrency: int = typer.Option(\n        None,\n        \"--limit\",\n        \"-l\",\n        help=(\n            \"Sets the concurrency limit on the work queue that handles this\"\n            \" deployment's runs\"\n        ),\n    ),\n):\n    \"\"\"\n    Create or update a deployment from a YAML file.\n    \"\"\"\n    deployment = None\n    async with get_client() as client:\n        for path in paths:\n            try:\n                deployment = await Deployment.load_from_yaml(path)\n                app.console.print(\n                    f\"Successfully loaded {deployment.name!r}\", style=\"green\"\n                )\n            except Exception as exc:\n                exit_with_error(\n                    f\"'{path!s}' did not conform to deployment spec: {exc!r}\"\n                )\n\n            assert deployment\n\n            await create_work_queue_and_set_concurrency_limit(\n                deployment.work_queue_name,\n                deployment.work_pool_name,\n                work_queue_concurrency,\n            )\n\n            if upload:\n                if (\n                    deployment.storage\n                    and \"put-directory\" in deployment.storage.get_block_capabilities()\n                ):\n                    file_count = await deployment.upload_to_storage()\n                    if file_count:\n                        app.console.print(\n                            (\n                                f\"Successfully uploaded {file_count} files to\"\n                                f\" {deployment.location}\"\n                            ),\n                            style=\"green\",\n                        )\n                else:\n                    app.console.print(\n                        (\n                            f\"Deployment storage {deployment.storage} does not have\"\n                            \" upload capabilities; no files uploaded.\"\n                        ),\n                        style=\"red\",\n                    )\n            await check_work_pool_exists(\n                work_pool_name=deployment.work_pool_name, client=client\n            )\n\n            if client.server_type != ServerType.CLOUD and deployment.triggers:\n                app.console.print(\n                    (\n                        \"Deployment triggers are only supported on \"\n                        f\"Prefect Cloud. Triggers defined in {path!r} will be \"\n                        \"ignored.\"\n                    ),\n                    style=\"red\",\n                )\n\n            deployment_id = await deployment.apply()\n            app.console.print(\n                (\n                    f\"Deployment '{deployment.flow_name}/{deployment.name}'\"\n                    f\" successfully created with id '{deployment_id}'.\"\n                ),\n                style=\"green\",\n            )\n\n            if PREFECT_UI_URL:\n                app.console.print(\n                    \"View Deployment in UI:\"\n                    f\" {PREFECT_UI_URL.value()}/deployments/deployment/{deployment_id}\"\n                )\n\n            if deployment.work_pool_name is not None:\n                await _print_deployment_work_pool_instructions(\n                    work_pool_name=deployment.work_pool_name, client=client\n                )\n            elif deployment.work_queue_name is not None:\n                app.console.print(\n                    \"\\nTo execute flow runs from this deployment, start an agent that\"\n                    f\" pulls work from the {deployment.work_queue_name!r} work queue:\"\n                )\n                app.console.print(\n                    f\"$ prefect agent start -q {deployment.work_queue_name!r}\",\n                    style=\"blue\",\n                )\n            else:\n                app.console.print(\n                    (\n                        \"\\nThis deployment does not specify a work queue name, which\"\n                        \" means agents will not be able to pick up its runs. To add a\"\n                        \" work queue, edit the deployment spec and re-run this command,\"\n                        \" or visit the deployment in the UI.\"\n                    ),\n                    style=\"red\",\n                )\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.build","title":"build async","text":"

    Generate a deployment YAML from /path/to/file.py:flow_function

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def build(\n    entrypoint: str = typer.Argument(\n        ...,\n        help=(\n            \"The path to a flow entrypoint, in the form of\"\n            \" `./path/to/file.py:flow_func_name`\"\n        ),\n    ),\n    name: str = typer.Option(\n        None, \"--name\", \"-n\", help=\"The name to give the deployment.\"\n    ),\n    description: str = typer.Option(\n        None,\n        \"--description\",\n        \"-d\",\n        help=(\n            \"The description to give the deployment. If not provided, the description\"\n            \" will be populated from the flow's description.\"\n        ),\n    ),\n    version: str = typer.Option(\n        None, \"--version\", \"-v\", help=\"A version to give the deployment.\"\n    ),\n    tags: List[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=(\n            \"One or more optional tags to apply to the deployment. Note: tags are used\"\n            \" only for organizational purposes. For delegating work to agents, use the\"\n            \" --work-queue flag.\"\n        ),\n    ),\n    work_queue_name: str = typer.Option(\n        None,\n        \"-q\",\n        \"--work-queue\",\n        help=(\n            \"The work queue that will handle this deployment's runs. \"\n            \"It will be created if it doesn't already exist. Defaults to `None`. \"\n            \"Note that if a work queue is not set, work will not be scheduled.\"\n        ),\n    ),\n    work_pool_name: str = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The work pool that will handle this deployment's runs.\",\n    ),\n    work_queue_concurrency: int = typer.Option(\n        None,\n        \"--limit\",\n        \"-l\",\n        help=(\n            \"Sets the concurrency limit on the work queue that handles this\"\n            \" deployment's runs\"\n        ),\n    ),\n    infra_type: str = typer.Option(\n        None,\n        \"--infra\",\n        \"-i\",\n        help=\"The infrastructure type to use, prepopulated with defaults. For example: \"\n        + listrepr(builtin_infrastructure_types, sep=\", \"),\n    ),\n    infra_block: str = typer.Option(\n        None,\n        \"--infra-block\",\n        \"-ib\",\n        help=\"The slug of the infrastructure block to use as a template.\",\n    ),\n    overrides: List[str] = typer.Option(\n        None,\n        \"--override\",\n        help=(\n            \"One or more optional infrastructure overrides provided as a dot delimited\"\n            \" path, e.g., `env.env_key=env_value`\"\n        ),\n    ),\n    storage_block: str = typer.Option(\n        None,\n        \"--storage-block\",\n        \"-sb\",\n        help=(\n            \"The slug of a remote storage block. Use the syntax:\"\n            \" 'block_type/block_name', where block_type is one of 'github', 's3',\"\n            \" 'gcs', 'azure', 'smb', or a registered block from a library that\"\n            \" implements the WritableDeploymentStorage interface such as\"\n            \" 'gitlab-repository', 'bitbucket-repository', 's3-bucket',\"\n            \" 'gcs-bucket'\"\n        ),\n    ),\n    skip_upload: bool = typer.Option(\n        False,\n        \"--skip-upload\",\n        help=(\n            \"A flag that, when provided, skips uploading this deployment's files to\"\n            \" remote storage.\"\n        ),\n    ),\n    cron: str = typer.Option(\n        None,\n        \"--cron\",\n        help=\"A cron string that will be used to set a CronSchedule on the deployment.\",\n    ),\n    interval: int = typer.Option(\n        None,\n        \"--interval\",\n        help=(\n            \"An integer specifying an interval (in seconds) that will be used to set an\"\n            \" IntervalSchedule on the deployment.\"\n        ),\n    ),\n    interval_anchor: Optional[str] = typer.Option(\n        None, \"--anchor-date\", help=\"The anchor date for an interval schedule\"\n    ),\n    rrule: str = typer.Option(\n        None,\n        \"--rrule\",\n        help=\"An RRule that will be used to set an RRuleSchedule on the deployment.\",\n    ),\n    timezone: str = typer.Option(\n        None,\n        \"--timezone\",\n        help=\"Deployment schedule timezone string e.g. 'America/New_York'\",\n    ),\n    path: str = typer.Option(\n        None,\n        \"--path\",\n        help=(\n            \"An optional path to specify a subdirectory of remote storage to upload to,\"\n            \" or to point to a subdirectory of a locally stored flow.\"\n        ),\n    ),\n    output: str = typer.Option(\n        None,\n        \"--output\",\n        \"-o\",\n        help=\"An optional filename to write the deployment file to.\",\n    ),\n    _apply: bool = typer.Option(\n        False,\n        \"--apply\",\n        \"-a\",\n        help=(\n            \"An optional flag to automatically register the resulting deployment with\"\n            \" the API.\"\n        ),\n    ),\n    param: List[str] = typer.Option(\n        None,\n        \"--param\",\n        help=(\n            \"An optional parameter override, values are parsed as JSON strings e.g.\"\n            \" --param question=ultimate --param answer=42\"\n        ),\n    ),\n    params: str = typer.Option(\n        None,\n        \"--params\",\n        help=(\n            \"An optional parameter override in a JSON string format e.g.\"\n            ' --params=\\'{\"question\": \"ultimate\", \"answer\": 42}\\''\n        ),\n    ),\n    no_schedule: bool = typer.Option(\n        False,\n        \"--no-schedule\",\n        help=\"An optional flag to disable scheduling for this deployment.\",\n    ),\n):\n    \"\"\"\n    Generate a deployment YAML from /path/to/file.py:flow_function\n    \"\"\"\n    # validate inputs\n    if not name:\n        exit_with_error(\n            \"A name for this deployment must be provided with the '--name' flag.\"\n        )\n\n    if (\n        len([value for value in (cron, rrule, interval) if value is not None])\n        + (1 if no_schedule else 0)\n        > 1\n    ):\n        exit_with_error(\"Only one schedule type can be provided.\")\n\n    if infra_block and infra_type:\n        exit_with_error(\n            \"Only one of `infra` or `infra_block` can be provided, please choose one.\"\n        )\n\n    output_file = None\n    if output:\n        output_file = Path(output)\n        if output_file.suffix and output_file.suffix != \".yaml\":\n            exit_with_error(\"Output file must be a '.yaml' file.\")\n        else:\n            output_file = output_file.with_suffix(\".yaml\")\n\n    # validate flow\n    try:\n        fpath, obj_name = entrypoint.rsplit(\":\", 1)\n    except ValueError as exc:\n        if str(exc) == \"not enough values to unpack (expected 2, got 1)\":\n            missing_flow_name_msg = (\n                \"Your flow entrypoint must include the name of the function that is\"\n                f\" the entrypoint to your flow.\\nTry {entrypoint}:<flow_name>\"\n            )\n            exit_with_error(missing_flow_name_msg)\n        else:\n            raise exc\n    try:\n        flow = await run_sync_in_worker_thread(load_flow_from_entrypoint, entrypoint)\n    except Exception as exc:\n        exit_with_error(exc)\n    app.console.print(f\"Found flow {flow.name!r}\", style=\"green\")\n    infra_overrides = {}\n    for override in overrides or []:\n        key, value = override.split(\"=\", 1)\n        infra_overrides[key] = value\n\n    if infra_block:\n        infrastructure = await Block.load(infra_block)\n    elif infra_type:\n        # Create an instance of the given type\n        infrastructure = Block.get_block_class_from_key(infra_type)()\n    else:\n        # will reset to a default of Process is no infra is present on the\n        # server-side definition of this deployment\n        infrastructure = None\n\n    if interval_anchor and not interval:\n        exit_with_error(\"An anchor date can only be provided with an interval schedule\")\n\n    schedule = None\n    if cron:\n        cron_kwargs = {\"cron\": cron, \"timezone\": timezone}\n        schedule = CronSchedule(\n            **{k: v for k, v in cron_kwargs.items() if v is not None}\n        )\n    elif interval:\n        interval_kwargs = {\n            \"interval\": timedelta(seconds=interval),\n            \"anchor_date\": interval_anchor,\n            \"timezone\": timezone,\n        }\n        schedule = IntervalSchedule(\n            **{k: v for k, v in interval_kwargs.items() if v is not None}\n        )\n    elif rrule:\n        try:\n            schedule = RRuleSchedule(**json.loads(rrule))\n            if timezone:\n                # override timezone if specified via CLI argument\n                schedule.timezone = timezone\n        except json.JSONDecodeError:\n            schedule = RRuleSchedule(rrule=rrule, timezone=timezone)\n\n    # parse storage_block\n    if storage_block:\n        block_type, block_name, *block_path = storage_block.split(\"/\")\n        if block_path and path:\n            exit_with_error(\n                \"Must provide a `path` explicitly or provide one on the storage block\"\n                \" specification, but not both.\"\n            )\n        elif not path:\n            path = \"/\".join(block_path)\n        storage_block = f\"{block_type}/{block_name}\"\n        storage = await Block.load(storage_block)\n    else:\n        storage = None\n\n    if create_default_ignore_file(path=\".\"):\n        app.console.print(\n            (\n                \"Default '.prefectignore' file written to\"\n                f\" {(Path('.') / '.prefectignore').absolute()}\"\n            ),\n            style=\"green\",\n        )\n\n    if param and (params is not None):\n        exit_with_error(\"Can only pass one of `param` or `params` options\")\n\n    parameters = dict()\n\n    if param:\n        for p in param or []:\n            k, unparsed_value = p.split(\"=\", 1)\n            try:\n                v = json.loads(unparsed_value)\n                app.console.print(\n                    f\"The parameter value {unparsed_value} is parsed as a JSON string\"\n                )\n            except json.JSONDecodeError:\n                v = unparsed_value\n            parameters[k] = v\n\n    if params is not None:\n        parameters = json.loads(params)\n\n    # set up deployment object\n    entrypoint = (\n        f\"{Path(fpath).absolute().relative_to(Path('.').absolute())}:{obj_name}\"\n    )\n\n    init_kwargs = dict(\n        path=path,\n        entrypoint=entrypoint,\n        version=version,\n        storage=storage,\n        infra_overrides=infra_overrides or {},\n    )\n\n    if parameters:\n        init_kwargs[\"parameters\"] = parameters\n\n    if description:\n        init_kwargs[\"description\"] = description\n\n    # if a schedule, tags, work_queue_name, or infrastructure are not provided via CLI,\n    # we let `build_from_flow` load them from the server\n    if schedule or no_schedule:\n        init_kwargs.update(schedule=schedule)\n    if tags:\n        init_kwargs.update(tags=tags)\n    if infrastructure:\n        init_kwargs.update(infrastructure=infrastructure)\n    if work_queue_name:\n        init_kwargs.update(work_queue_name=work_queue_name)\n    if work_pool_name:\n        init_kwargs.update(work_pool_name=work_pool_name)\n\n    deployment_loc = output_file or f\"{obj_name}-deployment.yaml\"\n    deployment = await Deployment.build_from_flow(\n        flow=flow,\n        name=name,\n        output=deployment_loc,\n        skip_upload=skip_upload,\n        apply=False,\n        **init_kwargs,\n    )\n    app.console.print(\n        f\"Deployment YAML created at '{Path(deployment_loc).absolute()!s}'.\",\n        style=\"green\",\n    )\n\n    await create_work_queue_and_set_concurrency_limit(\n        deployment.work_queue_name, deployment.work_pool_name, work_queue_concurrency\n    )\n\n    # we process these separately for informative output\n    if not skip_upload:\n        if (\n            deployment.storage\n            and \"put-directory\" in deployment.storage.get_block_capabilities()\n        ):\n            file_count = await deployment.upload_to_storage()\n            if file_count:\n                app.console.print(\n                    (\n                        f\"Successfully uploaded {file_count} files to\"\n                        f\" {deployment.location}\"\n                    ),\n                    style=\"green\",\n                )\n        else:\n            app.console.print(\n                (\n                    f\"Deployment storage {deployment.storage} does not have upload\"\n                    \" capabilities; no files uploaded.  Pass --skip-upload to suppress\"\n                    \" this warning.\"\n                ),\n                style=\"green\",\n            )\n\n    if _apply:\n        async with get_client() as client:\n            await check_work_pool_exists(\n                work_pool_name=deployment.work_pool_name, client=client\n            )\n            deployment_id = await deployment.apply()\n            app.console.print(\n                (\n                    f\"Deployment '{deployment.flow_name}/{deployment.name}'\"\n                    f\" successfully created with id '{deployment_id}'.\"\n                ),\n                style=\"green\",\n            )\n            if deployment.work_pool_name is not None:\n                await _print_deployment_work_pool_instructions(\n                    work_pool_name=deployment.work_pool_name, client=client\n                )\n\n            elif deployment.work_queue_name is not None:\n                app.console.print(\n                    \"\\nTo execute flow runs from this deployment, start an agent that\"\n                    f\" pulls work from the {deployment.work_queue_name!r} work queue:\"\n                )\n                app.console.print(\n                    f\"$ prefect agent start -q {deployment.work_queue_name!r}\",\n                    style=\"blue\",\n                )\n            else:\n                app.console.print(\n                    (\n                        \"\\nThis deployment does not specify a work queue name, which\"\n                        \" means agents will not be able to pick up its runs. To add a\"\n                        \" work queue, edit the deployment spec and re-run this command,\"\n                        \" or visit the deployment in the UI.\"\n                    ),\n                    style=\"red\",\n                )\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.delete","title":"delete async","text":"

    Delete a deployment.

    \b Examples: \b $ prefect deployment delete test_flow/test_deployment $ prefect deployment delete --id dfd3e220-a130-4149-9af6-8d487e02fea6

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def delete(\n    name: Optional[str] = typer.Argument(\n        None, help=\"A deployed flow's name: <FLOW_NAME>/<DEPLOYMENT_NAME>\"\n    ),\n    deployment_id: Optional[str] = typer.Option(\n        None, \"--id\", help=\"A deployment id to search for if no name is given\"\n    ),\n):\n    \"\"\"\n    Delete a deployment.\n\n    \\b\n    Examples:\n        \\b\n        $ prefect deployment delete test_flow/test_deployment\n        $ prefect deployment delete --id dfd3e220-a130-4149-9af6-8d487e02fea6\n    \"\"\"\n    async with get_client() as client:\n        if name is None and deployment_id is not None:\n            try:\n                await client.delete_deployment(deployment_id)\n                exit_with_success(f\"Deleted deployment '{deployment_id}'.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {deployment_id!r} not found!\")\n        elif name is not None:\n            try:\n                deployment = await client.read_deployment_by_name(name)\n                await client.delete_deployment(deployment.id)\n                exit_with_success(f\"Deleted deployment '{name}'.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {name!r} not found!\")\n        else:\n            exit_with_error(\"Must provide a deployment name or id\")\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.inspect","title":"inspect async","text":"

    View details about a deployment.

    \b Example: \b $ prefect deployment inspect \"hello-world/my-deployment\" { 'id': '610df9c3-0fb4-4856-b330-67f588d20201', 'created': '2022-08-01T18:36:25.192102+00:00', 'updated': '2022-08-01T18:36:25.188166+00:00', 'name': 'my-deployment', 'description': None, 'flow_id': 'b57b0aa2-ef3a-479e-be49-381fb0483b4e', 'schedule': None, 'is_schedule_active': True, 'parameters': {'name': 'Marvin'}, 'tags': ['test'], 'parameter_openapi_schema': { 'title': 'Parameters', 'type': 'object', 'properties': { 'name': { 'title': 'name', 'type': 'string' } }, 'required': ['name'] }, 'storage_document_id': '63ef008f-1e5d-4e07-a0d4-4535731adb32', 'infrastructure_document_id': '6702c598-7094-42c8-9785-338d2ec3a028', 'infrastructure': { 'type': 'process', 'env': {}, 'labels': {}, 'name': None, 'command': ['python', '-m', 'prefect.engine'], 'stream_output': True } }

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def inspect(name: str):\n    \"\"\"\n    View details about a deployment.\n\n    \\b\n    Example:\n        \\b\n        $ prefect deployment inspect \"hello-world/my-deployment\"\n        {\n            'id': '610df9c3-0fb4-4856-b330-67f588d20201',\n            'created': '2022-08-01T18:36:25.192102+00:00',\n            'updated': '2022-08-01T18:36:25.188166+00:00',\n            'name': 'my-deployment',\n            'description': None,\n            'flow_id': 'b57b0aa2-ef3a-479e-be49-381fb0483b4e',\n            'schedule': None,\n            'is_schedule_active': True,\n            'parameters': {'name': 'Marvin'},\n            'tags': ['test'],\n            'parameter_openapi_schema': {\n                'title': 'Parameters',\n                'type': 'object',\n                'properties': {\n                    'name': {\n                        'title': 'name',\n                        'type': 'string'\n                    }\n                },\n                'required': ['name']\n            },\n            'storage_document_id': '63ef008f-1e5d-4e07-a0d4-4535731adb32',\n            'infrastructure_document_id': '6702c598-7094-42c8-9785-338d2ec3a028',\n            'infrastructure': {\n                'type': 'process',\n                'env': {},\n                'labels': {},\n                'name': None,\n                'command': ['python', '-m', 'prefect.engine'],\n                'stream_output': True\n            }\n        }\n\n    \"\"\"\n    assert_deployment_name_format(name)\n\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(name)\n        except ObjectNotFound:\n            exit_with_error(f\"Deployment {name!r} not found!\")\n\n        deployment_json = deployment.dict(json_compatible=True)\n\n        if deployment.infrastructure_document_id:\n            deployment_json[\"infrastructure\"] = Block._from_block_document(\n                await client.read_block_document(deployment.infrastructure_document_id)\n            ).dict(\n                exclude={\"_block_document_id\", \"_block_document_name\", \"_is_anonymous\"}\n            )\n\n        if client.server_type == ServerType.CLOUD:\n            deployment_json[\"automations\"] = [\n                a.dict()\n                for a in await client.read_resource_related_automations(\n                    f\"prefect.deployment.{deployment.id}\"\n                )\n            ]\n\n    app.console.print(Pretty(deployment_json))\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.ls","title":"ls async","text":"

    View all deployments or deployments for specific flows.

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def ls(flow_name: List[str] = None, by_created: bool = False):\n    \"\"\"\n    View all deployments or deployments for specific flows.\n    \"\"\"\n    async with get_client() as client:\n        deployments = await client.read_deployments(\n            flow_filter=FlowFilter(name={\"any_\": flow_name}) if flow_name else None\n        )\n        flows = {\n            flow.id: flow\n            for flow in await client.read_flows(\n                flow_filter=FlowFilter(id={\"any_\": [d.flow_id for d in deployments]})\n            )\n        }\n\n    def sort_by_name_keys(d):\n        return flows[d.flow_id].name, d.name\n\n    def sort_by_created_key(d):\n        return pendulum.now(\"utc\") - d.created\n\n    table = Table(\n        title=\"Deployments\",\n    )\n    table.add_column(\"Name\", style=\"blue\", no_wrap=True)\n    table.add_column(\"ID\", style=\"cyan\", no_wrap=True)\n\n    for deployment in sorted(\n        deployments, key=sort_by_created_key if by_created else sort_by_name_keys\n    ):\n        table.add_row(\n            f\"{flows[deployment.flow_id].name}/[bold]{deployment.name}[/]\",\n            str(deployment.id),\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.pause_schedule","title":"pause_schedule async","text":"

    Pause schedule of a given deployment.

    Source code in prefect/cli/deployment.py
    @deployment_app.command(\"pause-schedule\")\nasync def pause_schedule(\n    name: str,\n):\n    \"\"\"\n    Pause schedule of a given deployment.\n    \"\"\"\n    assert_deployment_name_format(name)\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(name)\n        except ObjectNotFound:\n            exit_with_error(f\"Deployment {name!r} not found!\")\n\n        await client.update_deployment(deployment, is_schedule_active=False)\n        exit_with_success(f\"Paused schedule for deployment {name}\")\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.resume_schedule","title":"resume_schedule async","text":"

    Resume schedule of a given deployment.

    Source code in prefect/cli/deployment.py
    @deployment_app.command(\"resume-schedule\")\nasync def resume_schedule(\n    name: str,\n):\n    \"\"\"\n    Resume schedule of a given deployment.\n    \"\"\"\n    assert_deployment_name_format(name)\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(name)\n        except ObjectNotFound:\n            exit_with_error(f\"Deployment {name!r} not found!\")\n\n        await client.update_deployment(deployment, is_schedule_active=True)\n        exit_with_success(f\"Resumed schedule for deployment {name}\")\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.run","title":"run async","text":"

    Create a flow run for the given flow and deployment.

    The flow run will be scheduled to run immediately unless --start-in or --start-at is specified. The flow run will not execute until an agent starts.

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def run(\n    name: Optional[str] = typer.Argument(\n        None, help=\"A deployed flow's name: <FLOW_NAME>/<DEPLOYMENT_NAME>\"\n    ),\n    deployment_id: Optional[str] = typer.Option(\n        None, \"--id\", help=\"A deployment id to search for if no name is given\"\n    ),\n    params: List[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--param\",\n        help=(\n            \"A key, value pair (key=value) specifying a flow parameter. The value will\"\n            \" be interpreted as JSON. May be passed multiple times to specify multiple\"\n            \" parameter values.\"\n        ),\n    ),\n    multiparams: Optional[str] = typer.Option(\n        None,\n        \"--params\",\n        help=(\n            \"A mapping of parameters to values. To use a stdin, pass '-'. Any \"\n            \"parameters passed with `--param` will take precedence over these values.\"\n        ),\n    ),\n    start_in: Optional[str] = typer.Option(\n        None,\n        \"--start-in\",\n    ),\n    start_at: Optional[str] = typer.Option(\n        None,\n        \"--start-at\",\n    ),\n):\n    \"\"\"\n    Create a flow run for the given flow and deployment.\n\n    The flow run will be scheduled to run immediately unless `--start-in` or `--start-at` is specified.\n    The flow run will not execute until an agent starts.\n    \"\"\"\n    import dateparser\n\n    now = pendulum.now(\"UTC\")\n\n    multi_params = {}\n    if multiparams:\n        if multiparams == \"-\":\n            multiparams = sys.stdin.read()\n            if not multiparams:\n                exit_with_error(\"No data passed to stdin\")\n\n        try:\n            multi_params = json.loads(multiparams)\n        except ValueError as exc:\n            exit_with_error(f\"Failed to parse JSON: {exc}\")\n\n    cli_params = _load_json_key_values(params, \"parameter\")\n    conflicting_keys = set(cli_params.keys()).intersection(multi_params.keys())\n    if conflicting_keys:\n        app.console.print(\n            \"The following parameters were specified by `--param` and `--params`, the \"\n            f\"`--param` value will be used: {conflicting_keys}\"\n        )\n    parameters = {**multi_params, **cli_params}\n\n    if start_in and start_at:\n        exit_with_error(\n            \"Only one of `--start-in` or `--start-at` can be set, not both.\"\n        )\n\n    elif start_in is None and start_at is None:\n        scheduled_start_time = now\n        human_dt_diff = \" (now)\"\n    else:\n        if start_in:\n            start_time_raw = \"in \" + start_in\n        else:\n            start_time_raw = \"at \" + start_at\n        with warnings.catch_warnings():\n            # PyTZ throws a warning based on dateparser usage of the library\n            # See https://github.com/scrapinghub/dateparser/issues/1089\n            warnings.filterwarnings(\"ignore\", module=\"dateparser\")\n\n            try:\n                start_time_parsed = dateparser.parse(\n                    start_time_raw,\n                    settings={\n                        \"TO_TIMEZONE\": \"UTC\",\n                        \"RETURN_AS_TIMEZONE_AWARE\": False,\n                        \"PREFER_DATES_FROM\": \"future\",\n                        \"RELATIVE_BASE\": datetime.fromtimestamp(\n                            now.timestamp(), tz=pendulum.tz.UTC\n                        ),\n                    },\n                )\n\n            except Exception as exc:\n                exit_with_error(f\"Failed to parse '{start_time_raw!r}': {exc!s}\")\n\n        if start_time_parsed is None:\n            exit_with_error(f\"Unable to parse scheduled start time {start_time_raw!r}.\")\n\n        scheduled_start_time = pendulum.instance(start_time_parsed)\n        human_dt_diff = (\n            \" (\" + pendulum.format_diff(scheduled_start_time.diff(now)) + \")\"\n        )\n\n    async with get_client() as client:\n        deployment = await get_deployment(client, name, deployment_id)\n        flow = await client.read_flow(deployment.flow_id)\n\n        deployment_parameters = deployment.parameter_openapi_schema[\"properties\"].keys()\n        unknown_keys = set(parameters.keys()).difference(deployment_parameters)\n        if unknown_keys:\n            available_parameters = (\n                (\n                    \"The following parameters are available on the deployment: \"\n                    + listrepr(deployment_parameters, sep=\", \")\n                )\n                if deployment_parameters\n                else \"This deployment does not accept parameters.\"\n            )\n\n            exit_with_error(\n                \"The following parameters were specified but not found on the \"\n                f\"deployment: {listrepr(unknown_keys, sep=', ')}\"\n                f\"\\n{available_parameters}\"\n            )\n\n        app.console.print(\n            f\"Creating flow run for deployment '{flow.name}/{deployment.name}'...\",\n        )\n\n        try:\n            flow_run = await client.create_flow_run_from_deployment(\n                deployment.id,\n                parameters=parameters,\n                state=Scheduled(scheduled_time=scheduled_start_time),\n            )\n        except PrefectHTTPStatusError as exc:\n            detail = exc.response.json().get(\"detail\")\n            if detail:\n                exit_with_error(\n                    exc.response.json()[\"detail\"],\n                )\n            else:\n                raise\n\n    if PREFECT_UI_URL:\n        run_url = f\"{PREFECT_UI_URL.value()}/flow-runs/flow-run/{flow_run.id}\"\n    else:\n        run_url = \"<no dashboard available>\"\n\n    datetime_local_tz = scheduled_start_time.in_tz(pendulum.tz.local_timezone())\n    scheduled_display = (\n        datetime_local_tz.to_datetime_string()\n        + \" \"\n        + datetime_local_tz.tzname()\n        + human_dt_diff\n    )\n\n    app.console.print(f\"Created flow run {flow_run.name!r}.\")\n    app.console.print(\n        textwrap.dedent(\n            f\"\"\"\n        \u2514\u2500\u2500 UUID: {flow_run.id}\n        \u2514\u2500\u2500 Parameters: {flow_run.parameters}\n        \u2514\u2500\u2500 Scheduled start time: {scheduled_display}\n        \u2514\u2500\u2500 URL: {run_url}\n        \"\"\"\n        ).strip()\n    )\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.set_schedule","title":"set_schedule async","text":"

    Set schedule for a given deployment.

    Source code in prefect/cli/deployment.py
    @deployment_app.command(\"set-schedule\")\nasync def set_schedule(\n    name: str,\n    interval: Optional[float] = typer.Option(\n        None,\n        \"--interval\",\n        help=\"An interval to schedule on, specified in seconds\",\n        min=0.0001,\n    ),\n    interval_anchor: Optional[str] = typer.Option(\n        None,\n        \"--anchor-date\",\n        help=\"The anchor date for an interval schedule\",\n    ),\n    rrule_string: Optional[str] = typer.Option(\n        None, \"--rrule\", help=\"Deployment schedule rrule string\"\n    ),\n    cron_string: Optional[str] = typer.Option(\n        None, \"--cron\", help=\"Deployment schedule cron string\"\n    ),\n    cron_day_or: Optional[str] = typer.Option(\n        None,\n        \"--day_or\",\n        help=\"Control how croniter handles `day` and `day_of_week` entries\",\n    ),\n    timezone: Optional[str] = typer.Option(\n        None,\n        \"--timezone\",\n        help=\"Deployment schedule timezone string e.g. 'America/New_York'\",\n    ),\n    no_schedule: bool = typer.Option(\n        False,\n        \"--no-schedule\",\n        help=\"An optional flag to disable scheduling for this deployment.\",\n    ),\n):\n    \"\"\"\n    Set schedule for a given deployment.\n    \"\"\"\n    assert_deployment_name_format(name)\n\n    if (\n        sum(option is not None for option in [interval, rrule_string, cron_string])\n        + (1 if no_schedule else 0)\n        != 1\n    ):\n        exit_with_error(\n            \"Exactly one of `--interval`, `--rrule`, `--cron` or `--no-schedule` must\"\n            \" be provided.\"\n        )\n\n    if interval_anchor and not interval:\n        exit_with_error(\"An anchor date can only be provided with an interval schedule\")\n\n    if interval is not None:\n        if interval_anchor:\n            try:\n                pendulum.parse(interval_anchor)\n            except ValueError:\n                exit_with_error(\"The anchor date must be a valid date string.\")\n        interval_schedule = {\n            \"interval\": interval,\n            \"anchor_date\": interval_anchor,\n            \"timezone\": timezone,\n        }\n        updated_schedule = IntervalSchedule(\n            **{k: v for k, v in interval_schedule.items() if v is not None}\n        )\n\n    if cron_string is not None:\n        cron_schedule = {\n            \"cron\": cron_string,\n            \"day_or\": cron_day_or,\n            \"timezone\": timezone,\n        }\n        updated_schedule = CronSchedule(\n            **{k: v for k, v in cron_schedule.items() if v is not None}\n        )\n\n    if rrule_string is not None:\n        # a timezone in the `rrule_string` gets ignored by the RRuleSchedule constructor\n        if \"TZID\" in rrule_string and not timezone:\n            exit_with_error(\n                \"You can provide a timezone by providing a dict with a `timezone` key\"\n                \" to the --rrule option. E.g. {'rrule': 'FREQ=MINUTELY;INTERVAL=5',\"\n                \" 'timezone': 'America/New_York'}.\\nAlternatively, you can provide a\"\n                \" timezone by passing in a --timezone argument.\"\n            )\n        try:\n            updated_schedule = RRuleSchedule(**json.loads(rrule_string))\n            if timezone:\n                # override timezone if specified via CLI argument\n                updated_schedule.timezone = timezone\n        except json.JSONDecodeError:\n            updated_schedule = RRuleSchedule(rrule=rrule_string, timezone=timezone)\n\n    if no_schedule:\n        updated_schedule = NoSchedule()\n\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(name)\n        except ObjectNotFound:\n            exit_with_error(f\"Deployment {name!r} not found!\")\n\n        await client.update_deployment(deployment, schedule=updated_schedule)\n        exit_with_success(\"Updated deployment schedule!\")\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.str_presenter","title":"str_presenter","text":"

    configures yaml for dumping multiline strings Ref: https://stackoverflow.com/questions/8640959/how-can-i-control-what-scalar-form-pyyaml-uses-for-my-data

    Source code in prefect/cli/deployment.py
    def str_presenter(dumper, data):\n    \"\"\"\n    configures yaml for dumping multiline strings\n    Ref: https://stackoverflow.com/questions/8640959/how-can-i-control-what-scalar-form-pyyaml-uses-for-my-data\n    \"\"\"\n    if len(data.splitlines()) > 1:  # check for multiline string\n        return dumper.represent_scalar(\"tag:yaml.org,2002:str\", data, style=\"|\")\n    return dumper.represent_scalar(\"tag:yaml.org,2002:str\", data)\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/dev/","title":"dev","text":"","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev","title":"prefect.cli.dev","text":"

    Command line interface for working with Prefect Server

    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.agent","title":"agent async","text":"

    Starts a hot-reloading development agent process.

    Source code in prefect/cli/dev.py
    @dev_app.command()\nasync def agent(\n    api_url: str = SettingsOption(PREFECT_API_URL),\n    work_queues: List[str] = typer.Option(\n        [\"default\"],\n        \"-q\",\n        \"--work-queue\",\n        help=\"One or more work queue names for the agent to pull from.\",\n    ),\n):\n    \"\"\"\n    Starts a hot-reloading development agent process.\n    \"\"\"\n    # Delayed import since this is only a 'dev' dependency\n    import watchfiles\n\n    app.console.print(\"Creating hot-reloading agent process...\")\n\n    try:\n        await watchfiles.arun_process(\n            prefect.__module_path__,\n            target=agent_process_entrypoint,\n            kwargs=dict(api=api_url, work_queues=work_queues),\n        )\n    except RuntimeError as err:\n        # a bug in watchfiles causes an 'Already borrowed' error from Rust when\n        # exiting: https://github.com/samuelcolvin/watchfiles/issues/200\n        if str(err).strip() != \"Already borrowed\":\n            raise\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.agent_process_entrypoint","title":"agent_process_entrypoint","text":"

    An entrypoint for starting an agent in a subprocess. Adds a Rich console to the Typer app, processes Typer default parameters, then starts an agent. All kwargs are forwarded to prefect.cli.agent.start.

    Source code in prefect/cli/dev.py
    def agent_process_entrypoint(**kwargs):\n    \"\"\"\n    An entrypoint for starting an agent in a subprocess. Adds a Rich console\n    to the Typer app, processes Typer default parameters, then starts an agent.\n    All kwargs are forwarded to  `prefect.cli.agent.start`.\n    \"\"\"\n    import inspect\n\n    import rich\n\n    # import locally so only the `dev` command breaks if Typer internals change\n    from typer.models import ParameterInfo\n\n    # Typer does not process default parameters when calling a function\n    # directly, so we must set `start_agent`'s default parameters manually.\n    # get the signature of the `start_agent` function\n    start_agent_signature = inspect.signature(start_agent)\n\n    # for any arguments not present in kwargs, use the default value.\n    for name, param in start_agent_signature.parameters.items():\n        if name not in kwargs:\n            # All `param.default` values for start_agent are Typer params that store the\n            # actual default value in their `default` attribute and we must call\n            # `param.default.default` to get the actual default value. We should also\n            # ensure we extract the right default if non-Typer defaults are added\n            # to `start_agent` in the future.\n            if isinstance(param.default, ParameterInfo):\n                default = param.default.default\n            else:\n                default = param.default\n\n            # Some defaults are Prefect `SettingsOption.value` methods\n            # that must be called to get the actual value.\n            kwargs[name] = default() if callable(default) else default\n\n    # add a console, because calling the agent start function directly\n    # instead of via CLI call means `app` has no `console` attached.\n    app.console = (\n        rich.console.Console(\n            highlight=False,\n            color_system=\"auto\" if PREFECT_CLI_COLORS else None,\n            soft_wrap=not PREFECT_CLI_WRAP_LINES.value(),\n        )\n        if not getattr(app, \"console\", None)\n        else app.console\n    )\n\n    try:\n        start_agent(**kwargs)  # type: ignore\n    except KeyboardInterrupt:\n        # expected when watchfiles kills the process\n        pass\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.api","title":"api async","text":"

    Starts a hot-reloading development API.

    Source code in prefect/cli/dev.py
    @dev_app.command()\nasync def api(\n    host: str = SettingsOption(PREFECT_SERVER_API_HOST),\n    port: int = SettingsOption(PREFECT_SERVER_API_PORT),\n    log_level: str = \"DEBUG\",\n    services: bool = True,\n):\n    \"\"\"\n    Starts a hot-reloading development API.\n    \"\"\"\n    import watchfiles\n\n    server_env = os.environ.copy()\n    server_env[\"PREFECT_API_SERVICES_RUN_IN_APP\"] = str(services)\n    server_env[\"PREFECT_API_SERVICES_UI\"] = \"False\"\n    server_env[\"PREFECT_UI_API_URL\"] = f\"http://{host}:{port}/api\"\n\n    command = [\n        sys.executable,\n        \"-m\",\n        \"uvicorn\",\n        \"--factory\",\n        \"prefect.server.api.server:create_app\",\n        \"--host\",\n        str(host),\n        \"--port\",\n        str(port),\n        \"--log-level\",\n        log_level.lower(),\n    ]\n\n    app.console.print(f\"Running: {' '.join(command)}\")\n    import signal\n\n    stop_event = anyio.Event()\n    start_command = partial(\n        run_process, command=command, env=server_env, stream_output=True\n    )\n\n    async with anyio.create_task_group() as tg:\n        try:\n            server_pid = await tg.start(start_command)\n            async for _ in watchfiles.awatch(\n                prefect.__module_path__, stop_event=stop_event  # type: ignore\n            ):\n                # when any watched files change, restart the server\n                app.console.print(\"Restarting Prefect Server...\")\n                os.kill(server_pid, signal.SIGTERM)  # type: ignore\n                # start a new server\n                server_pid = await tg.start(start_command)\n        except RuntimeError as err:\n            # a bug in watchfiles causes an 'Already borrowed' error from Rust when\n            # exiting: https://github.com/samuelcolvin/watchfiles/issues/200\n            if str(err).strip() != \"Already borrowed\":\n                raise\n        except KeyboardInterrupt:\n            # exit cleanly on ctrl-c by killing the server process if it's\n            # still running\n            try:\n                os.kill(server_pid, signal.SIGTERM)  # type: ignore\n            except ProcessLookupError:\n                # process already exited\n                pass\n\n            stop_event.set()\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.build_docs","title":"build_docs","text":"

    Builds REST API reference documentation for static display.

    Source code in prefect/cli/dev.py
    @dev_app.command()\ndef build_docs(\n    schema_path: str = None,\n):\n    \"\"\"\n    Builds REST API reference documentation for static display.\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n\n    from prefect.server.api.server import create_app\n\n    schema = create_app(ephemeral=True).openapi()\n\n    if not schema_path:\n        schema_path = (\n            prefect.__development_base_path__ / \"docs\" / \"api-ref\" / \"schema.json\"\n        ).absolute()\n    # overwrite info for display purposes\n    schema[\"info\"] = {}\n    with open(schema_path, \"w\") as f:\n        json.dump(schema, f)\n    app.console.print(f\"OpenAPI schema written to {schema_path}\")\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.build_image","title":"build_image","text":"

    Build a docker image for development.

    Source code in prefect/cli/dev.py
    @dev_app.command()\ndef build_image(\n    arch: str = typer.Option(\n        None,\n        help=(\n            \"The architecture to build the container for. \"\n            \"Defaults to the architecture of the host Python. \"\n            f\"[default: {platform.machine()}]\"\n        ),\n    ),\n    python_version: str = typer.Option(\n        None,\n        help=(\n            \"The Python version to build the container for. \"\n            \"Defaults to the version of the host Python. \"\n            f\"[default: {python_version_minor()}]\"\n        ),\n    ),\n    flavor: str = typer.Option(\n        None,\n        help=(\n            \"An alternative flavor to build, for example 'conda'. \"\n            \"Defaults to the standard Python base image\"\n        ),\n    ),\n    dry_run: bool = False,\n):\n    \"\"\"\n    Build a docker image for development.\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n    # TODO: Once https://github.com/tiangolo/typer/issues/354 is addressed, the\n    #       default can be set in the function signature\n    arch = arch or platform.machine()\n    python_version = python_version or python_version_minor()\n\n    tag = get_prefect_image_name(python_version=python_version, flavor=flavor)\n\n    # Here we use a subprocess instead of the docker-py client to easily stream output\n    # as it comes\n    command = [\n        \"docker\",\n        \"build\",\n        str(prefect.__development_base_path__),\n        \"--tag\",\n        tag,\n        \"--platform\",\n        f\"linux/{arch}\",\n        \"--build-arg\",\n        \"PREFECT_EXTRAS=[dev]\",\n        \"--build-arg\",\n        f\"PYTHON_VERSION={python_version}\",\n    ]\n\n    if flavor:\n        command += [\"--build-arg\", f\"BASE_IMAGE=prefect-{flavor}\"]\n\n    if dry_run:\n        print(\" \".join(command))\n        return\n\n    try:\n        subprocess.check_call(command, shell=sys.platform == \"win32\")\n    except subprocess.CalledProcessError:\n        exit_with_error(\"Failed to build image!\")\n    else:\n        exit_with_success(f\"Built image {tag!r} for linux/{arch}\")\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.container","title":"container","text":"

    Run a docker container with local code mounted and installed.

    Source code in prefect/cli/dev.py
    @dev_app.command()\ndef container(bg: bool = False, name=\"prefect-dev\", api: bool = True, tag: str = None):\n    \"\"\"\n    Run a docker container with local code mounted and installed.\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n    import docker\n    from docker.models.containers import Container\n\n    client = docker.from_env()\n\n    containers = client.containers.list()\n    container_names = {container.name for container in containers}\n    if name in container_names:\n        exit_with_error(\n            f\"Container {name!r} already exists. Specify a different name or stop \"\n            \"the existing container.\"\n        )\n\n    blocking_cmd = \"prefect dev api\" if api else \"sleep infinity\"\n    tag = tag or get_prefect_image_name()\n\n    container: Container = client.containers.create(\n        image=tag,\n        command=[\n            \"/bin/bash\",\n            \"-c\",\n            (  # noqa\n                \"pip install -e /opt/prefect/repo\\\\[dev\\\\] && touch /READY &&\"\n                f\" {blocking_cmd}\"\n            ),\n        ],\n        name=name,\n        auto_remove=True,\n        working_dir=\"/opt/prefect/repo\",\n        volumes=[f\"{prefect.__development_base_path__}:/opt/prefect/repo\"],\n        shm_size=\"4G\",\n    )\n\n    print(f\"Starting container for image {tag!r}...\")\n    container.start()\n\n    print(\"Waiting for installation to complete\", end=\"\", flush=True)\n    try:\n        ready = False\n        while not ready:\n            print(\".\", end=\"\", flush=True)\n            result = container.exec_run(\"test -f /READY\")\n            ready = result.exit_code == 0\n            if not ready:\n                time.sleep(3)\n    except BaseException:\n        print(\"\\nInterrupted. Stopping container...\")\n        container.stop()\n        raise\n\n    print(\n        textwrap.dedent(\n            f\"\"\"\n            Container {container.name!r} is ready! To connect to the container, run:\n\n                docker exec -it {container.name} /bin/bash\n            \"\"\"\n        )\n    )\n\n    if bg:\n        print(\n            textwrap.dedent(\n                f\"\"\"\n                The container will run forever. Stop the container with:\n\n                    docker stop {container.name}\n                \"\"\"\n            )\n        )\n        # Exit without stopping\n        return\n\n    try:\n        print(\"Send a keyboard interrupt to exit...\")\n        container.wait()\n    except KeyboardInterrupt:\n        pass  # Avoid showing \"Abort\"\n    finally:\n        print(\"\\nStopping container...\")\n        container.stop()\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.kubernetes_manifest","title":"kubernetes_manifest","text":"

    Generates a Kubernetes manifest for development.

    Example

    $ prefect dev kubernetes-manifest | kubectl apply -f -

    Source code in prefect/cli/dev.py
    @dev_app.command()\ndef kubernetes_manifest():\n    \"\"\"\n    Generates a Kubernetes manifest for development.\n\n    Example:\n        $ prefect dev kubernetes-manifest | kubectl apply -f -\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n\n    template = Template(\n        (\n            prefect.__module_path__ / \"cli\" / \"templates\" / \"kubernetes-dev.yaml\"\n        ).read_text()\n    )\n    manifest = template.substitute(\n        {\n            \"prefect_root_directory\": prefect.__development_base_path__,\n            \"image_name\": get_prefect_image_name(),\n        }\n    )\n    print(manifest)\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.start","title":"start async","text":"

    Starts a hot-reloading development server with API, UI, and agent processes.

    Each service has an individual command if you wish to start them separately. Each service can be excluded here as well.

    Source code in prefect/cli/dev.py
    @dev_app.command()\nasync def start(\n    exclude_api: bool = typer.Option(False, \"--no-api\"),\n    exclude_ui: bool = typer.Option(False, \"--no-ui\"),\n    exclude_agent: bool = typer.Option(False, \"--no-agent\"),\n    work_queues: List[str] = typer.Option(\n        [\"default\"],\n        \"-q\",\n        \"--work-queue\",\n        help=\"One or more work queue names for the dev agent to pull from.\",\n    ),\n):\n    \"\"\"\n    Starts a hot-reloading development server with API, UI, and agent processes.\n\n    Each service has an individual command if you wish to start them separately.\n    Each service can be excluded here as well.\n    \"\"\"\n    async with anyio.create_task_group() as tg:\n        if not exclude_api:\n            tg.start_soon(\n                partial(\n                    api,\n                    host=PREFECT_SERVER_API_HOST.value(),\n                    port=PREFECT_SERVER_API_PORT.value(),\n                )\n            )\n        if not exclude_ui:\n            tg.start_soon(ui)\n        if not exclude_agent:\n            # Hook the agent to the hosted API if running\n            if not exclude_api:\n                host = f\"http://{PREFECT_SERVER_API_HOST.value()}:{PREFECT_SERVER_API_PORT.value()}/api\"  # noqa\n            else:\n                host = PREFECT_API_URL.value()\n            tg.start_soon(agent, host, work_queues)\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.ui","title":"ui async","text":"

    Starts a hot-reloading development UI.

    Source code in prefect/cli/dev.py
    @dev_app.command()\nasync def ui():\n    \"\"\"\n    Starts a hot-reloading development UI.\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n    with tmpchdir(prefect.__development_base_path__):\n        with tmpchdir(prefect.__development_base_path__ / \"ui\"):\n            app.console.print(\"Installing npm packages...\")\n            await run_process([\"npm\", \"install\"], stream_output=True)\n\n            app.console.print(\"Starting UI development server...\")\n            await run_process(command=[\"npm\", \"run\", \"serve\"], stream_output=True)\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/flow/","title":"flow","text":"","tags":["Python API","flows","CLI"]},{"location":"api-ref/prefect/cli/flow/#prefect.cli.flow","title":"prefect.cli.flow","text":"

    Command line interface for working with flows.

    ","tags":["Python API","flows","CLI"]},{"location":"api-ref/prefect/cli/flow/#prefect.cli.flow.ls","title":"ls async","text":"

    View flows.

    Source code in prefect/cli/flow.py
    @flow_app.command()\nasync def ls(\n    limit: int = 15,\n):\n    \"\"\"\n    View flows.\n    \"\"\"\n    async with get_client() as client:\n        flows = await client.read_flows(\n            limit=limit,\n            sort=FlowSort.CREATED_DESC,\n        )\n\n    table = Table(title=\"Flows\")\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Created\", no_wrap=True)\n\n    for flow in flows:\n        table.add_row(\n            str(flow.id),\n            str(flow.name),\n            str(flow.created),\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","flows","CLI"]},{"location":"api-ref/prefect/cli/flow/#prefect.cli.flow.serve","title":"serve async","text":"

    Serve a flow via an entrypoint.

    Source code in prefect/cli/flow.py
    @flow_app.command()\nasync def serve(\n    entrypoint: str = typer.Argument(\n        ...,\n        help=(\n            \"The path to a file containing a flow and the name of the flow function in\"\n            \" the format `./path/to/file.py:flow_func_name`.\"\n        ),\n    ),\n    name: str = typer.Option(\n        ...,\n        \"--name\",\n        \"-n\",\n        help=\"The name to give the deployment created for the flow.\",\n    ),\n    description: Optional[str] = typer.Option(\n        None,\n        \"--description\",\n        \"-d\",\n        help=(\n            \"The description to give the created deployment. If not provided, the\"\n            \" description will be populated from the flow's description.\"\n        ),\n    ),\n    version: Optional[str] = typer.Option(\n        None, \"-v\", \"--version\", help=\"A version to give the created deployment.\"\n    ),\n    tags: Optional[List[str]] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=\"One or more optional tags to apply to the created deployment.\",\n    ),\n    cron: Optional[str] = typer.Option(\n        None,\n        \"--cron\",\n        help=(\n            \"A cron string that will be used to set a schedule for the created\"\n            \" deployment.\"\n        ),\n    ),\n    interval: Optional[int] = typer.Option(\n        None,\n        \"--interval\",\n        help=(\n            \"An integer specifying an interval (in seconds) between scheduled runs of\"\n            \" the flow.\"\n        ),\n    ),\n    interval_anchor: Optional[str] = typer.Option(\n        None, \"--anchor-date\", help=\"The start date for an interval schedule.\"\n    ),\n    rrule: Optional[str] = typer.Option(\n        None,\n        \"--rrule\",\n        help=\"An RRule that will be used to set a schedule for the created deployment.\",\n    ),\n    timezone: Optional[str] = typer.Option(\n        None,\n        \"--timezone\",\n        help=\"Timezone to used scheduling flow runs e.g. 'America/New_York'\",\n    ),\n    pause_on_shutdown: bool = typer.Option(\n        True,\n        help=(\n            \"If set, provided schedule will be paused when the serve command is\"\n            \" stopped. If not set, the schedules will continue running.\"\n        ),\n    ),\n):\n    \"\"\"\n    Serve a flow via an entrypoint.\n    \"\"\"\n    runner = Runner(name=name, pause_on_shutdown=pause_on_shutdown)\n    try:\n        schedule = None\n        if interval or cron or rrule:\n            schedule = construct_schedule(\n                interval=interval,\n                cron=cron,\n                rrule=rrule,\n                timezone=timezone,\n                anchor_date=interval_anchor,\n            )\n        runner_deployment = RunnerDeployment.from_entrypoint(\n            entrypoint=entrypoint,\n            name=name,\n            schedule=schedule,\n            description=description,\n            tags=tags or [],\n            version=version,\n        )\n    except (MissingFlowError, ValueError) as exc:\n        exit_with_error(str(exc))\n    deployment_id = await runner.add_deployment(runner_deployment)\n\n    help_message = (\n        f\"[green]Your flow {runner_deployment.flow_name!r} is being served and polling\"\n        \" for scheduled runs!\\n[/]\\nTo trigger a run for this flow, use the following\"\n        \" command:\\n[blue]\\n\\t$ prefect deployment run\"\n        f\" '{runner_deployment.flow_name}/{name}'\\n[/]\"\n    )\n    if PREFECT_UI_URL:\n        help_message += (\n            \"\\nYou can also run your flow via the Prefect UI:\"\n            f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_id}[/]\\n\"\n        )\n    app.console.print(Panel(help_message))\n    await runner.start()\n
    ","tags":["Python API","flows","CLI"]},{"location":"api-ref/prefect/cli/flow_run/","title":"flow_run","text":"","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run","title":"prefect.cli.flow_run","text":"

    Command line interface for working with flow runs

    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.cancel","title":"cancel async","text":"

    Cancel a flow run by ID.

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def cancel(id: UUID):\n    \"\"\"Cancel a flow run by ID.\"\"\"\n    async with get_client() as client:\n        cancelling_state = State(type=StateType.CANCELLING)\n        try:\n            result = await client.set_flow_run_state(\n                flow_run_id=id, state=cancelling_state\n            )\n        except ObjectNotFound:\n            exit_with_error(f\"Flow run '{id}' not found!\")\n\n    if result.status == SetStateStatus.ABORT:\n        exit_with_error(\n            f\"Flow run '{id}' was unable to be cancelled. Reason:\"\n            f\" '{result.details.reason}'\"\n        )\n\n    exit_with_success(f\"Flow run '{id}' was successfully scheduled for cancellation.\")\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.delete","title":"delete async","text":"

    Delete a flow run by ID.

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def delete(id: UUID):\n    \"\"\"\n    Delete a flow run by ID.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.delete_flow_run(id)\n        except ObjectNotFound:\n            exit_with_error(f\"Flow run '{id}' not found!\")\n\n    exit_with_success(f\"Successfully deleted flow run '{id}'.\")\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.inspect","title":"inspect async","text":"

    View details about a flow run.

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def inspect(id: UUID):\n    \"\"\"\n    View details about a flow run.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            flow_run = await client.read_flow_run(id)\n        except httpx.HTTPStatusError as exc:\n            if exc.response.status_code == status.HTTP_404_NOT_FOUND:\n                exit_with_error(f\"Flow run {id!r} not found!\")\n            else:\n                raise\n\n    app.console.print(Pretty(flow_run))\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.logs","title":"logs async","text":"

    View logs for a flow run.

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def logs(\n    id: UUID,\n    head: bool = typer.Option(\n        False,\n        \"--head\",\n        \"-h\",\n        help=(\n            f\"Show the first {LOGS_WITH_LIMIT_FLAG_DEFAULT_NUM_LOGS} logs instead of\"\n            \" all logs.\"\n        ),\n    ),\n    num_logs: int = typer.Option(\n        None,\n        \"--num-logs\",\n        \"-n\",\n        help=(\n            \"Number of logs to show when using the --head or --tail flag. If None,\"\n            f\" defaults to {LOGS_WITH_LIMIT_FLAG_DEFAULT_NUM_LOGS}.\"\n        ),\n        min=1,\n    ),\n    reverse: bool = typer.Option(\n        False,\n        \"--reverse\",\n        \"-r\",\n        help=\"Reverse the logs order to print the most recent logs first\",\n    ),\n    tail: bool = typer.Option(\n        False,\n        \"--tail\",\n        \"-t\",\n        help=(\n            f\"Show the last {LOGS_WITH_LIMIT_FLAG_DEFAULT_NUM_LOGS} logs instead of\"\n            \" all logs.\"\n        ),\n    ),\n):\n    \"\"\"\n    View logs for a flow run.\n    \"\"\"\n    # Pagination - API returns max 200 (LOGS_DEFAULT_PAGE_SIZE) logs at a time\n    offset = 0\n    more_logs = True\n    num_logs_returned = 0\n\n    # if head and tail flags are being used together\n    if head and tail:\n        exit_with_error(\"Please provide either a `head` or `tail` option but not both.\")\n\n    user_specified_num_logs = (\n        num_logs or LOGS_WITH_LIMIT_FLAG_DEFAULT_NUM_LOGS\n        if head or tail or num_logs\n        else None\n    )\n\n    # if using tail update offset according to LOGS_DEFAULT_PAGE_SIZE\n    if tail:\n        offset = max(0, user_specified_num_logs - LOGS_DEFAULT_PAGE_SIZE)\n\n    log_filter = LogFilter(flow_run_id={\"any_\": [id]})\n\n    async with get_client() as client:\n        # Get the flow run\n        try:\n            flow_run = await client.read_flow_run(id)\n        except ObjectNotFound:\n            exit_with_error(f\"Flow run {str(id)!r} not found!\")\n\n        while more_logs:\n            num_logs_to_return_from_page = (\n                LOGS_DEFAULT_PAGE_SIZE\n                if user_specified_num_logs is None\n                else min(\n                    LOGS_DEFAULT_PAGE_SIZE, user_specified_num_logs - num_logs_returned\n                )\n            )\n\n            # Get the next page of logs\n            page_logs = await client.read_logs(\n                log_filter=log_filter,\n                limit=num_logs_to_return_from_page,\n                offset=offset,\n                sort=(\n                    LogSort.TIMESTAMP_DESC if reverse or tail else LogSort.TIMESTAMP_ASC\n                ),\n            )\n\n            for log in reversed(page_logs) if tail and not reverse else page_logs:\n                app.console.print(\n                    # Print following the flow run format (declared in logging.yml)\n                    (\n                        f\"{pendulum.instance(log.timestamp).to_datetime_string()}.{log.timestamp.microsecond // 1000:03d} |\"\n                        f\" {logging.getLevelName(log.level):7s} | Flow run\"\n                        f\" {flow_run.name!r} - {log.message}\"\n                    ),\n                    soft_wrap=True,\n                )\n\n            # Update the number of logs retrieved\n            num_logs_returned += num_logs_to_return_from_page\n\n            if tail:\n                #  If the current offset is not 0, update the offset for the next page\n                if offset != 0:\n                    offset = (\n                        0\n                        # Reset the offset to 0 if there are less logs than the LOGS_DEFAULT_PAGE_SIZE to get the remaining log\n                        if offset < LOGS_DEFAULT_PAGE_SIZE\n                        else offset - LOGS_DEFAULT_PAGE_SIZE\n                    )\n                else:\n                    more_logs = False\n            else:\n                if len(page_logs) == LOGS_DEFAULT_PAGE_SIZE:\n                    offset += LOGS_DEFAULT_PAGE_SIZE\n                else:\n                    # No more logs to show, exit\n                    more_logs = False\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.ls","title":"ls async","text":"

    View recent flow runs or flow runs for specific flows

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def ls(\n    flow_name: List[str] = typer.Option(None, help=\"Name of the flow\"),\n    limit: int = typer.Option(15, help=\"Maximum number of flow runs to list\"),\n    state: List[str] = typer.Option(None, help=\"Name of the flow run's state\"),\n    state_type: List[StateType] = typer.Option(\n        None, help=\"Type of the flow run's state\"\n    ),\n):\n    \"\"\"\n    View recent flow runs or flow runs for specific flows\n    \"\"\"\n\n    state_filter = {}\n    if state:\n        state_filter[\"name\"] = {\"any_\": state}\n    if state_type:\n        state_filter[\"type\"] = {\"any_\": state_type}\n\n    async with get_client() as client:\n        flow_runs = await client.read_flow_runs(\n            flow_filter=FlowFilter(name={\"any_\": flow_name}) if flow_name else None,\n            flow_run_filter=FlowRunFilter(state=state_filter) if state_filter else None,\n            limit=limit,\n            sort=FlowRunSort.EXPECTED_START_TIME_DESC,\n        )\n        flows_by_id = {\n            flow.id: flow\n            for flow in await client.read_flows(\n                flow_filter=FlowFilter(id={\"any_\": [run.flow_id for run in flow_runs]})\n            )\n        }\n\n    table = Table(title=\"Flow Runs\")\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Flow\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"State\", no_wrap=True)\n    table.add_column(\"When\", style=\"bold\", no_wrap=True)\n\n    for flow_run in sorted(flow_runs, key=lambda d: d.created, reverse=True):\n        flow = flows_by_id[flow_run.flow_id]\n        timestamp = (\n            flow_run.state.state_details.scheduled_time\n            if flow_run.state.is_scheduled()\n            else flow_run.state.timestamp\n        )\n        table.add_row(\n            str(flow_run.id),\n            str(flow.name),\n            str(flow_run.name),\n            str(flow_run.state.type.value),\n            pendulum.instance(timestamp).diff_for_humans(),\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/kubernetes/","title":"kubernetes","text":"","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/kubernetes/#prefect.cli.kubernetes","title":"prefect.cli.kubernetes","text":"

    Command line interface for working with Prefect on Kubernetes

    ","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/kubernetes/#prefect.cli.kubernetes.manifest_agent","title":"manifest_agent","text":"

    Generates a manifest for deploying Agent on Kubernetes.

    Example

    $ prefect kubernetes manifest agent | kubectl apply -f -

    Source code in prefect/cli/kubernetes.py
    @manifest_app.command(\"agent\")\ndef manifest_agent(\n    api_url: str = SettingsOption(PREFECT_API_URL),\n    api_key: str = SettingsOption(PREFECT_API_KEY),\n    image_tag: str = typer.Option(\n        get_prefect_image_name(),\n        \"-i\",\n        \"--image-tag\",\n        help=\"The tag of a Docker image to use for the Agent.\",\n    ),\n    namespace: str = typer.Option(\n        \"default\",\n        \"-n\",\n        \"--namespace\",\n        help=\"A Kubernetes namespace to create agent in.\",\n    ),\n    work_queue: str = typer.Option(\n        \"kubernetes\",\n        \"-q\",\n        \"--work-queue\",\n        help=\"A work queue name for the agent to pull from.\",\n    ),\n):\n    \"\"\"\n    Generates a manifest for deploying Agent on Kubernetes.\n\n    Example:\n        $ prefect kubernetes manifest agent | kubectl apply -f -\n    \"\"\"\n\n    template = Template(\n        (\n            prefect.__module_path__ / \"cli\" / \"templates\" / \"kubernetes-agent.yaml\"\n        ).read_text()\n    )\n    manifest = template.substitute(\n        {\n            \"api_url\": api_url,\n            \"api_key\": api_key,\n            \"image_name\": image_tag,\n            \"namespace\": namespace,\n            \"work_queue\": work_queue,\n        }\n    )\n    print(manifest)\n
    ","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/kubernetes/#prefect.cli.kubernetes.manifest_flow_run_job","title":"manifest_flow_run_job async","text":"

    Prints the default KubernetesJob Job manifest.

    Use this file to fully customize your KubernetesJob deployments.

    \b Example: \b $ prefect kubernetes manifest flow-run-job

    \b Output, a YAML file: \b apiVersion: batch/v1 kind: Job ...

    Source code in prefect/cli/kubernetes.py
    @manifest_app.command(\"flow-run-job\")\nasync def manifest_flow_run_job():\n    \"\"\"\n    Prints the default KubernetesJob Job manifest.\n\n    Use this file to fully customize your `KubernetesJob` deployments.\n\n    \\b\n    Example:\n        \\b\n        $ prefect kubernetes manifest flow-run-job\n\n    \\b\n    Output, a YAML file:\n        \\b\n        apiVersion: batch/v1\n        kind: Job\n        ...\n    \"\"\"\n\n    KubernetesJob.base_job_manifest()\n\n    output = yaml.dump(KubernetesJob.base_job_manifest())\n\n    # add some commentary where appropriate\n    output = output.replace(\n        \"metadata:\\n  labels:\",\n        \"metadata:\\n  # labels are required, even if empty\\n  labels:\",\n    )\n    output = output.replace(\n        \"containers:\\n\",\n        \"containers:  # the first container is required\\n\",\n    )\n    output = output.replace(\n        \"env: []\\n\",\n        \"env: []  # env is required, even if empty\\n\",\n    )\n\n    print(output)\n
    ","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/kubernetes/#prefect.cli.kubernetes.manifest_server","title":"manifest_server","text":"

    Generates a manifest for deploying Prefect on Kubernetes.

    Example

    $ prefect kubernetes manifest server | kubectl apply -f -

    Source code in prefect/cli/kubernetes.py
    @manifest_app.command(\"server\")\ndef manifest_server(\n    image_tag: str = typer.Option(\n        get_prefect_image_name(),\n        \"-i\",\n        \"--image-tag\",\n        help=\"The tag of a Docker image to use for the server.\",\n    ),\n    namespace: str = typer.Option(\n        \"default\",\n        \"-n\",\n        \"--namespace\",\n        help=\"A Kubernetes namespace to create the server in.\",\n    ),\n    log_level: str = SettingsOption(PREFECT_LOGGING_SERVER_LEVEL),\n):\n    \"\"\"\n    Generates a manifest for deploying Prefect on Kubernetes.\n\n    Example:\n        $ prefect kubernetes manifest server | kubectl apply -f -\n    \"\"\"\n\n    template = Template(\n        (\n            prefect.__module_path__ / \"cli\" / \"templates\" / \"kubernetes-server.yaml\"\n        ).read_text()\n    )\n    manifest = template.substitute(\n        {\n            \"image_name\": image_tag,\n            \"namespace\": namespace,\n            \"log_level\": log_level,\n        }\n    )\n    print(manifest)\n
    ","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/profile/","title":"profile","text":"","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile","title":"prefect.cli.profile","text":"

    Command line interface for working with profiles.

    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.create","title":"create","text":"

    Create a new profile.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef create(\n    name: str,\n    from_name: str = typer.Option(None, \"--from\", help=\"Copy an existing profile.\"),\n):\n    \"\"\"\n    Create a new profile.\n    \"\"\"\n\n    profiles = prefect.settings.load_profiles()\n    if name in profiles:\n        app.console.print(\n            textwrap.dedent(\n                f\"\"\"\n                [red]Profile {name!r} already exists.[/red]\n                To create a new profile, remove the existing profile first:\n\n                    prefect profile delete {name!r}\n                \"\"\"\n            ).strip()\n        )\n        raise typer.Exit(1)\n\n    if from_name:\n        if from_name not in profiles:\n            exit_with_error(f\"Profile {from_name!r} not found.\")\n\n        # Create a copy of the profile with a new name and add to the collection\n        profiles.add_profile(profiles[from_name].copy(update={\"name\": name}))\n    else:\n        profiles.add_profile(prefect.settings.Profile(name=name, settings={}))\n\n    prefect.settings.save_profiles(profiles)\n\n    app.console.print(\n        textwrap.dedent(\n            f\"\"\"\n            Created profile with properties:\n                name - {name!r}\n                from name - {from_name or None}\n\n            Use created profile for future, subsequent commands:\n                prefect profile use {name!r}\n\n            Use created profile temporarily for a single command:\n                prefect -p {name!r} config view\n            \"\"\"\n        )\n    )\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.delete","title":"delete","text":"

    Delete the given profile.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef delete(name: str):\n    \"\"\"\n    Delete the given profile.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    if name not in profiles:\n        exit_with_error(f\"Profile {name!r} not found.\")\n\n    current_profile = prefect.context.get_settings_context().profile\n    if current_profile.name == name:\n        exit_with_error(\n            f\"Profile {name!r} is the active profile. You must switch profiles before\"\n            \" it can be deleted.\"\n        )\n\n    profiles.remove_profile(name)\n\n    verb = \"Removed\"\n    if name == \"default\":\n        verb = \"Reset\"\n\n    prefect.settings.save_profiles(profiles)\n    exit_with_success(f\"{verb} profile {name!r}.\")\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.inspect","title":"inspect","text":"

    Display settings from a given profile; defaults to active.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef inspect(\n    name: Optional[str] = typer.Argument(\n        None, help=\"Name of profile to inspect; defaults to active profile.\"\n    )\n):\n    \"\"\"\n    Display settings from a given profile; defaults to active.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    if name is None:\n        current_profile = prefect.context.get_settings_context().profile\n        if not current_profile:\n            exit_with_error(\"No active profile set - please provide a name to inspect.\")\n        name = current_profile.name\n        print(f\"No name provided, defaulting to {name!r}\")\n    if name not in profiles:\n        exit_with_error(f\"Profile {name!r} not found.\")\n\n    if not profiles[name].settings:\n        # TODO: Consider instructing on how to add settings.\n        print(f\"Profile {name!r} is empty.\")\n\n    for setting, value in profiles[name].settings.items():\n        app.console.print(f\"{setting.name}='{value}'\")\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.ls","title":"ls","text":"

    List profile names.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef ls():\n    \"\"\"\n    List profile names.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    current_profile = prefect.context.get_settings_context().profile\n    current_name = current_profile.name if current_profile is not None else None\n\n    table = Table(caption=\"* active profile\")\n    table.add_column(\n        \"[#024dfd]Available Profiles:\", justify=\"right\", style=\"#8ea0ae\", no_wrap=True\n    )\n\n    for name in profiles:\n        if name == current_name:\n            table.add_row(f\"[green]  * {name}[/green]\")\n        else:\n            table.add_row(f\"  {name}\")\n    app.console.print(table)\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.rename","title":"rename","text":"

    Change the name of a profile.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef rename(name: str, new_name: str):\n    \"\"\"\n    Change the name of a profile.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    if name not in profiles:\n        exit_with_error(f\"Profile {name!r} not found.\")\n\n    if new_name in profiles:\n        exit_with_error(f\"Profile {new_name!r} already exists.\")\n\n    profiles.add_profile(profiles[name].copy(update={\"name\": new_name}))\n    profiles.remove_profile(name)\n\n    # If the active profile was renamed switch the active profile to the new name.\n    prefect.context.get_settings_context().profile\n    if profiles.active_name == name:\n        profiles.set_active(new_name)\n    if os.environ.get(\"PREFECT_PROFILE\") == name:\n        app.console.print(\n            f\"You have set your current profile to {name!r} with the \"\n            \"PREFECT_PROFILE environment variable. You must update this variable to \"\n            f\"{new_name!r} to continue using the profile.\"\n        )\n\n    prefect.settings.save_profiles(profiles)\n    exit_with_success(f\"Renamed profile {name!r} to {new_name!r}.\")\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.use","title":"use async","text":"

    Set the given profile to active.

    Source code in prefect/cli/profile.py
    @profile_app.command()\nasync def use(name: str):\n    \"\"\"\n    Set the given profile to active.\n    \"\"\"\n    status_messages = {\n        ConnectionStatus.CLOUD_CONNECTED: (\n            exit_with_success,\n            f\"Connected to Prefect Cloud using profile {name!r}\",\n        ),\n        ConnectionStatus.CLOUD_ERROR: (\n            exit_with_error,\n            f\"Error connecting to Prefect Cloud using profile {name!r}\",\n        ),\n        ConnectionStatus.CLOUD_UNAUTHORIZED: (\n            exit_with_error,\n            f\"Error authenticating with Prefect Cloud using profile {name!r}\",\n        ),\n        ConnectionStatus.ORION_CONNECTED: (\n            exit_with_success,\n            f\"Connected to Prefect server using profile {name!r}\",\n        ),\n        ConnectionStatus.ORION_ERROR: (\n            exit_with_error,\n            f\"Error connecting to Prefect server using profile {name!r}\",\n        ),\n        ConnectionStatus.EPHEMERAL: (\n            exit_with_success,\n            (\n                f\"No Prefect server specified using profile {name!r} - the API will run\"\n                \" in ephemeral mode.\"\n            ),\n        ),\n        ConnectionStatus.INVALID_API: (\n            exit_with_error,\n            \"Error connecting to Prefect API URL\",\n        ),\n    }\n\n    profiles = prefect.settings.load_profiles()\n    if name not in profiles.names:\n        exit_with_error(f\"Profile {name!r} not found.\")\n\n    profiles.set_active(name)\n    prefect.settings.save_profiles(profiles)\n\n    with Progress(\n        SpinnerColumn(),\n        TextColumn(\"[progress.description]{task.description}\"),\n        transient=False,\n    ) as progress:\n        progress.add_task(\n            description=\"Checking API connectivity...\",\n            total=None,\n        )\n\n        with use_profile(name, include_current_context=False):\n            connection_status = await check_orion_connection()\n\n        exit_method, msg = status_messages[connection_status]\n\n    exit_method(msg)\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/project/","title":"project","text":"","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project","title":"prefect.cli.project","text":"

    Deprecated - Command line interface for working with projects.

    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project.clone","title":"clone async","text":"

    Clone an existing project for a given deployment.

    Source code in prefect/cli/project.py
    @project_app.command()\nasync def clone(\n    deployment_name: str = typer.Option(\n        None,\n        \"--deployment\",\n        \"-d\",\n        help=\"The name of the deployment to clone a project for.\",\n    ),\n    deployment_id: str = typer.Option(\n        None,\n        \"--id\",\n        \"-i\",\n        help=\"The id of the deployment to clone a project for.\",\n    ),\n):\n    \"\"\"\n    Clone an existing project for a given deployment.\n    \"\"\"\n    app.console.print(\n        generate_deprecation_message(\n            \"The `prefect project clone` command\",\n            start_date=\"Jun 2023\",\n        )\n    )\n    if deployment_name and deployment_id:\n        exit_with_error(\n            \"Can only pass one of deployment name or deployment ID options.\"\n        )\n\n    if not deployment_name and not deployment_id:\n        exit_with_error(\"Must pass either a deployment name or deployment ID.\")\n\n    if deployment_name:\n        async with get_client() as client:\n            try:\n                deployment = await client.read_deployment_by_name(deployment_name)\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {deployment_name!r} not found!\")\n    else:\n        async with get_client() as client:\n            try:\n                deployment = await client.read_deployment(deployment_id)\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {deployment_id!r} not found!\")\n\n    if deployment.pull_steps:\n        output = await run_steps(deployment.pull_steps)\n        app.console.out(output[\"directory\"])\n    else:\n        exit_with_error(\"No pull steps found, exiting early.\")\n
    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project.init","title":"init async","text":"

    Initialize a new project.

    Source code in prefect/cli/project.py
    @project_app.command()\n@app.command()\nasync def init(\n    name: str = None,\n    recipe: str = None,\n    fields: List[str] = typer.Option(\n        None,\n        \"-f\",\n        \"--field\",\n        help=(\n            \"One or more fields to pass to the recipe (e.g., image_name) in the format\"\n            \" of key=value.\"\n        ),\n    ),\n):\n    \"\"\"\n    Initialize a new project.\n    \"\"\"\n    inputs = {}\n    fields = fields or []\n    recipe_paths = prefect.__module_path__ / \"deployments\" / \"recipes\"\n\n    for field in fields:\n        key, value = field.split(\"=\")\n        inputs[key] = value\n\n    if not recipe and is_interactive():\n        recipe_paths = prefect.__module_path__ / \"deployments\" / \"recipes\"\n        recipes = []\n\n        for r in recipe_paths.iterdir():\n            if r.is_dir() and (r / \"prefect.yaml\").exists():\n                with open(r / \"prefect.yaml\") as f:\n                    recipe_data = yaml.safe_load(f)\n                    recipe_name = r.name\n                    recipe_description = recipe_data.get(\n                        \"description\", \"(no description available)\"\n                    )\n                    recipe_dict = {\n                        \"name\": recipe_name,\n                        \"description\": recipe_description,\n                    }\n                    recipes.append(recipe_dict)\n\n        selected_recipe = prompt_select_from_table(\n            app.console,\n            \"Would you like to initialize your deployment configuration with a recipe?\",\n            columns=[\n                {\"header\": \"Name\", \"key\": \"name\"},\n                {\"header\": \"Description\", \"key\": \"description\"},\n            ],\n            data=recipes,\n            opt_out_message=\"No, I'll use the default deployment configuration.\",\n            opt_out_response={},\n        )\n        if selected_recipe != {}:\n            recipe = selected_recipe[\"name\"]\n\n    if recipe and (recipe_paths / recipe / \"prefect.yaml\").exists():\n        with open(recipe_paths / recipe / \"prefect.yaml\") as f:\n            recipe_inputs = yaml.safe_load(f).get(\"required_inputs\") or {}\n\n        if recipe_inputs:\n            if set(recipe_inputs.keys()) < set(inputs.keys()):\n                # message to user about extra fields\n                app.console.print(\n                    (\n                        f\"Warning: extra fields provided for {recipe!r} recipe:\"\n                        f\" '{', '.join(set(inputs.keys()) - set(recipe_inputs.keys()))}'\"\n                    ),\n                    style=\"red\",\n                )\n            elif set(recipe_inputs.keys()) > set(inputs.keys()):\n                table = Table(\n                    title=f\"[red]Required inputs for {recipe!r} recipe[/red]\",\n                )\n                table.add_column(\"Field Name\", style=\"green\", no_wrap=True)\n                table.add_column(\n                    \"Description\", justify=\"left\", style=\"white\", no_wrap=False\n                )\n                for field, description in recipe_inputs.items():\n                    if field not in inputs:\n                        table.add_row(field, description)\n\n                app.console.print(table)\n\n                for key, description in recipe_inputs.items():\n                    if key not in inputs:\n                        inputs[key] = typer.prompt(key)\n\n            app.console.print(\"-\" * 15)\n\n    try:\n        files = [\n            f\"[green]{fname}[/green]\"\n            for fname in initialize_project(name=name, recipe=recipe, inputs=inputs)\n        ]\n    except ValueError as exc:\n        if \"Unknown recipe\" in str(exc):\n            exit_with_error(\n                f\"Unknown recipe {recipe!r} provided - run [yellow]`prefect init\"\n                \"`[/yellow] to see all available recipes.\"\n            )\n        else:\n            raise\n\n    files = \"\\n\".join(files)\n    empty_msg = (\n        f\"Created project in [green]{Path('.').resolve()}[/green]; no new files\"\n        \" created.\"\n    )\n    file_msg = (\n        f\"Created project in [green]{Path('.').resolve()}[/green] with the following\"\n        f\" new files:\\n{files}\"\n    )\n    app.console.print(file_msg if files else empty_msg)\n
    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project.ls","title":"ls async","text":"

    List available recipes.

    Source code in prefect/cli/project.py
    @recipe_app.command()\nasync def ls():\n    \"\"\"\n    List available recipes.\n    \"\"\"\n\n    recipe_paths = prefect.__module_path__ / \"deployments\" / \"recipes\"\n    recipes = {}\n\n    for recipe in recipe_paths.iterdir():\n        if recipe.is_dir() and (recipe / \"prefect.yaml\").exists():\n            with open(recipe / \"prefect.yaml\") as f:\n                recipes[recipe.name] = yaml.safe_load(f).get(\n                    \"description\", \"(no description available)\"\n                )\n\n    table = Table(\n        title=\"Available project recipes\",\n        caption=(\n            \"Run `prefect project init --recipe <recipe>` to initialize a project with\"\n            \" a recipe.\"\n        ),\n        caption_style=\"red\",\n    )\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Description\", justify=\"left\", style=\"white\", no_wrap=False)\n    for name, description in sorted(recipes.items(), key=lambda x: x[0]):\n        table.add_row(name, description)\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project.register_flow","title":"register_flow async","text":"

    Register a flow with this project.

    Source code in prefect/cli/project.py
    @project_app.command()\nasync def register_flow(\n    entrypoint: str = typer.Argument(\n        ...,\n        help=(\n            \"The path to a flow entrypoint, in the form of\"\n            \" `./path/to/file.py:flow_func_name`\"\n        ),\n    ),\n    force: bool = typer.Option(\n        False,\n        \"--force\",\n        \"-f\",\n        help=(\n            \"An optional flag to force register this flow and overwrite any existing\"\n            \" entry\"\n        ),\n    ),\n):\n    \"\"\"\n    Register a flow with this project.\n    \"\"\"\n    try:\n        flow = await register(entrypoint, force=force)\n    except Exception as exc:\n        exit_with_error(exc)\n\n    app.console.print(\n        (\n            f\"Registered flow {flow.name!r} in\"\n            f\" {(find_prefect_directory()/'flows.json').resolve()!s}\"\n        ),\n        style=\"green\",\n    )\n
    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/root/","title":"root","text":"","tags":["Python API","CLI"]},{"location":"api-ref/prefect/cli/root/#prefect.cli.root","title":"prefect.cli.root","text":"

    Base prefect command-line application

    ","tags":["Python API","CLI"]},{"location":"api-ref/prefect/cli/root/#prefect.cli.root.version","title":"version async","text":"

    Get the current Prefect version.

    Source code in prefect/cli/root.py
    @app.command()\nasync def version():\n    \"\"\"Get the current Prefect version.\"\"\"\n    import sqlite3\n\n    from prefect.server.utilities.database import get_dialect\n    from prefect.settings import PREFECT_API_DATABASE_CONNECTION_URL\n\n    version_info = {\n        \"Version\": prefect.__version__,\n        \"API version\": SERVER_API_VERSION,\n        \"Python version\": platform.python_version(),\n        \"Git commit\": prefect.__version_info__[\"full-revisionid\"][:8],\n        \"Built\": pendulum.parse(\n            prefect.__version_info__[\"date\"]\n        ).to_day_datetime_string(),\n        \"OS/Arch\": f\"{sys.platform}/{platform.machine()}\",\n        \"Profile\": prefect.context.get_settings_context().profile.name,\n    }\n\n    server_type: str\n\n    try:\n        # We do not context manage the client because when using an ephemeral app we do not\n        # want to create the database or run migrations\n        client = prefect.get_client()\n        server_type = client.server_type.value\n    except Exception:\n        server_type = \"<client error>\"\n\n    version_info[\"Server type\"] = server_type.lower()\n\n    # TODO: Consider adding an API route to retrieve this information?\n    if server_type == ServerType.EPHEMERAL.value:\n        database = get_dialect(PREFECT_API_DATABASE_CONNECTION_URL.value()).name\n        version_info[\"Server\"] = {\"Database\": database}\n        if database == \"sqlite\":\n            version_info[\"Server\"][\"SQLite version\"] = sqlite3.sqlite_version\n\n    def display(object: dict, nesting: int = 0):\n        # Recursive display of a dictionary with nesting\n        for key, value in object.items():\n            key += \":\"\n            if isinstance(value, dict):\n                app.console.print(key)\n                return display(value, nesting + 2)\n            prefix = \" \" * nesting\n            app.console.print(f\"{prefix}{key.ljust(20 - len(prefix))} {value}\")\n\n    display(version_info)\n
    ","tags":["Python API","CLI"]},{"location":"api-ref/prefect/cli/server/","title":"server","text":"","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server","title":"prefect.cli.server","text":"

    Command line interface for working with Prefect

    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.downgrade","title":"downgrade async","text":"

    Downgrade the Prefect database

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def downgrade(\n    yes: bool = typer.Option(False, \"--yes\", \"-y\"),\n    revision: str = typer.Option(\n        \"base\",\n        \"-r\",\n        help=(\n            \"The revision to pass to `alembic downgrade`. If not provided, runs all\"\n            \" migrations.\"\n        ),\n    ),\n    dry_run: bool = typer.Option(\n        False,\n        help=(\n            \"Flag to show what migrations would be made without applying them. Will\"\n            \" emit sql statements to stdout.\"\n        ),\n    ),\n):\n    \"\"\"Downgrade the Prefect database\"\"\"\n    from prefect.server.database.alembic_commands import alembic_downgrade\n    from prefect.server.database.dependencies import provide_database_interface\n\n    db = provide_database_interface()\n\n    engine = await db.engine()\n\n    if not yes:\n        confirm = typer.confirm(\n            \"Are you sure you want to downgrade the Prefect \"\n            f\"database at {engine.url!r}?\"\n        )\n        if not confirm:\n            exit_with_error(\"Database downgrade aborted!\")\n\n    app.console.print(\"Running downgrade migrations ...\")\n    await run_sync_in_worker_thread(\n        alembic_downgrade, revision=revision, dry_run=dry_run\n    )\n    app.console.print(\"Migrations succeeded!\")\n    exit_with_success(f\"Prefect database at {engine.url!r} downgraded!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.reset","title":"reset async","text":"

    Drop and recreate all Prefect database tables

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def reset(yes: bool = typer.Option(False, \"--yes\", \"-y\")):\n    \"\"\"Drop and recreate all Prefect database tables\"\"\"\n    from prefect.server.database.dependencies import provide_database_interface\n\n    db = provide_database_interface()\n    engine = await db.engine()\n    if not yes:\n        confirm = typer.confirm(\n            \"Are you sure you want to reset the Prefect database located \"\n            f'at \"{engine.url!r}\"? This will drop and recreate all tables.'\n        )\n        if not confirm:\n            exit_with_error(\"Database reset aborted\")\n    app.console.print(\"Downgrading database...\")\n    await db.drop_db()\n    app.console.print(\"Upgrading database...\")\n    await db.create_db()\n    exit_with_success(f'Prefect database \"{engine.url!r}\" reset!')\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.revision","title":"revision async","text":"

    Create a new migration for the Prefect database

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def revision(\n    message: str = typer.Option(\n        None,\n        \"--message\",\n        \"-m\",\n        help=\"A message to describe the migration.\",\n    ),\n    autogenerate: bool = False,\n):\n    \"\"\"Create a new migration for the Prefect database\"\"\"\n    from prefect.server.database.alembic_commands import alembic_revision\n\n    app.console.print(\"Running migration file creation ...\")\n    await run_sync_in_worker_thread(\n        alembic_revision,\n        message=message,\n        autogenerate=autogenerate,\n    )\n    exit_with_success(\"Creating new migration file succeeded!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.stamp","title":"stamp async","text":"

    Stamp the revision table with the given revision; don't run any migrations

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def stamp(revision: str):\n    \"\"\"Stamp the revision table with the given revision; don't run any migrations\"\"\"\n    from prefect.server.database.alembic_commands import alembic_stamp\n\n    app.console.print(\"Stamping database with revision ...\")\n    await run_sync_in_worker_thread(alembic_stamp, revision=revision)\n    exit_with_success(\"Stamping database with revision succeeded!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.start","title":"start async","text":"

    Start a Prefect server

    Source code in prefect/cli/server.py
    @server_app.command()\nasync def start(\n    host: str = SettingsOption(PREFECT_SERVER_API_HOST),\n    port: int = SettingsOption(PREFECT_SERVER_API_PORT),\n    keep_alive_timeout: int = SettingsOption(PREFECT_SERVER_API_KEEPALIVE_TIMEOUT),\n    log_level: str = SettingsOption(PREFECT_LOGGING_SERVER_LEVEL),\n    scheduler: bool = SettingsOption(PREFECT_API_SERVICES_SCHEDULER_ENABLED),\n    analytics: bool = SettingsOption(\n        PREFECT_SERVER_ANALYTICS_ENABLED, \"--analytics-on/--analytics-off\"\n    ),\n    late_runs: bool = SettingsOption(PREFECT_API_SERVICES_LATE_RUNS_ENABLED),\n    ui: bool = SettingsOption(PREFECT_UI_ENABLED),\n):\n    \"\"\"Start a Prefect server\"\"\"\n\n    server_env = os.environ.copy()\n    server_env[\"PREFECT_API_SERVICES_SCHEDULER_ENABLED\"] = str(scheduler)\n    server_env[\"PREFECT_SERVER_ANALYTICS_ENABLED\"] = str(analytics)\n    server_env[\"PREFECT_API_SERVICES_LATE_RUNS_ENABLED\"] = str(late_runs)\n    server_env[\"PREFECT_API_SERVICES_UI\"] = str(ui)\n    server_env[\"PREFECT_LOGGING_SERVER_LEVEL\"] = log_level\n\n    base_url = f\"http://{host}:{port}\"\n\n    async with anyio.create_task_group() as tg:\n        app.console.print(generate_welcome_blurb(base_url, ui_enabled=ui))\n        app.console.print(\"\\n\")\n\n        server_process_id = await tg.start(\n            partial(\n                run_process,\n                command=[\n                    get_sys_executable(),\n                    \"-m\",\n                    \"uvicorn\",\n                    \"--app-dir\",\n                    # quote wrapping needed for windows paths with spaces\n                    f'\"{prefect.__module_path__.parent}\"',\n                    \"--factory\",\n                    \"prefect.server.api.server:create_app\",\n                    \"--host\",\n                    str(host),\n                    \"--port\",\n                    str(port),\n                    \"--timeout-keep-alive\",\n                    str(keep_alive_timeout),\n                ],\n                env=server_env,\n                stream_output=True,\n            )\n        )\n\n        # Explicitly handle the interrupt signal here, as it will allow us to\n        # cleanly stop the uvicorn server. Failing to do that may cause a\n        # large amount of anyio error traces on the terminal, because the\n        # SIGINT is handled by Typer/Click in this process (the parent process)\n        # and will start shutting down subprocesses:\n        # https://github.com/PrefectHQ/server/issues/2475\n\n        setup_signal_handlers_server(\n            server_process_id, \"the Prefect server\", app.console.print\n        )\n\n    app.console.print(\"Server stopped!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.upgrade","title":"upgrade async","text":"

    Upgrade the Prefect database

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def upgrade(\n    yes: bool = typer.Option(False, \"--yes\", \"-y\"),\n    revision: str = typer.Option(\n        \"head\",\n        \"-r\",\n        help=(\n            \"The revision to pass to `alembic upgrade`. If not provided, runs all\"\n            \" migrations.\"\n        ),\n    ),\n    dry_run: bool = typer.Option(\n        False,\n        help=(\n            \"Flag to show what migrations would be made without applying them. Will\"\n            \" emit sql statements to stdout.\"\n        ),\n    ),\n):\n    \"\"\"Upgrade the Prefect database\"\"\"\n    from prefect.server.database.alembic_commands import alembic_upgrade\n    from prefect.server.database.dependencies import provide_database_interface\n\n    db = provide_database_interface()\n    engine = await db.engine()\n\n    if not yes:\n        confirm = typer.confirm(\n            f\"Are you sure you want to upgrade the Prefect database at {engine.url!r}?\"\n        )\n        if not confirm:\n            exit_with_error(\"Database upgrade aborted!\")\n\n    app.console.print(\"Running upgrade migrations ...\")\n    await run_sync_in_worker_thread(alembic_upgrade, revision=revision, dry_run=dry_run)\n    app.console.print(\"Migrations succeeded!\")\n    exit_with_success(f\"Prefect database at {engine.url!r} upgraded!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/variable/","title":"variable","text":"","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/variable/#prefect.cli.variable","title":"prefect.cli.variable","text":"","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/variable/#prefect.cli.variable.delete","title":"delete async","text":"

    Delete a variable.

    Parameters:

    Name Type Description Default name str

    the name of the variable to delete

    required Source code in prefect/cli/variable.py
    @variable_app.command(\"delete\")\nasync def delete(\n    name: str,\n):\n    \"\"\"\n    Delete a variable.\n\n    Arguments:\n        name: the name of the variable to delete\n    \"\"\"\n\n    async with get_client() as client:\n        try:\n            await client.delete_variable_by_name(\n                name=name,\n            )\n        except ObjectNotFound:\n            exit_with_error(f\"Variable {name!r} not found.\")\n\n        exit_with_success(f\"Deleted variable {name!r}.\")\n
    ","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/variable/#prefect.cli.variable.inspect","title":"inspect async","text":"

    View details about a variable.

    Parameters:

    Name Type Description Default name str

    the name of the variable to inspect

    required Source code in prefect/cli/variable.py
    @variable_app.command(\"inspect\")\nasync def inspect(\n    name: str,\n):\n    \"\"\"\n    View details about a variable.\n\n    Arguments:\n        name: the name of the variable to inspect\n    \"\"\"\n\n    async with get_client() as client:\n        variable = await client.read_variable_by_name(\n            name=name,\n        )\n        if not variable:\n            exit_with_error(f\"Variable {name!r} not found.\")\n\n        app.console.print(Pretty(variable))\n
    ","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/variable/#prefect.cli.variable.list_variables","title":"list_variables async","text":"

    List variables.

    Source code in prefect/cli/variable.py
    @variable_app.command(\"ls\")\nasync def list_variables(\n    limit: int = typer.Option(\n        100,\n        \"--limit\",\n        help=\"The maximum number of variables to return.\",\n    ),\n):\n    \"\"\"\n    List variables.\n    \"\"\"\n    async with get_client() as client:\n        variables = await client.read_variables(\n            limit=limit,\n        )\n\n        table = Table(\n            title=\"Variables\",\n            caption=\"List Variables using `prefect variable ls`\",\n            show_header=True,\n        )\n\n        table.add_column(\"Name\", style=\"blue\", no_wrap=True)\n        # values can be up 5000 characters so truncate early\n        table.add_column(\"Value\", style=\"blue\", no_wrap=True, max_width=50)\n        table.add_column(\"Created\", style=\"blue\", no_wrap=True)\n        table.add_column(\"Updated\", style=\"blue\", no_wrap=True)\n\n        for variable in sorted(variables, key=lambda x: f\"{x.name}\"):\n            table.add_row(\n                variable.name,\n                variable.value,\n                pendulum.instance(variable.created).diff_for_humans(),\n                pendulum.instance(variable.updated).diff_for_humans(),\n            )\n\n        app.console.print(table)\n
    ","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/work_pool/","title":"work_pool","text":"","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool","title":"prefect.cli.work_pool","text":"

    Command line interface for working with work queues.

    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.clear_concurrency_limit","title":"clear_concurrency_limit async","text":"

    Clear the concurrency limit for a work pool.

    \b Examples: $ prefect work-pool clear-concurrency-limit \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def clear_concurrency_limit(\n    name: str = typer.Argument(..., help=\"The name of the work pool to update.\"),\n):\n    \"\"\"\n    Clear the concurrency limit for a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool clear-concurrency-limit \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    concurrency_limit=None,\n                ),\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(f\"Cleared concurrency limit for work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.create","title":"create async","text":"

    Create a new work pool.

    \b Examples: \b Create a Kubernetes work pool in a paused state: \b $ prefect work-pool create \"my-pool\" --type kubernetes --paused \b Create a Docker work pool with a custom base job template: \b $ prefect work-pool create \"my-pool\" --type docker --base-job-template ./base-job-template.json

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def create(\n    name: str = typer.Argument(..., help=\"The name of the work pool.\"),\n    base_job_template: typer.FileText = typer.Option(\n        None,\n        \"--base-job-template\",\n        help=(\n            \"The path to a JSON file containing the base job template to use. If\"\n            \" unspecified, Prefect will use the default base job template for the given\"\n            \" worker type.\"\n        ),\n    ),\n    paused: bool = typer.Option(\n        False,\n        \"--paused\",\n        help=\"Whether or not to create the work pool in a paused state.\",\n    ),\n    type: str = typer.Option(\n        None, \"-t\", \"--type\", help=\"The type of work pool to create.\"\n    ),\n    set_as_default: bool = typer.Option(\n        False,\n        \"--set-as-default\",\n        help=(\n            \"Whether or not to use the created work pool as the local default for\"\n            \" deployment.\"\n        ),\n    ),\n    provision_infrastructure: bool = typer.Option(\n        False,\n        \"--provision-infrastructure\",\n        \"--provision-infra\",\n        help=(\n            \"Whether or not to provision infrastructure for the work pool if supported\"\n            \" for the given work pool type.\"\n        ),\n    ),\n):\n    \"\"\"\n    Create a new work pool.\n\n    \\b\n    Examples:\n        \\b\n        Create a Kubernetes work pool in a paused state:\n            \\b\n            $ prefect work-pool create \"my-pool\" --type kubernetes --paused\n        \\b\n        Create a Docker work pool with a custom base job template:\n            \\b\n            $ prefect work-pool create \"my-pool\" --type docker --base-job-template ./base-job-template.json\n\n    \"\"\"\n    if not name.lower().strip(\"'\\\" \"):\n        exit_with_error(\"Work pool name cannot be empty.\")\n    async with get_client() as client:\n        try:\n            await client.read_work_pool(work_pool_name=name)\n        except ObjectNotFound:\n            pass\n        else:\n            exit_with_error(\n                f\"Work pool named {name!r} already exists. Please try creating your\"\n                \" work pool again with a different name.\"\n            )\n\n        if type is None:\n            async with get_collections_metadata_client() as collections_client:\n                if not is_interactive():\n                    exit_with_error(\n                        \"When not using an interactive terminal, you must supply a\"\n                        \" `--type` value.\"\n                    )\n                worker_metadata = await collections_client.read_worker_metadata()\n\n                # Retrieve only push pools if provisioning infrastructure\n                data = [\n                    worker\n                    for collection in worker_metadata.values()\n                    for worker in collection.values()\n                    if provision_infrastructure\n                    and has_provisioner_for_type(worker[\"type\"])\n                    or not provision_infrastructure\n                ]\n                worker = prompt_select_from_table(\n                    app.console,\n                    \"What type of work pool infrastructure would you like to use?\",\n                    columns=[\n                        {\"header\": \"Infrastructure Type\", \"key\": \"display_name\"},\n                        {\"header\": \"Description\", \"key\": \"description\"},\n                    ],\n                    data=data,\n                    table_kwargs={\"show_lines\": True},\n                )\n                type = worker[\"type\"]\n\n        available_work_pool_types = await get_available_work_pool_types()\n        if type not in available_work_pool_types:\n            exit_with_error(\n                f\"Unknown work pool type {type!r}. \"\n                \"Please choose from\"\n                f\" {', '.join(available_work_pool_types)}.\"\n            )\n\n        if base_job_template is None:\n            template_contents = (\n                await get_default_base_job_template_for_infrastructure_type(type)\n            )\n        else:\n            template_contents = json.load(base_job_template)\n\n        if provision_infrastructure:\n            try:\n                provisioner = get_infrastructure_provisioner_for_work_pool_type(type)\n                provisioner.console = app.console\n                template_contents = await provisioner.provision(\n                    work_pool_name=name, base_job_template=template_contents\n                )\n            except ValueError as exc:\n                print(exc)\n                app.console.print(\n                    (\n                        \"Automatic infrastructure provisioning is not supported for\"\n                        f\" {type!r} work pools.\"\n                    ),\n                    style=\"yellow\",\n                )\n            except RuntimeError as exc:\n                exit_with_error(f\"Failed to provision infrastructure: {exc}\")\n\n        try:\n            wp = WorkPoolCreate(\n                name=name,\n                type=type,\n                base_job_template=template_contents,\n                is_paused=paused,\n            )\n            work_pool = await client.create_work_pool(work_pool=wp)\n            app.console.print(f\"Created work pool {work_pool.name!r}!\\n\", style=\"green\")\n            if (\n                not work_pool.is_paused\n                and not work_pool.is_managed_pool\n                and not work_pool.is_push_pool\n            ):\n                app.console.print(\"To start a worker for this work pool, run:\\n\")\n                app.console.print(\n                    f\"\\t[blue]prefect worker start --pool {work_pool.name}[/]\\n\"\n                )\n            if set_as_default:\n                set_work_pool_as_default(work_pool.name)\n            exit_with_success(\"\")\n        except ObjectAlreadyExists:\n            exit_with_error(\n                f\"Work pool named {name!r} already exists. Please try creating your\"\n                \" work pool again with a different name.\"\n            )\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.delete","title":"delete async","text":"

    Delete a work pool.

    \b Examples: $ prefect work-pool delete \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def delete(\n    name: str = typer.Argument(..., help=\"The name of the work pool to delete.\"),\n):\n    \"\"\"\n    Delete a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool delete \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.delete_work_pool(work_pool_name=name)\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(f\"Deleted work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.get_default_base_job_template","title":"get_default_base_job_template async","text":"

    Get the default base job template for a given work pool type.

    \b Examples: $ prefect work-pool get-default-base-job-template --type kubernetes

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def get_default_base_job_template(\n    type: str = typer.Option(\n        None,\n        \"-t\",\n        \"--type\",\n        help=\"The type of work pool for which to get the default base job template.\",\n    ),\n    file: str = typer.Option(\n        None, \"-f\", \"--file\", help=\"If set, write the output to a file.\"\n    ),\n):\n    \"\"\"\n    Get the default base job template for a given work pool type.\n\n    \\b\n    Examples:\n        $ prefect work-pool get-default-base-job-template --type kubernetes\n    \"\"\"\n    base_job_template = await get_default_base_job_template_for_infrastructure_type(\n        type\n    )\n    if base_job_template is None:\n        exit_with_error(\n            f\"Unknown work pool type {type!r}. \"\n            \"Please choose from\"\n            f\" {', '.join(await get_available_work_pool_types())}.\"\n        )\n\n    if file is None:\n        print(json.dumps(base_job_template, indent=2))\n    else:\n        with open(file, mode=\"w\") as f:\n            json.dump(base_job_template, fp=f, indent=2)\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.has_provisioner_for_type","title":"has_provisioner_for_type","text":"

    Check if there is a provisioner for the given work pool type.

    Parameters:

    Name Type Description Default work_pool_type str

    The type of the work pool.

    required

    Returns:

    Name Type Description bool bool

    True if a provisioner exists for the given type, False otherwise.

    Source code in prefect/cli/work_pool.py
    def has_provisioner_for_type(work_pool_type: str) -> bool:\n    \"\"\"\n    Check if there is a provisioner for the given work pool type.\n\n    Args:\n        work_pool_type (str): The type of the work pool.\n\n    Returns:\n        bool: True if a provisioner exists for the given type, False otherwise.\n    \"\"\"\n    return work_pool_type in _provisioners\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.inspect","title":"inspect async","text":"

    Inspect a work pool.

    \b Examples: $ prefect work-pool inspect \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def inspect(\n    name: str = typer.Argument(..., help=\"The name of the work pool to inspect.\"),\n):\n    \"\"\"\n    Inspect a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool inspect \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            pool = await client.read_work_pool(work_pool_name=name)\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        app.console.print(Pretty(pool))\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.ls","title":"ls async","text":"

    List work pools.

    \b Examples: $ prefect work-pool ls

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def ls(\n    verbose: bool = typer.Option(\n        False,\n        \"--verbose\",\n        \"-v\",\n        help=\"Show additional information about work pools.\",\n    ),\n):\n    \"\"\"\n    List work pools.\n\n    \\b\n    Examples:\n        $ prefect work-pool ls\n    \"\"\"\n    table = Table(\n        title=\"Work Pools\", caption=\"(**) denotes a paused pool\", caption_style=\"red\"\n    )\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Type\", style=\"magenta\", no_wrap=True)\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n    if verbose:\n        table.add_column(\"Base Job Template\", style=\"magenta\", no_wrap=True)\n\n    async with get_client() as client:\n        pools = await client.read_work_pools()\n\n    def sort_by_created_key(q):\n        return pendulum.now(\"utc\") - q.created\n\n    for pool in sorted(pools, key=sort_by_created_key):\n        row = [\n            f\"{pool.name} [red](**)\" if pool.is_paused else pool.name,\n            str(pool.type),\n            str(pool.id),\n            (\n                f\"[red]{pool.concurrency_limit}\"\n                if pool.concurrency_limit\n                else \"[blue]None\"\n            ),\n        ]\n        if verbose:\n            row.append(str(pool.base_job_template))\n        table.add_row(*row)\n\n    app.console.print(table)\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.pause","title":"pause async","text":"

    Pause a work pool.

    \b Examples: $ prefect work-pool pause \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def pause(\n    name: str = typer.Argument(..., help=\"The name of the work pool to pause.\"),\n):\n    \"\"\"\n    Pause a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool pause \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    is_paused=True,\n                ),\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(f\"Paused work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.preview","title":"preview async","text":"

    Preview the work pool's scheduled work for all queues.

    \b Examples: $ prefect work-pool preview \"my-pool\" --hours 24

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def preview(\n    name: str = typer.Argument(None, help=\"The name or ID of the work pool to preview\"),\n    hours: int = typer.Option(\n        None,\n        \"-h\",\n        \"--hours\",\n        help=\"The number of hours to look ahead; defaults to 1 hour\",\n    ),\n):\n    \"\"\"\n    Preview the work pool's scheduled work for all queues.\n\n    \\b\n    Examples:\n        $ prefect work-pool preview \"my-pool\" --hours 24\n\n    \"\"\"\n    if hours is None:\n        hours = 1\n\n    async with get_client() as client:\n        try:\n            responses = await client.get_scheduled_flow_runs_for_work_pool(\n                work_pool_name=name,\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n    runs = [response.flow_run for response in responses]\n    table = Table(caption=\"(**) denotes a late run\", caption_style=\"red\")\n\n    table.add_column(\n        \"Scheduled Start Time\", justify=\"left\", style=\"yellow\", no_wrap=True\n    )\n    table.add_column(\"Run ID\", justify=\"left\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Deployment ID\", style=\"blue\", no_wrap=True)\n\n    pendulum.now(\"utc\").add(hours=hours or 1)\n\n    now = pendulum.now(\"utc\")\n\n    def sort_by_created_key(r):\n        return now - r.created\n\n    for run in sorted(runs, key=sort_by_created_key):\n        table.add_row(\n            (\n                f\"{run.expected_start_time} [red](**)\"\n                if run.expected_start_time < now\n                else f\"{run.expected_start_time}\"\n            ),\n            str(run.id),\n            run.name,\n            str(run.deployment_id),\n        )\n\n    if runs:\n        app.console.print(table)\n    else:\n        app.console.print(\n            (\n                \"No runs found - try increasing how far into the future you preview\"\n                \" with the --hours flag\"\n            ),\n            style=\"yellow\",\n        )\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.provision_infrastructure","title":"provision_infrastructure async","text":"

    Provision infrastructure for a work pool.

    \b Examples: $ prefect work-pool provision-infrastructure \"my-pool\"

    $ prefect work-pool provision-infra \"my-pool\"\n
    Source code in prefect/cli/work_pool.py
    @work_pool_app.command(aliases=[\"provision-infra\"])\nasync def provision_infrastructure(\n    name: str = typer.Argument(\n        ..., help=\"The name of the work pool to provision infrastructure for.\"\n    ),\n):\n    \"\"\"\n    Provision infrastructure for a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool provision-infrastructure \"my-pool\"\n\n        $ prefect work-pool provision-infra \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            work_pool = await client.read_work_pool(work_pool_name=name)\n            if not work_pool.is_push_pool:\n                exit_with_error(\n                    f\"Work pool {name!r} is not a push pool type. \"\n                    \"Please try provisioning infrastructure for a push pool.\"\n                )\n        except ObjectNotFound:\n            exit_with_error(f\"Work pool {name!r} does not exist.\")\n        except Exception as exc:\n            exit_with_error(f\"Failed to read work pool {name!r}: {exc}\")\n\n        try:\n            provisioner = get_infrastructure_provisioner_for_work_pool_type(\n                work_pool.type\n            )\n            provisioner.console = app.console\n            new_base_job_template = await provisioner.provision(\n                work_pool_name=name, base_job_template=work_pool.base_job_template\n            )\n\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    base_job_template=new_base_job_template,\n                ),\n            )\n\n        except ValueError as exc:\n            app.console.print(f\"Error: {exc}\")\n            app.console.print(\n                (\n                    \"Automatic infrastructure provisioning is not supported for\"\n                    f\" {work_pool.type!r} work pools.\"\n                ),\n                style=\"yellow\",\n            )\n        except RuntimeError as exc:\n            exit_with_error(\n                f\"Failed to provision infrastructure for '{name}' work pool: {exc}\"\n            )\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.resume","title":"resume async","text":"

    Resume a work pool.

    \b Examples: $ prefect work-pool resume \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def resume(\n    name: str = typer.Argument(..., help=\"The name of the work pool to resume.\"),\n):\n    \"\"\"\n    Resume a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool resume \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    is_paused=False,\n                ),\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(f\"Resumed work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.set_concurrency_limit","title":"set_concurrency_limit async","text":"

    Set the concurrency limit for a work pool.

    \b Examples: $ prefect work-pool set-concurrency-limit \"my-pool\" 10

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def set_concurrency_limit(\n    name: str = typer.Argument(..., help=\"The name of the work pool to update.\"),\n    concurrency_limit: int = typer.Argument(\n        ..., help=\"The new concurrency limit for the work pool.\"\n    ),\n):\n    \"\"\"\n    Set the concurrency limit for a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool set-concurrency-limit \"my-pool\" 10\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    concurrency_limit=concurrency_limit,\n                ),\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(\n            f\"Set concurrency limit for work pool {name!r} to {concurrency_limit}\"\n        )\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.update","title":"update async","text":"

    Update a work pool.

    \b Examples: $ prefect work-pool update \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def update(\n    name: str = typer.Argument(..., help=\"The name of the work pool to update.\"),\n    base_job_template: typer.FileText = typer.Option(\n        None,\n        \"--base-job-template\",\n        help=(\n            \"The path to a JSON file containing the base job template to use. If\"\n            \" unspecified, Prefect will use the default base job template for the given\"\n            \" worker type. If None, the base job template will not be modified.\"\n        ),\n    ),\n    concurrency_limit: int = typer.Option(\n        None,\n        \"--concurrency-limit\",\n        help=(\n            \"The concurrency limit for the work pool. If None, the concurrency limit\"\n            \" will not be modified.\"\n        ),\n    ),\n    description: str = typer.Option(\n        None,\n        \"--description\",\n        help=(\n            \"The description for the work pool. If None, the description will not be\"\n            \" modified.\"\n        ),\n    ),\n):\n    \"\"\"\n    Update a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool update \"my-pool\"\n\n    \"\"\"\n    wp = WorkPoolUpdate()\n    if base_job_template:\n        wp.base_job_template = json.load(base_job_template)\n    if concurrency_limit:\n        wp.concurrency_limit = concurrency_limit\n    if description:\n        wp.description = description\n\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=wp,\n            )\n        except ObjectNotFound:\n            exit_with_error(\"Work pool named {name!r} does not exist.\")\n\n        exit_with_success(f\"Updated work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_queue/","title":"work_queue","text":"","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue","title":"prefect.cli.work_queue","text":"

    Command line interface for working with work queues.

    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.clear_concurrency_limit","title":"clear_concurrency_limit async","text":"

    Clear any concurrency limits from a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def clear_concurrency_limit(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue to clear\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Clear any concurrency limits from a work queue.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n    async with get_client() as client:\n        try:\n            await client.update_work_queue(\n                id=queue_id,\n                concurrency_limit=None,\n            )\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n\n    if pool:\n        success_message = (\n            f\"Concurrency limits removed on work queue {name!r} in work pool {pool!r}\"\n        )\n    else:\n        success_message = f\"Concurrency limits removed on work queue {name!r}\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.create","title":"create async","text":"

    Create a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def create(\n    name: str = typer.Argument(..., help=\"The unique name to assign this work queue\"),\n    limit: int = typer.Option(\n        None, \"-l\", \"--limit\", help=\"The concurrency limit to set on the queue.\"\n    ),\n    tags: List[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=(\n            \"DEPRECATED: One or more optional tags. This option will be removed on\"\n            \" 2023-02-23.\"\n        ),\n    ),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool to create the work queue in.\",\n    ),\n    priority: Optional[int] = typer.Option(\n        None,\n        \"-q\",\n        \"--priority\",\n        help=\"The associated priority for the created work queue\",\n    ),\n):\n    \"\"\"\n    Create a work queue.\n    \"\"\"\n    if tags:\n        app.console.print(\n            (\n                \"Supplying `tags` for work queues is deprecated. This work \"\n                \"queue will use legacy tag-matching behavior. \"\n                \"This option will be removed on 2023-02-23.\"\n            ),\n            style=\"red\",\n        )\n\n    if pool and tags:\n        exit_with_error(\n            \"Work queues created with tags cannot specify work pools or set priorities.\"\n        )\n\n    async with get_client() as client:\n        try:\n            result = await client.create_work_queue(\n                name=name, tags=tags or None, work_pool_name=pool, priority=priority\n            )\n            if limit is not None:\n                await client.update_work_queue(\n                    id=result.id,\n                    concurrency_limit=limit,\n                )\n        except ObjectAlreadyExists:\n            exit_with_error(f\"Work queue with name: {name!r} already exists.\")\n        except ObjectNotFound:\n            exit_with_error(f\"Work pool with name: {pool!r} not found.\")\n\n    if tags:\n        tags_message = f\"tags - {', '.join(sorted(tags))}\\n\" or \"\"\n        output_msg = dedent(\n            f\"\"\"\n            Created work queue with properties:\n                name - {name!r}\n                id - {result.id}\n                concurrency limit - {limit}\n                {tags_message}\n            Start an agent to pick up flow runs from the work queue:\n                prefect agent start -q '{result.name}'\n\n            Inspect the work queue:\n                prefect work-queue inspect '{result.name}'\n            \"\"\"\n        )\n    else:\n        if not pool:\n            # specify the default work pool name after work queue creation to allow the server\n            # to handle a bunch of logic associated with agents without work pools\n            pool = DEFAULT_AGENT_WORK_POOL_NAME\n        output_msg = dedent(\n            f\"\"\"\n            Created work queue with properties:\n                name - {name!r}\n                work pool - {pool!r}\n                id - {result.id}\n                concurrency limit - {limit}\n            Start an agent to pick up flow runs from the work queue:\n                prefect agent start -q '{result.name} -p {pool}'\n\n            Inspect the work queue:\n                prefect work-queue inspect '{result.name}'\n            \"\"\"\n        )\n    exit_with_success(output_msg)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.delete","title":"delete async","text":"

    Delete a work queue by ID.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def delete(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue to delete\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool containing the work queue to delete.\",\n    ),\n):\n    \"\"\"\n    Delete a work queue by ID.\n    \"\"\"\n\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n    async with get_client() as client:\n        try:\n            await client.delete_work_queue_by_id(id=queue_id)\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n    if pool:\n        success_message = (\n            f\"Successfully deleted work queue {name!r} in work pool {pool!r}\"\n        )\n    else:\n        success_message = f\"Successfully deleted work queue {name!r}\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.inspect","title":"inspect async","text":"

    Inspect a work queue by ID.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def inspect(\n    name: str = typer.Argument(\n        None, help=\"The name or ID of the work queue to inspect\"\n    ),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Inspect a work queue by ID.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n    async with get_client() as client:\n        try:\n            result = await client.read_work_queue(id=queue_id)\n            app.console.print(Pretty(result))\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n\n        try:\n            status = await client.read_work_queue_status(id=queue_id)\n            app.console.print(Pretty(status))\n        except ObjectNotFound:\n            pass\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.ls","title":"ls async","text":"

    View all work queues.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def ls(\n    verbose: bool = typer.Option(\n        False, \"--verbose\", \"-v\", help=\"Display more information.\"\n    ),\n    work_queue_prefix: str = typer.Option(\n        None,\n        \"--match\",\n        \"-m\",\n        help=(\n            \"Will match work queues with names that start with the specified prefix\"\n            \" string\"\n        ),\n    ),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool containing the work queues to list.\",\n    ),\n):\n    \"\"\"\n    View all work queues.\n    \"\"\"\n    if not pool and not experiment_enabled(\"work_pools\"):\n        table = Table(\n            title=\"Work Queues\",\n            caption=\"(**) denotes a paused queue\",\n            caption_style=\"red\",\n        )\n        table.add_column(\"Name\", style=\"green\", no_wrap=True)\n        table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n        table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n        if verbose:\n            table.add_column(\"Filter (Deprecated)\", style=\"magenta\", no_wrap=True)\n\n        async with get_client() as client:\n            if work_queue_prefix is not None:\n                queues = await client.match_work_queues([work_queue_prefix])\n            else:\n                queues = await client.read_work_queues()\n\n            def sort_by_created_key(q):\n                return pendulum.now(\"utc\") - q.created\n\n            for queue in sorted(queues, key=sort_by_created_key):\n                row = [\n                    f\"{queue.name} [red](**)\" if queue.is_paused else queue.name,\n                    str(queue.id),\n                    (\n                        f\"[red]{queue.concurrency_limit}\"\n                        if queue.concurrency_limit\n                        else \"[blue]None\"\n                    ),\n                ]\n                if verbose and queue.filter is not None:\n                    row.append(queue.filter.json())\n                table.add_row(*row)\n    elif not pool:\n        table = Table(\n            title=\"Work Queues\",\n            caption=\"(**) denotes a paused queue\",\n            caption_style=\"red\",\n        )\n        table.add_column(\"Name\", style=\"green\", no_wrap=True)\n        table.add_column(\"Pool\", style=\"magenta\", no_wrap=True)\n        table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n        table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n        if verbose:\n            table.add_column(\"Filter (Deprecated)\", style=\"magenta\", no_wrap=True)\n\n        async with get_client() as client:\n            if work_queue_prefix is not None:\n                queues = await client.match_work_queues([work_queue_prefix])\n            else:\n                queues = await client.read_work_queues()\n\n            pool_ids = [q.work_pool_id for q in queues]\n            wp_filter = WorkPoolFilter(id=WorkPoolFilterId(any_=pool_ids))\n            pools = await client.read_work_pools(work_pool_filter=wp_filter)\n            pool_id_name_map = {p.id: p.name for p in pools}\n\n            def sort_by_created_key(q):\n                return pendulum.now(\"utc\") - q.created\n\n            for queue in sorted(queues, key=sort_by_created_key):\n                row = [\n                    f\"{queue.name} [red](**)\" if queue.is_paused else queue.name,\n                    pool_id_name_map[queue.work_pool_id],\n                    str(queue.id),\n                    (\n                        f\"[red]{queue.concurrency_limit}\"\n                        if queue.concurrency_limit\n                        else \"[blue]None\"\n                    ),\n                ]\n                if verbose and queue.filter is not None:\n                    row.append(queue.filter.json())\n                table.add_row(*row)\n\n    else:\n        table = Table(\n            title=f\"Work Queues in Work Pool {pool!r}\",\n            caption=\"(**) denotes a paused queue\",\n            caption_style=\"red\",\n        )\n        table.add_column(\"Name\", style=\"green\", no_wrap=True)\n        table.add_column(\"Priority\", style=\"magenta\", no_wrap=True)\n        table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n        if verbose:\n            table.add_column(\"Description\", style=\"cyan\", no_wrap=False)\n\n        async with get_client() as client:\n            try:\n                queues = await client.read_work_queues(work_pool_name=pool)\n            except ObjectNotFound:\n                exit_with_error(f\"No work pool found: {pool!r}\")\n\n            def sort_by_created_key(q):\n                return pendulum.now(\"utc\") - q.created\n\n            for queue in sorted(queues, key=sort_by_created_key):\n                row = [\n                    f\"{queue.name} [red](**)\" if queue.is_paused else queue.name,\n                    f\"{queue.priority}\",\n                    (\n                        f\"[red]{queue.concurrency_limit}\"\n                        if queue.concurrency_limit\n                        else \"[blue]None\"\n                    ),\n                ]\n                if verbose:\n                    row.append(queue.description)\n                table.add_row(*row)\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.pause","title":"pause async","text":"

    Pause a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def pause(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue to pause\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Pause a work queue.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n\n    async with get_client() as client:\n        try:\n            await client.update_work_queue(\n                id=queue_id,\n                is_paused=True,\n            )\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n\n    if pool:\n        success_message = f\"Work queue {name!r} in work pool {pool!r} paused\"\n    else:\n        success_message = f\"Work queue {name!r} paused\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.preview","title":"preview async","text":"

    Preview a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def preview(\n    name: str = typer.Argument(\n        None, help=\"The name or ID of the work queue to preview\"\n    ),\n    hours: int = typer.Option(\n        None,\n        \"-h\",\n        \"--hours\",\n        help=\"The number of hours to look ahead; defaults to 1 hour\",\n    ),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Preview a work queue.\n    \"\"\"\n    if pool:\n        title = f\"Preview of Work Queue {name!r} in Work Pool {pool!r}\"\n    else:\n        title = f\"Preview of Work Queue {name!r}\"\n\n    table = Table(title=title, caption=\"(**) denotes a late run\", caption_style=\"red\")\n    table.add_column(\n        \"Scheduled Start Time\", justify=\"left\", style=\"yellow\", no_wrap=True\n    )\n    table.add_column(\"Run ID\", justify=\"left\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Deployment ID\", style=\"blue\", no_wrap=True)\n\n    window = pendulum.now(\"utc\").add(hours=hours or 1)\n\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name, work_pool_name=pool\n    )\n    async with get_client() as client:\n        if pool:\n            try:\n                responses = await client.get_scheduled_flow_runs_for_work_pool(\n                    work_pool_name=pool,\n                    work_queue_names=[name],\n                )\n                runs = [response.flow_run for response in responses]\n            except ObjectNotFound:\n                exit_with_error(f\"No work queue found: {name!r} in work pool {pool!r}\")\n        else:\n            try:\n                runs = await client.get_runs_in_work_queue(\n                    queue_id,\n                    limit=10,\n                    scheduled_before=window,\n                )\n            except ObjectNotFound:\n                exit_with_error(f\"No work queue found: {name!r}\")\n    now = pendulum.now(\"utc\")\n\n    def sort_by_created_key(r):\n        return now - r.created\n\n    for run in sorted(runs, key=sort_by_created_key):\n        table.add_row(\n            (\n                f\"{run.expected_start_time} [red](**)\"\n                if run.expected_start_time < now\n                else f\"{run.expected_start_time}\"\n            ),\n            str(run.id),\n            run.name,\n            str(run.deployment_id),\n        )\n\n    if runs:\n        app.console.print(table)\n    else:\n        app.console.print(\n            (\n                \"No runs found - try increasing how far into the future you preview\"\n                \" with the --hours flag\"\n            ),\n            style=\"yellow\",\n        )\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.resume","title":"resume async","text":"

    Resume a paused work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def resume(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue to resume\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Resume a paused work queue.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n\n    async with get_client() as client:\n        try:\n            await client.update_work_queue(\n                id=queue_id,\n                is_paused=False,\n            )\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n\n    if pool:\n        success_message = f\"Work queue {name!r} in work pool {pool!r} resumed\"\n    else:\n        success_message = f\"Work queue {name!r} resumed\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.set_concurrency_limit","title":"set_concurrency_limit async","text":"

    Set a concurrency limit on a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def set_concurrency_limit(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue\"),\n    limit: int = typer.Argument(..., help=\"The concurrency limit to set on the queue.\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Set a concurrency limit on a work queue.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n\n    async with get_client() as client:\n        try:\n            await client.update_work_queue(\n                id=queue_id,\n                concurrency_limit=limit,\n            )\n        except ObjectNotFound:\n            if pool:\n                error_message = (\n                    f\"No work queue named {name!r} found in work pool {pool!r}.\"\n                )\n            else:\n                error_message = f\"No work queue named {name!r} found.\"\n            exit_with_error(error_message)\n\n    if pool:\n        success_message = (\n            f\"Concurrency limit of {limit} set on work queue {name!r} in work pool\"\n            f\" {pool!r}\"\n        )\n    else:\n        success_message = f\"Concurrency limit of {limit} set on work queue {name!r}\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/worker/","title":"worker","text":"","tags":["Python API","workers","CLI"]},{"location":"api-ref/prefect/cli/worker/#prefect.cli.worker","title":"prefect.cli.worker","text":"","tags":["Python API","workers","CLI"]},{"location":"api-ref/prefect/cli/worker/#prefect.cli.worker.start","title":"start async","text":"

    Start a worker process to poll a work pool for flow runs.

    Source code in prefect/cli/worker.py
    @worker_app.command()\nasync def start(\n    worker_name: str = typer.Option(\n        None,\n        \"-n\",\n        \"--name\",\n        help=(\n            \"The name to give to the started worker. If not provided, a unique name\"\n            \" will be generated.\"\n        ),\n    ),\n    work_pool_name: str = typer.Option(\n        ...,\n        \"-p\",\n        \"--pool\",\n        help=\"The work pool the started worker should poll.\",\n        prompt=True,\n    ),\n    work_queues: List[str] = typer.Option(\n        None,\n        \"-q\",\n        \"--work-queue\",\n        help=(\n            \"One or more work queue names for the worker to pull from. If not provided,\"\n            \" the worker will pull from all work queues in the work pool.\"\n        ),\n    ),\n    worker_type: Optional[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--type\",\n        help=(\n            \"The type of worker to start. If not provided, the worker type will be\"\n            \" inferred from the work pool.\"\n        ),\n    ),\n    prefetch_seconds: int = SettingsOption(\n        PREFECT_WORKER_PREFETCH_SECONDS,\n        help=\"Number of seconds to look into the future for scheduled flow runs.\",\n    ),\n    run_once: bool = typer.Option(\n        False, help=\"Only run worker polling once. By default, the worker runs forever.\"\n    ),\n    limit: int = typer.Option(\n        None,\n        \"-l\",\n        \"--limit\",\n        help=\"Maximum number of flow runs to start simultaneously.\",\n    ),\n    with_healthcheck: bool = typer.Option(\n        False, help=\"Start a healthcheck server for the worker.\"\n    ),\n    install_policy: InstallPolicy = typer.Option(\n        InstallPolicy.PROMPT.value,\n        \"--install-policy\",\n        help=\"Install policy to use workers from Prefect integration packages.\",\n        case_sensitive=False,\n    ),\n    base_job_template: typer.FileText = typer.Option(\n        None,\n        \"--base-job-template\",\n        help=(\n            \"The path to a JSON file containing the base job template to use. If\"\n            \" unspecified, Prefect will use the default base job template for the given\"\n            \" worker type. If the work pool already exists, this will be ignored.\"\n        ),\n    ),\n):\n    \"\"\"\n    Start a worker process to poll a work pool for flow runs.\n    \"\"\"\n\n    is_paused = await _check_work_pool_paused(work_pool_name)\n    if is_paused:\n        app.console.print(\n            (\n                f\"The work pool {work_pool_name!r} is currently paused. This worker\"\n                \" will not execute any flow runs until the work pool is unpaused.\"\n            ),\n            style=\"yellow\",\n        )\n\n    worker_cls = await _get_worker_class(worker_type, work_pool_name, install_policy)\n\n    if worker_cls is None:\n        exit_with_error(\n            \"Unable to start worker. Please ensure you have the necessary dependencies\"\n            \" installed to run your desired worker type.\"\n        )\n\n    worker_process_id = os.getpid()\n    setup_signal_handlers_worker(\n        worker_process_id, f\"the {worker_type} worker\", app.console.print\n    )\n\n    template_contents = None\n    if base_job_template is not None:\n        template_contents = json.load(fp=base_job_template)\n\n    async with worker_cls(\n        name=worker_name,\n        work_pool_name=work_pool_name,\n        work_queues=work_queues,\n        limit=limit,\n        prefetch_seconds=prefetch_seconds,\n        heartbeat_interval_seconds=PREFECT_WORKER_HEARTBEAT_SECONDS.value(),\n        base_job_template=template_contents,\n    ) as worker:\n        app.console.print(f\"Worker {worker.name!r} started!\", style=\"green\")\n        async with anyio.create_task_group() as tg:\n            # wait for an initial heartbeat to configure the worker\n            await worker.sync_with_backend()\n            # schedule the scheduled flow run polling loop\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=worker.get_and_submit_flow_runs,\n                    interval=PREFECT_WORKER_QUERY_SECONDS.value(),\n                    run_once=run_once,\n                    printer=app.console.print,\n                    jitter_range=0.3,\n                    backoff=4,  # Up to ~1 minute interval during backoff\n                )\n            )\n            # schedule the sync loop\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=worker.sync_with_backend,\n                    interval=worker.heartbeat_interval_seconds,\n                    run_once=run_once,\n                    printer=app.console.print,\n                    jitter_range=0.3,\n                    backoff=4,\n                )\n            )\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=worker.check_for_cancelled_flow_runs,\n                    interval=PREFECT_WORKER_QUERY_SECONDS.value() * 2,\n                    run_once=run_once,\n                    printer=app.console.print,\n                    jitter_range=0.3,\n                    backoff=4,\n                )\n            )\n\n            started_event = await worker._emit_worker_started_event()\n\n            # if --with-healthcheck was passed, start the healthcheck server\n            if with_healthcheck:\n                # we'll start the ASGI server in a separate thread so that\n                # uvicorn does not block the main thread\n                server_thread = threading.Thread(\n                    name=\"healthcheck-server-thread\",\n                    target=partial(\n                        start_healthcheck_server,\n                        worker=worker,\n                        query_interval_seconds=PREFECT_WORKER_QUERY_SECONDS.value(),\n                    ),\n                    daemon=True,\n                )\n                server_thread.start()\n\n    await worker._emit_worker_stopped_event(started_event)\n    app.console.print(f\"Worker {worker.name!r} stopped!\")\n
    ","tags":["Python API","workers","CLI"]},{"location":"api-ref/prefect/client/base/","title":"base","text":"","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base","title":"prefect.client.base","text":"","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectResponse","title":"PrefectResponse","text":"

    Bases: Response

    A Prefect wrapper for the httpx.Response class.

    Provides more informative error messages.

    Source code in prefect/client/base.py
    class PrefectResponse(httpx.Response):\n    \"\"\"\n    A Prefect wrapper for the `httpx.Response` class.\n\n    Provides more informative error messages.\n    \"\"\"\n\n    def raise_for_status(self) -> None:\n        \"\"\"\n        Raise an exception if the response contains an HTTPStatusError.\n\n        The `PrefectHTTPStatusError` contains useful additional information that\n        is not contained in the `HTTPStatusError`.\n        \"\"\"\n        try:\n            return super().raise_for_status()\n        except HTTPStatusError as exc:\n            raise PrefectHTTPStatusError.from_httpx_error(exc) from exc.__cause__\n\n    @classmethod\n    def from_httpx_response(cls: Type[Self], response: httpx.Response) -> Self:\n        \"\"\"\n        Create a `PrefectReponse` from an `httpx.Response`.\n\n        By changing the `__class__` attribute of the Response, we change the method\n        resolution order to look for methods defined in PrefectResponse, while leaving\n        everything else about the original Response instance intact.\n        \"\"\"\n        new_response = copy.copy(response)\n        new_response.__class__ = cls\n        return new_response\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectResponse.raise_for_status","title":"raise_for_status","text":"

    Raise an exception if the response contains an HTTPStatusError.

    The PrefectHTTPStatusError contains useful additional information that is not contained in the HTTPStatusError.

    Source code in prefect/client/base.py
    def raise_for_status(self) -> None:\n    \"\"\"\n    Raise an exception if the response contains an HTTPStatusError.\n\n    The `PrefectHTTPStatusError` contains useful additional information that\n    is not contained in the `HTTPStatusError`.\n    \"\"\"\n    try:\n        return super().raise_for_status()\n    except HTTPStatusError as exc:\n        raise PrefectHTTPStatusError.from_httpx_error(exc) from exc.__cause__\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectResponse.from_httpx_response","title":"from_httpx_response classmethod","text":"

    Create a PrefectReponse from an httpx.Response.

    By changing the __class__ attribute of the Response, we change the method resolution order to look for methods defined in PrefectResponse, while leaving everything else about the original Response instance intact.

    Source code in prefect/client/base.py
    @classmethod\ndef from_httpx_response(cls: Type[Self], response: httpx.Response) -> Self:\n    \"\"\"\n    Create a `PrefectReponse` from an `httpx.Response`.\n\n    By changing the `__class__` attribute of the Response, we change the method\n    resolution order to look for methods defined in PrefectResponse, while leaving\n    everything else about the original Response instance intact.\n    \"\"\"\n    new_response = copy.copy(response)\n    new_response.__class__ = cls\n    return new_response\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectHttpxClient","title":"PrefectHttpxClient","text":"

    Bases: AsyncClient

    A Prefect wrapper for the async httpx client with support for retry-after headers for the provided status codes (typically 429, 502 and 503).

    Additionally, this client will always call raise_for_status on responses.

    For more details on rate limit headers, see: Configuring Cloudflare Rate Limiting

    Source code in prefect/client/base.py
    class PrefectHttpxClient(httpx.AsyncClient):\n    \"\"\"\n    A Prefect wrapper for the async httpx client with support for retry-after headers\n    for the provided status codes (typically 429, 502 and 503).\n\n    Additionally, this client will always call `raise_for_status` on responses.\n\n    For more details on rate limit headers, see:\n    [Configuring Cloudflare Rate Limiting](https://support.cloudflare.com/hc/en-us/articles/115001635128-Configuring-Rate-Limiting-from-UI)\n    \"\"\"\n\n    async def _send_with_retry(\n        self,\n        request: Callable,\n        retry_codes: Set[int] = set(),\n        retry_exceptions: Tuple[Exception, ...] = tuple(),\n    ):\n        \"\"\"\n        Send a request and retry it if it fails.\n\n        Sends the provided request and retries it up to PREFECT_CLIENT_MAX_RETRIES times\n        if the request either raises an exception listed in `retry_exceptions` or\n        receives a response with a status code listed in `retry_codes`.\n\n        Retries will be delayed based on either the retry header (preferred) or\n        exponential backoff if a retry header is not provided.\n        \"\"\"\n        try_count = 0\n        response = None\n\n        while try_count <= PREFECT_CLIENT_MAX_RETRIES.value():\n            try_count += 1\n            retry_seconds = None\n            exc_info = None\n\n            try:\n                response = await request()\n            except retry_exceptions:  # type: ignore\n                if try_count > PREFECT_CLIENT_MAX_RETRIES.value():\n                    raise\n                # Otherwise, we will ignore this error but capture the info for logging\n                exc_info = sys.exc_info()\n            else:\n                # We got a response; return immediately if it is not retryable\n                if response.status_code not in retry_codes:\n                    return response\n\n                if \"Retry-After\" in response.headers:\n                    retry_seconds = float(response.headers[\"Retry-After\"])\n\n            # Use an exponential back-off if not set in a header\n            if retry_seconds is None:\n                retry_seconds = 2**try_count\n\n            # Add jitter\n            jitter_factor = PREFECT_CLIENT_RETRY_JITTER_FACTOR.value()\n            if retry_seconds > 0 and jitter_factor > 0:\n                if response is not None and \"Retry-After\" in response.headers:\n                    # Always wait for _at least_ retry seconds if requested by the API\n                    retry_seconds = bounded_poisson_interval(\n                        retry_seconds, retry_seconds * (1 + jitter_factor)\n                    )\n                else:\n                    # Otherwise, use a symmetrical jitter\n                    retry_seconds = clamped_poisson_interval(\n                        retry_seconds, jitter_factor\n                    )\n\n            logger.debug(\n                (\n                    \"Encountered retryable exception during request. \"\n                    if exc_info\n                    else (\n                        \"Received response with retryable status code\"\n                        f\" {response.status_code}. \"\n                    )\n                )\n                + f\"Another attempt will be made in {retry_seconds}s. \"\n                \"This is attempt\"\n                f\" {try_count}/{PREFECT_CLIENT_MAX_RETRIES.value() + 1}.\",\n                exc_info=exc_info,\n            )\n            await anyio.sleep(retry_seconds)\n\n        assert (\n            response is not None\n        ), \"Retry handling ended without response or exception\"\n\n        # We ran out of retries, return the failed response\n        return response\n\n    async def send(self, *args, **kwargs) -> Response:\n        \"\"\"\n        Send a request with automatic retry behavior for the following status codes:\n\n        - 429 CloudFlare-style rate limiting\n        - 502 Bad Gateway\n        - 503 Service unavailable\n        \"\"\"\n\n        api_request = partial(super().send, *args, **kwargs)\n\n        response = await self._send_with_retry(\n            request=api_request,\n            retry_codes={\n                status.HTTP_429_TOO_MANY_REQUESTS,\n                status.HTTP_503_SERVICE_UNAVAILABLE,\n                status.HTTP_502_BAD_GATEWAY,\n                status.HTTP_408_REQUEST_TIMEOUT,\n                *PREFECT_CLIENT_RETRY_EXTRA_CODES.value(),\n            },\n            retry_exceptions=(\n                httpx.ReadTimeout,\n                httpx.PoolTimeout,\n                httpx.ConnectTimeout,\n                # `ConnectionResetError` when reading socket raises as a `ReadError`\n                httpx.ReadError,\n                # Sockets can be closed during writes resulting in a `WriteError`\n                httpx.WriteError,\n                # Uvicorn bug, see https://github.com/PrefectHQ/prefect/issues/7512\n                httpx.RemoteProtocolError,\n                # HTTP2 bug, see https://github.com/PrefectHQ/prefect/issues/7442\n                httpx.LocalProtocolError,\n            ),\n        )\n\n        # Convert to a Prefect response to add nicer errors messages\n        response = PrefectResponse.from_httpx_response(response)\n\n        # Always raise bad responses\n        # NOTE: We may want to remove this and handle responses per route in the\n        #       `PrefectClient`\n        response.raise_for_status()\n\n        return response\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectHttpxClient.send","title":"send async","text":"

    Send a request with automatic retry behavior for the following status codes:

    • 429 CloudFlare-style rate limiting
    • 502 Bad Gateway
    • 503 Service unavailable
    Source code in prefect/client/base.py
    async def send(self, *args, **kwargs) -> Response:\n    \"\"\"\n    Send a request with automatic retry behavior for the following status codes:\n\n    - 429 CloudFlare-style rate limiting\n    - 502 Bad Gateway\n    - 503 Service unavailable\n    \"\"\"\n\n    api_request = partial(super().send, *args, **kwargs)\n\n    response = await self._send_with_retry(\n        request=api_request,\n        retry_codes={\n            status.HTTP_429_TOO_MANY_REQUESTS,\n            status.HTTP_503_SERVICE_UNAVAILABLE,\n            status.HTTP_502_BAD_GATEWAY,\n            status.HTTP_408_REQUEST_TIMEOUT,\n            *PREFECT_CLIENT_RETRY_EXTRA_CODES.value(),\n        },\n        retry_exceptions=(\n            httpx.ReadTimeout,\n            httpx.PoolTimeout,\n            httpx.ConnectTimeout,\n            # `ConnectionResetError` when reading socket raises as a `ReadError`\n            httpx.ReadError,\n            # Sockets can be closed during writes resulting in a `WriteError`\n            httpx.WriteError,\n            # Uvicorn bug, see https://github.com/PrefectHQ/prefect/issues/7512\n            httpx.RemoteProtocolError,\n            # HTTP2 bug, see https://github.com/PrefectHQ/prefect/issues/7442\n            httpx.LocalProtocolError,\n        ),\n    )\n\n    # Convert to a Prefect response to add nicer errors messages\n    response = PrefectResponse.from_httpx_response(response)\n\n    # Always raise bad responses\n    # NOTE: We may want to remove this and handle responses per route in the\n    #       `PrefectClient`\n    response.raise_for_status()\n\n    return response\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.app_lifespan_context","title":"app_lifespan_context async","text":"

    A context manager that calls startup/shutdown hooks for the given application.

    Lifespan contexts are cached per application to avoid calling the lifespan hooks more than once if the context is entered in nested code. A no-op context will be returned if the context for the given application is already being managed.

    This manager is robust to concurrent access within the event loop. For example, if you have concurrent contexts for the same application, it is guaranteed that startup hooks will be called before their context starts and shutdown hooks will only be called after their context exits.

    A reference count is used to support nested use of clients without running lifespan hooks excessively. The first client context entered will create and enter a lifespan context. Each subsequent client will increment a reference count but will not create a new lifespan context. When each client context exits, the reference count is decremented. When the last client context exits, the lifespan will be closed.

    In simple nested cases, the first client context will be the one to exit the lifespan. However, if client contexts are entered concurrently they may not exit in a consistent order. If the first client context was responsible for closing the lifespan, it would have to wait until all other client contexts to exit to avoid firing shutdown hooks while the application is in use. Waiting for the other clients to exit can introduce deadlocks, so, instead, the first client will exit without closing the lifespan context and reference counts will be used to ensure the lifespan is closed once all of the clients are done.

    Source code in prefect/client/base.py
    @asynccontextmanager\nasync def app_lifespan_context(app: ASGIApp) -> AsyncGenerator[None, None]:\n    \"\"\"\n    A context manager that calls startup/shutdown hooks for the given application.\n\n    Lifespan contexts are cached per application to avoid calling the lifespan hooks\n    more than once if the context is entered in nested code. A no-op context will be\n    returned if the context for the given application is already being managed.\n\n    This manager is robust to concurrent access within the event loop. For example,\n    if you have concurrent contexts for the same application, it is guaranteed that\n    startup hooks will be called before their context starts and shutdown hooks will\n    only be called after their context exits.\n\n    A reference count is used to support nested use of clients without running\n    lifespan hooks excessively. The first client context entered will create and enter\n    a lifespan context. Each subsequent client will increment a reference count but will\n    not create a new lifespan context. When each client context exits, the reference\n    count is decremented. When the last client context exits, the lifespan will be\n    closed.\n\n    In simple nested cases, the first client context will be the one to exit the\n    lifespan. However, if client contexts are entered concurrently they may not exit\n    in a consistent order. If the first client context was responsible for closing\n    the lifespan, it would have to wait until all other client contexts to exit to\n    avoid firing shutdown hooks while the application is in use. Waiting for the other\n    clients to exit can introduce deadlocks, so, instead, the first client will exit\n    without closing the lifespan context and reference counts will be used to ensure\n    the lifespan is closed once all of the clients are done.\n    \"\"\"\n    # TODO: A deadlock has been observed during multithreaded use of clients while this\n    #       lifespan context is being used. This has only been reproduced on Python 3.7\n    #       and while we hope to discourage using multiple event loops in threads, this\n    #       bug may emerge again.\n    #       See https://github.com/PrefectHQ/orion/pull/1696\n    thread_id = threading.get_ident()\n\n    # The id of the application is used instead of the hash so each application instance\n    # is managed independently even if they share the same settings. We include the\n    # thread id since applications are managed separately per thread.\n    key = (thread_id, id(app))\n\n    # On exception, this will be populated with exception details\n    exc_info = (None, None, None)\n\n    # Get a lock unique to this thread since anyio locks are not threadsafe\n    lock = APP_LIFESPANS_LOCKS[thread_id]\n\n    async with lock:\n        if key in APP_LIFESPANS:\n            # The lifespan is already being managed, just increment the reference count\n            APP_LIFESPANS_REF_COUNTS[key] += 1\n        else:\n            # Create a new lifespan manager\n            APP_LIFESPANS[key] = context = LifespanManager(\n                app, startup_timeout=30, shutdown_timeout=30\n            )\n            APP_LIFESPANS_REF_COUNTS[key] = 1\n\n            # Ensure we enter the context before releasing the lock so startup hooks\n            # are complete before another client can be used\n            await context.__aenter__()\n\n    try:\n        yield\n    except BaseException:\n        exc_info = sys.exc_info()\n        raise\n    finally:\n        # If we do not shield against anyio cancellation, the lock will return\n        # immediately and the code in its context will not run, leaving the lifespan\n        # open\n        with anyio.CancelScope(shield=True):\n            async with lock:\n                # After the consumer exits the context, decrement the reference count\n                APP_LIFESPANS_REF_COUNTS[key] -= 1\n\n                # If this the last context to exit, close the lifespan\n                if APP_LIFESPANS_REF_COUNTS[key] <= 0:\n                    APP_LIFESPANS_REF_COUNTS.pop(key)\n                    context = APP_LIFESPANS.pop(key)\n                    await context.__aexit__(*exc_info)\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/","title":"cloud","text":"","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud","title":"prefect.client.cloud","text":"","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud.CloudUnauthorizedError","title":"CloudUnauthorizedError","text":"

    Bases: PrefectException

    Raised when the CloudClient receives a 401 or 403 from the Cloud API.

    Source code in prefect/client/cloud.py
    class CloudUnauthorizedError(PrefectException):\n    \"\"\"\n    Raised when the CloudClient receives a 401 or 403 from the Cloud API.\n    \"\"\"\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud.CloudClient","title":"CloudClient","text":"Source code in prefect/client/cloud.py
    class CloudClient:\n    def __init__(\n        self,\n        host: str,\n        api_key: str,\n        httpx_settings: dict = None,\n    ) -> None:\n        httpx_settings = httpx_settings or dict()\n        httpx_settings.setdefault(\"headers\", dict())\n        httpx_settings[\"headers\"].setdefault(\"Authorization\", f\"Bearer {api_key}\")\n\n        httpx_settings.setdefault(\"base_url\", host)\n        if not PREFECT_UNIT_TEST_MODE.value():\n            httpx_settings.setdefault(\"follow_redirects\", True)\n        self._client = PrefectHttpxClient(**httpx_settings)\n\n    async def api_healthcheck(self):\n        \"\"\"\n        Attempts to connect to the Cloud API and raises the encountered exception if not\n        successful.\n\n        If successful, returns `None`.\n        \"\"\"\n        with anyio.fail_after(10):\n            await self.read_workspaces()\n\n    async def read_workspaces(self) -> List[Workspace]:\n        return pydantic.parse_obj_as(List[Workspace], await self.get(\"/me/workspaces\"))\n\n    async def read_worker_metadata(self) -> Dict[str, Any]:\n        configured_url = prefect.settings.PREFECT_API_URL.value()\n        account_id, workspace_id = re.findall(PARSE_API_URL_REGEX, configured_url)[0]\n        return await self.get(\n            f\"accounts/{account_id}/workspaces/{workspace_id}/collections/work_pool_types\"\n        )\n\n    async def __aenter__(self):\n        await self._client.__aenter__()\n        return self\n\n    async def __aexit__(self, *exc_info):\n        return await self._client.__aexit__(*exc_info)\n\n    def __enter__(self):\n        raise RuntimeError(\n            \"The `CloudClient` must be entered with an async context. Use 'async \"\n            \"with CloudClient(...)' not 'with CloudClient(...)'\"\n        )\n\n    def __exit__(self, *_):\n        assert False, \"This should never be called but must be defined for __enter__\"\n\n    async def get(self, route, **kwargs):\n        return await self.request(\"GET\", route, **kwargs)\n\n    async def request(self, method, route, **kwargs):\n        try:\n            res = await self._client.request(method, route, **kwargs)\n            res.raise_for_status()\n        except httpx.HTTPStatusError as exc:\n            if exc.response.status_code in (\n                status.HTTP_401_UNAUTHORIZED,\n                status.HTTP_403_FORBIDDEN,\n            ):\n                raise CloudUnauthorizedError\n            else:\n                raise exc\n\n        if res.status_code == status.HTTP_204_NO_CONTENT:\n            return\n\n        return res.json()\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud.CloudClient.api_healthcheck","title":"api_healthcheck async","text":"

    Attempts to connect to the Cloud API and raises the encountered exception if not successful.

    If successful, returns None.

    Source code in prefect/client/cloud.py
    async def api_healthcheck(self):\n    \"\"\"\n    Attempts to connect to the Cloud API and raises the encountered exception if not\n    successful.\n\n    If successful, returns `None`.\n    \"\"\"\n    with anyio.fail_after(10):\n        await self.read_workspaces()\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud.get_cloud_client","title":"get_cloud_client","text":"

    Needs a docstring.

    Source code in prefect/client/cloud.py
    def get_cloud_client(\n    host: Optional[str] = None,\n    api_key: Optional[str] = None,\n    httpx_settings: Optional[dict] = None,\n    infer_cloud_url: bool = False,\n) -> \"CloudClient\":\n    \"\"\"\n    Needs a docstring.\n    \"\"\"\n    if httpx_settings is not None:\n        httpx_settings = httpx_settings.copy()\n\n    if infer_cloud_url is False:\n        host = host or PREFECT_CLOUD_API_URL.value()\n    else:\n        configured_url = prefect.settings.PREFECT_API_URL.value()\n        host = re.sub(PARSE_API_URL_REGEX, \"\", configured_url)\n\n    return CloudClient(\n        host=host,\n        api_key=api_key or PREFECT_API_KEY.value(),\n        httpx_settings=httpx_settings,\n    )\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/orchestration/","title":"orchestration","text":"

    Asynchronous client implementation for communicating with the Prefect REST API.

    Explore the client by communicating with an in-memory webserver \u2014 no setup required:

    $ # start python REPL with native await functionality\n$ python -m asyncio\n>>> from prefect import get_client\n>>> async with get_client() as client:\n...     response = await client.hello()\n...     print(response.json())\n\ud83d\udc4b\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration","title":"prefect.client.orchestration","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient","title":"PrefectClient","text":"

    An asynchronous client for interacting with the Prefect REST API.

    Parameters:

    Name Type Description Default api Union[str, ASGIApp]

    the REST API URL or FastAPI application to connect to

    required api_key str

    An optional API key for authentication.

    None api_version str

    The API version this client is compatible with.

    None httpx_settings dict

    An optional dictionary of settings to pass to the underlying httpx.AsyncClient

    None
    Say hello to a Prefect REST API\n\n<div class=\"terminal\">\n```\n>>> async with get_client() as client:\n>>>     response = await client.hello()\n>>>\n>>> print(response.json())\n\ud83d\udc4b\n```\n</div>\n
    Source code in prefect/client/orchestration.py
    class PrefectClient:\n    \"\"\"\n    An asynchronous client for interacting with the [Prefect REST API](/api-ref/rest-api/).\n\n    Args:\n        api: the REST API URL or FastAPI application to connect to\n        api_key: An optional API key for authentication.\n        api_version: The API version this client is compatible with.\n        httpx_settings: An optional dictionary of settings to pass to the underlying\n            `httpx.AsyncClient`\n\n    Examples:\n\n        Say hello to a Prefect REST API\n\n        <div class=\"terminal\">\n        ```\n        >>> async with get_client() as client:\n        >>>     response = await client.hello()\n        >>>\n        >>> print(response.json())\n        \ud83d\udc4b\n        ```\n        </div>\n    \"\"\"\n\n    def __init__(\n        self,\n        api: Union[str, ASGIApp],\n        *,\n        api_key: str = None,\n        api_version: str = None,\n        httpx_settings: dict = None,\n    ) -> None:\n        httpx_settings = httpx_settings.copy() if httpx_settings else {}\n        httpx_settings.setdefault(\"headers\", {})\n\n        if PREFECT_API_TLS_INSECURE_SKIP_VERIFY:\n            httpx_settings.setdefault(\"verify\", False)\n\n        if api_version is None:\n            api_version = SERVER_API_VERSION\n        httpx_settings[\"headers\"].setdefault(\"X-PREFECT-API-VERSION\", api_version)\n        if api_key:\n            httpx_settings[\"headers\"].setdefault(\"Authorization\", f\"Bearer {api_key}\")\n\n        # Context management\n        self._exit_stack = AsyncExitStack()\n        self._ephemeral_app: Optional[ASGIApp] = None\n        self.manage_lifespan = True\n        self.server_type: ServerType\n\n        # Only set if this client started the lifespan of the application\n        self._ephemeral_lifespan: Optional[LifespanManager] = None\n\n        self._closed = False\n        self._started = False\n\n        # Connect to an external application\n        if isinstance(api, str):\n            if httpx_settings.get(\"app\"):\n                raise ValueError(\n                    \"Invalid httpx settings: `app` cannot be set when providing an \"\n                    \"api url. `app` is only for use with ephemeral instances. Provide \"\n                    \"it as the `api` parameter instead.\"\n                )\n            httpx_settings.setdefault(\"base_url\", api)\n\n            # See https://www.python-httpx.org/advanced/#pool-limit-configuration\n            httpx_settings.setdefault(\n                \"limits\",\n                httpx.Limits(\n                    # We see instability when allowing the client to open many connections at once.\n                    # Limiting concurrency results in more stable performance.\n                    max_connections=16,\n                    max_keepalive_connections=8,\n                    # The Prefect Cloud LB will keep connections alive for 30s.\n                    # Only allow the client to keep connections alive for 25s.\n                    keepalive_expiry=25,\n                ),\n            )\n\n            # See https://www.python-httpx.org/http2/\n            # Enabling HTTP/2 support on the client does not necessarily mean that your requests\n            # and responses will be transported over HTTP/2, since both the client and the server\n            # need to support HTTP/2. If you connect to a server that only supports HTTP/1.1 the\n            # client will use a standard HTTP/1.1 connection instead.\n            httpx_settings.setdefault(\"http2\", PREFECT_API_ENABLE_HTTP2.value())\n\n            self.server_type = (\n                ServerType.CLOUD\n                if api.startswith(PREFECT_CLOUD_API_URL.value())\n                else ServerType.SERVER\n            )\n\n        # Connect to an in-process application\n        elif isinstance(api, ASGIApp):\n            self._ephemeral_app = api\n            self.server_type = ServerType.EPHEMERAL\n\n            # When using an ephemeral server, server-side exceptions can be raised\n            # client-side breaking all of our response error code handling. To work\n            # around this, we create an ASGI transport with application exceptions\n            # disabled instead of using the application directly.\n            # refs:\n            # - https://github.com/PrefectHQ/prefect/pull/9637\n            # - https://github.com/encode/starlette/blob/d3a11205ed35f8e5a58a711db0ff59c86fa7bb31/starlette/middleware/errors.py#L184\n            # - https://github.com/tiangolo/fastapi/blob/8cc967a7605d3883bd04ceb5d25cc94ae079612f/fastapi/applications.py#L163-L164\n            httpx_settings.setdefault(\n                \"transport\",\n                httpx.ASGITransport(\n                    app=self._ephemeral_app, raise_app_exceptions=False\n                ),\n            )\n            httpx_settings.setdefault(\"base_url\", \"http://ephemeral-prefect/api\")\n\n        else:\n            raise TypeError(\n                f\"Unexpected type {type(api).__name__!r} for argument `api`. Expected\"\n                \" 'str' or 'ASGIApp/FastAPI'\"\n            )\n\n        # See https://www.python-httpx.org/advanced/#timeout-configuration\n        httpx_settings.setdefault(\n            \"timeout\",\n            httpx.Timeout(\n                connect=PREFECT_API_REQUEST_TIMEOUT.value(),\n                read=PREFECT_API_REQUEST_TIMEOUT.value(),\n                write=PREFECT_API_REQUEST_TIMEOUT.value(),\n                pool=PREFECT_API_REQUEST_TIMEOUT.value(),\n            ),\n        )\n\n        if not PREFECT_UNIT_TEST_MODE:\n            httpx_settings.setdefault(\"follow_redirects\", True)\n        self._client = PrefectHttpxClient(**httpx_settings)\n        self._loop = None\n\n        # See https://www.python-httpx.org/advanced/#custom-transports\n        #\n        # If we're using an HTTP/S client (not the ephemeral client), adjust the\n        # transport to add retries _after_ it is instantiated. If we alter the transport\n        # before instantiation, the transport will not be aware of proxies unless we\n        # reproduce all of the logic to make it so.\n        #\n        # Only alter the transport to set our default of 3 retries, don't modify any\n        # transport a user may have provided via httpx_settings.\n        #\n        # Making liberal use of getattr and isinstance checks here to avoid any\n        # surprises if the internals of httpx or httpcore change on us\n        if isinstance(api, str) and not httpx_settings.get(\"transport\"):\n            transport_for_url = getattr(self._client, \"_transport_for_url\", None)\n            if callable(transport_for_url):\n                server_transport = transport_for_url(httpx.URL(api))\n                if isinstance(server_transport, httpx.AsyncHTTPTransport):\n                    pool = getattr(server_transport, \"_pool\", None)\n                    if isinstance(pool, httpcore.AsyncConnectionPool):\n                        pool._retries = 3\n\n        self.logger = get_logger(\"client\")\n\n    @property\n    def api_url(self) -> httpx.URL:\n        \"\"\"\n        Get the base URL for the API.\n        \"\"\"\n        return self._client.base_url\n\n    # API methods ----------------------------------------------------------------------\n\n    async def api_healthcheck(self) -> Optional[Exception]:\n        \"\"\"\n        Attempts to connect to the API and returns the encountered exception if not\n        successful.\n\n        If successful, returns `None`.\n        \"\"\"\n        try:\n            await self._client.get(\"/health\")\n            return None\n        except Exception as exc:\n            return exc\n\n    async def hello(self) -> httpx.Response:\n        \"\"\"\n        Send a GET request to /hello for testing purposes.\n        \"\"\"\n        return await self._client.get(\"/hello\")\n\n    async def create_flow(self, flow: \"FlowObject\") -> UUID:\n        \"\"\"\n        Create a flow in the Prefect API.\n\n        Args:\n            flow: a [Flow][prefect.flows.Flow] object\n\n        Raises:\n            httpx.RequestError: if a flow was not created for any reason\n\n        Returns:\n            the ID of the flow in the backend\n        \"\"\"\n        return await self.create_flow_from_name(flow.name)\n\n    async def create_flow_from_name(self, flow_name: str) -> UUID:\n        \"\"\"\n        Create a flow in the Prefect API.\n\n        Args:\n            flow_name: the name of the new flow\n\n        Raises:\n            httpx.RequestError: if a flow was not created for any reason\n\n        Returns:\n            the ID of the flow in the backend\n        \"\"\"\n        flow_data = FlowCreate(name=flow_name)\n        response = await self._client.post(\n            \"/flows/\", json=flow_data.dict(json_compatible=True)\n        )\n\n        flow_id = response.json().get(\"id\")\n        if not flow_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        # Return the id of the created flow\n        return UUID(flow_id)\n\n    async def read_flow(self, flow_id: UUID) -> Flow:\n        \"\"\"\n        Query the Prefect API for a flow by id.\n\n        Args:\n            flow_id: the flow ID of interest\n\n        Returns:\n            a [Flow model][prefect.client.schemas.objects.Flow] representation of the flow\n        \"\"\"\n        response = await self._client.get(f\"/flows/{flow_id}\")\n        return Flow.parse_obj(response.json())\n\n    async def read_flows(\n        self,\n        *,\n        flow_filter: FlowFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        deployment_filter: DeploymentFilter = None,\n        work_pool_filter: WorkPoolFilter = None,\n        work_queue_filter: WorkQueueFilter = None,\n        sort: FlowSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[Flow]:\n        \"\"\"\n        Query the Prefect API for flows. Only flows matching all criteria will\n        be returned.\n\n        Args:\n            flow_filter: filter criteria for flows\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            deployment_filter: filter criteria for deployments\n            work_pool_filter: filter criteria for work pools\n            work_queue_filter: filter criteria for work pool queues\n            sort: sort criteria for the flows\n            limit: limit for the flow query\n            offset: offset for the flow query\n\n        Returns:\n            a list of Flow model representations of the flows\n        \"\"\"\n        body = {\n            \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n                if flow_run_filter\n                else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"deployments\": (\n                deployment_filter.dict(json_compatible=True)\n                if deployment_filter\n                else None\n            ),\n            \"work_pools\": (\n                work_pool_filter.dict(json_compatible=True)\n                if work_pool_filter\n                else None\n            ),\n            \"work_queues\": (\n                work_queue_filter.dict(json_compatible=True)\n                if work_queue_filter\n                else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n\n        response = await self._client.post(\"/flows/filter\", json=body)\n        return pydantic.parse_obj_as(List[Flow], response.json())\n\n    async def read_flow_by_name(\n        self,\n        flow_name: str,\n    ) -> Flow:\n        \"\"\"\n        Query the Prefect API for a flow by name.\n\n        Args:\n            flow_name: the name of a flow\n\n        Returns:\n            a fully hydrated Flow model\n        \"\"\"\n        response = await self._client.get(f\"/flows/name/{flow_name}\")\n        return Flow.parse_obj(response.json())\n\n    async def create_flow_run_from_deployment(\n        self,\n        deployment_id: UUID,\n        *,\n        parameters: Dict[str, Any] = None,\n        context: dict = None,\n        state: prefect.states.State = None,\n        name: str = None,\n        tags: Iterable[str] = None,\n        idempotency_key: str = None,\n        parent_task_run_id: UUID = None,\n        work_queue_name: str = None,\n    ) -> FlowRun:\n        \"\"\"\n        Create a flow run for a deployment.\n\n        Args:\n            deployment_id: The deployment ID to create the flow run from\n            parameters: Parameter overrides for this flow run. Merged with the\n                deployment defaults\n            context: Optional run context data\n            state: The initial state for the run. If not provided, defaults to\n                `Scheduled` for now. Should always be a `Scheduled` type.\n            name: An optional name for the flow run. If not provided, the server will\n                generate a name.\n            tags: An optional iterable of tags to apply to the flow run; these tags\n                are merged with the deployment's tags.\n            idempotency_key: Optional idempotency key for creation of the flow run.\n                If the key matches the key of an existing flow run, the existing run will\n                be returned instead of creating a new one.\n            parent_task_run_id: if a subflow run is being created, the placeholder task\n                run identifier in the parent flow\n            work_queue_name: An optional work queue name to add this run to. If not provided,\n                will default to the deployment's set work queue.  If one is provided that does not\n                exist, a new work queue will be created within the deployment's work pool.\n\n        Raises:\n            httpx.RequestError: if the Prefect API does not successfully create a run for any reason\n\n        Returns:\n            The flow run model\n        \"\"\"\n        parameters = parameters or {}\n        context = context or {}\n        state = state or prefect.states.Scheduled()\n        tags = tags or []\n\n        flow_run_create = DeploymentFlowRunCreate(\n            parameters=parameters,\n            context=context,\n            state=state.to_state_create(),\n            tags=tags,\n            name=name,\n            idempotency_key=idempotency_key,\n            parent_task_run_id=parent_task_run_id,\n        )\n\n        # done separately to avoid including this field in payloads sent to older API versions\n        if work_queue_name:\n            flow_run_create.work_queue_name = work_queue_name\n\n        response = await self._client.post(\n            f\"/deployments/{deployment_id}/create_flow_run\",\n            json=flow_run_create.dict(json_compatible=True, exclude_unset=True),\n        )\n        return FlowRun.parse_obj(response.json())\n\n    async def create_flow_run(\n        self,\n        flow: \"FlowObject\",\n        name: str = None,\n        parameters: Dict[str, Any] = None,\n        context: dict = None,\n        tags: Iterable[str] = None,\n        parent_task_run_id: UUID = None,\n        state: \"prefect.states.State\" = None,\n    ) -> FlowRun:\n        \"\"\"\n        Create a flow run for a flow.\n\n        Args:\n            flow: The flow model to create the flow run for\n            name: An optional name for the flow run\n            parameters: Parameter overrides for this flow run.\n            context: Optional run context data\n            tags: a list of tags to apply to this flow run\n            parent_task_run_id: if a subflow run is being created, the placeholder task\n                run identifier in the parent flow\n            state: The initial state for the run. If not provided, defaults to\n                `Scheduled` for now. Should always be a `Scheduled` type.\n\n        Raises:\n            httpx.RequestError: if the Prefect API does not successfully create a run for any reason\n\n        Returns:\n            The flow run model\n        \"\"\"\n        parameters = parameters or {}\n        context = context or {}\n\n        if state is None:\n            state = prefect.states.Pending()\n\n        # Retrieve the flow id\n        flow_id = await self.create_flow(flow)\n\n        flow_run_create = FlowRunCreate(\n            flow_id=flow_id,\n            flow_version=flow.version,\n            name=name,\n            parameters=parameters,\n            context=context,\n            tags=list(tags or []),\n            parent_task_run_id=parent_task_run_id,\n            state=state.to_state_create(),\n            empirical_policy=FlowRunPolicy(\n                retries=flow.retries,\n                retry_delay=flow.retry_delay_seconds,\n            ),\n        )\n\n        flow_run_create_json = flow_run_create.dict(json_compatible=True)\n        response = await self._client.post(\"/flow_runs/\", json=flow_run_create_json)\n        flow_run = FlowRun.parse_obj(response.json())\n\n        # Restore the parameters to the local objects to retain expectations about\n        # Python objects\n        flow_run.parameters = parameters\n\n        return flow_run\n\n    async def update_flow_run(\n        self,\n        flow_run_id: UUID,\n        flow_version: Optional[str] = None,\n        parameters: Optional[dict] = None,\n        name: Optional[str] = None,\n        tags: Optional[Iterable[str]] = None,\n        empirical_policy: Optional[FlowRunPolicy] = None,\n        infrastructure_pid: Optional[str] = None,\n    ) -> httpx.Response:\n        \"\"\"\n        Update a flow run's details.\n\n        Args:\n            flow_run_id: The identifier for the flow run to update.\n            flow_version: A new version string for the flow run.\n            parameters: A dictionary of parameter values for the flow run. This will not\n                be merged with any existing parameters.\n            name: A new name for the flow run.\n            empirical_policy: A new flow run orchestration policy. This will not be\n                merged with any existing policy.\n            tags: An iterable of new tags for the flow run. These will not be merged with\n                any existing tags.\n            infrastructure_pid: The id of flow run as returned by an\n                infrastructure block.\n\n        Returns:\n            an `httpx.Response` object from the PATCH request\n        \"\"\"\n        params = {}\n        if flow_version is not None:\n            params[\"flow_version\"] = flow_version\n        if parameters is not None:\n            params[\"parameters\"] = parameters\n        if name is not None:\n            params[\"name\"] = name\n        if tags is not None:\n            params[\"tags\"] = tags\n        if empirical_policy is not None:\n            params[\"empirical_policy\"] = empirical_policy\n        if infrastructure_pid:\n            params[\"infrastructure_pid\"] = infrastructure_pid\n\n        flow_run_data = FlowRunUpdate(**params)\n\n        return await self._client.patch(\n            f\"/flow_runs/{flow_run_id}\",\n            json=flow_run_data.dict(json_compatible=True, exclude_unset=True),\n        )\n\n    async def delete_flow_run(\n        self,\n        flow_run_id: UUID,\n    ) -> None:\n        \"\"\"\n        Delete a flow run by UUID.\n\n        Args:\n            flow_run_id: The flow run UUID of interest.\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If requests fails\n        \"\"\"\n        try:\n            await self._client.delete(f\"/flow_runs/{flow_run_id}\"),\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def create_concurrency_limit(\n        self,\n        tag: str,\n        concurrency_limit: int,\n    ) -> UUID:\n        \"\"\"\n        Create a tag concurrency limit in the Prefect API. These limits govern concurrently\n        running tasks.\n\n        Args:\n            tag: a tag the concurrency limit is applied to\n            concurrency_limit: the maximum number of concurrent task runs for a given tag\n\n        Raises:\n            httpx.RequestError: if the concurrency limit was not created for any reason\n\n        Returns:\n            the ID of the concurrency limit in the backend\n        \"\"\"\n\n        concurrency_limit_create = ConcurrencyLimitCreate(\n            tag=tag,\n            concurrency_limit=concurrency_limit,\n        )\n        response = await self._client.post(\n            \"/concurrency_limits/\",\n            json=concurrency_limit_create.dict(json_compatible=True),\n        )\n\n        concurrency_limit_id = response.json().get(\"id\")\n\n        if not concurrency_limit_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        return UUID(concurrency_limit_id)\n\n    async def read_concurrency_limit_by_tag(\n        self,\n        tag: str,\n    ):\n        \"\"\"\n        Read the concurrency limit set on a specific tag.\n\n        Args:\n            tag: a tag the concurrency limit is applied to\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: if the concurrency limit was not created for any reason\n\n        Returns:\n            the concurrency limit set on a specific tag\n        \"\"\"\n        try:\n            response = await self._client.get(\n                f\"/concurrency_limits/tag/{tag}\",\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n        concurrency_limit_id = response.json().get(\"id\")\n\n        if not concurrency_limit_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        concurrency_limit = ConcurrencyLimit.parse_obj(response.json())\n        return concurrency_limit\n\n    async def read_concurrency_limits(\n        self,\n        limit: int,\n        offset: int,\n    ):\n        \"\"\"\n        Lists concurrency limits set on task run tags.\n\n        Args:\n            limit: the maximum number of concurrency limits returned\n            offset: the concurrency limit query offset\n\n        Returns:\n            a list of concurrency limits\n        \"\"\"\n\n        body = {\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n\n        response = await self._client.post(\"/concurrency_limits/filter\", json=body)\n        return pydantic.parse_obj_as(List[ConcurrencyLimit], response.json())\n\n    async def reset_concurrency_limit_by_tag(\n        self,\n        tag: str,\n        slot_override: Optional[List[Union[UUID, str]]] = None,\n    ):\n        \"\"\"\n        Resets the concurrency limit slots set on a specific tag.\n\n        Args:\n            tag: a tag the concurrency limit is applied to\n            slot_override: a list of task run IDs that are currently using a\n                concurrency slot, please check that any task run IDs included in\n                `slot_override` are currently running, otherwise those concurrency\n                slots will never be released.\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        \"\"\"\n        if slot_override is not None:\n            slot_override = [str(slot) for slot in slot_override]\n\n        try:\n            await self._client.post(\n                f\"/concurrency_limits/tag/{tag}/reset\",\n                json=dict(slot_override=slot_override),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_concurrency_limit_by_tag(\n        self,\n        tag: str,\n    ):\n        \"\"\"\n        Delete the concurrency limit set on a specific tag.\n\n        Args:\n            tag: a tag the concurrency limit is applied to\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        \"\"\"\n        try:\n            await self._client.delete(\n                f\"/concurrency_limits/tag/{tag}\",\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def create_work_queue(\n        self,\n        name: str,\n        tags: Optional[List[str]] = None,\n        description: Optional[str] = None,\n        is_paused: Optional[bool] = None,\n        concurrency_limit: Optional[int] = None,\n        priority: Optional[int] = None,\n        work_pool_name: Optional[str] = None,\n    ) -> WorkQueue:\n        \"\"\"\n        Create a work queue.\n\n        Args:\n            name: a unique name for the work queue\n            tags: DEPRECATED: an optional list of tags to filter on; only work scheduled with these tags\n                will be included in the queue. This option will be removed on 2023-02-23.\n            description: An optional description for the work queue.\n            is_paused: Whether or not the work queue is paused.\n            concurrency_limit: An optional concurrency limit for the work queue.\n            priority: The queue's priority. Lower values are higher priority (1 is the highest).\n            work_pool_name: The name of the work pool to use for this queue.\n\n        Raises:\n            prefect.exceptions.ObjectAlreadyExists: If request returns 409\n            httpx.RequestError: If request fails\n\n        Returns:\n            The created work queue\n        \"\"\"\n        if tags:\n            warnings.warn(\n                (\n                    \"The use of tags for creating work queue filters is deprecated.\"\n                    \" This option will be removed on 2023-02-23.\"\n                ),\n                DeprecationWarning,\n            )\n            filter = QueueFilter(tags=tags)\n        else:\n            filter = None\n        create_model = WorkQueueCreate(name=name, filter=filter)\n        if description is not None:\n            create_model.description = description\n        if is_paused is not None:\n            create_model.is_paused = is_paused\n        if concurrency_limit is not None:\n            create_model.concurrency_limit = concurrency_limit\n        if priority is not None:\n            create_model.priority = priority\n\n        data = create_model.dict(json_compatible=True)\n        try:\n            if work_pool_name is not None:\n                response = await self._client.post(\n                    f\"/work_pools/{work_pool_name}/queues\", json=data\n                )\n            else:\n                response = await self._client.post(\"/work_queues/\", json=data)\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            elif e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return WorkQueue.parse_obj(response.json())\n\n    async def read_work_queue_by_name(\n        self,\n        name: str,\n        work_pool_name: Optional[str] = None,\n    ) -> WorkQueue:\n        \"\"\"\n        Read a work queue by name.\n\n        Args:\n            name (str): a unique name for the work queue\n            work_pool_name (str, optional): the name of the work pool\n                the queue belongs to.\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: if no work queue is found\n            httpx.HTTPStatusError: other status errors\n\n        Returns:\n            WorkQueue: a work queue API object\n        \"\"\"\n        try:\n            if work_pool_name is not None:\n                response = await self._client.get(\n                    f\"/work_pools/{work_pool_name}/queues/{name}\"\n                )\n            else:\n                response = await self._client.get(f\"/work_queues/name/{name}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n        return WorkQueue.parse_obj(response.json())\n\n    async def update_work_queue(self, id: UUID, **kwargs):\n        \"\"\"\n        Update properties of a work queue.\n\n        Args:\n            id: the ID of the work queue to update\n            **kwargs: the fields to update\n\n        Raises:\n            ValueError: if no kwargs are provided\n            prefect.exceptions.ObjectNotFound: if request returns 404\n            httpx.RequestError: if the request fails\n\n        \"\"\"\n        if not kwargs:\n            raise ValueError(\"No fields provided to update.\")\n\n        data = WorkQueueUpdate(**kwargs).dict(json_compatible=True, exclude_unset=True)\n        try:\n            await self._client.patch(f\"/work_queues/{id}\", json=data)\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def get_runs_in_work_queue(\n        self,\n        id: UUID,\n        limit: int = 10,\n        scheduled_before: datetime.datetime = None,\n    ) -> List[FlowRun]:\n        \"\"\"\n        Read flow runs off a work queue.\n\n        Args:\n            id: the id of the work queue to read from\n            limit: a limit on the number of runs to return\n            scheduled_before: a timestamp; only runs scheduled before this time will be returned.\n                Defaults to now.\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        Returns:\n            List[FlowRun]: a list of FlowRun objects read from the queue\n        \"\"\"\n        if scheduled_before is None:\n            scheduled_before = pendulum.now(\"UTC\")\n\n        try:\n            response = await self._client.post(\n                f\"/work_queues/{id}/get_runs\",\n                json={\n                    \"limit\": limit,\n                    \"scheduled_before\": scheduled_before.isoformat(),\n                },\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return pydantic.parse_obj_as(List[FlowRun], response.json())\n\n    async def read_work_queue(\n        self,\n        id: UUID,\n    ) -> WorkQueue:\n        \"\"\"\n        Read a work queue.\n\n        Args:\n            id: the id of the work queue to load\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        Returns:\n            WorkQueue: an instantiated WorkQueue object\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/work_queues/{id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return WorkQueue.parse_obj(response.json())\n\n    async def read_work_queue_status(\n        self,\n        id: UUID,\n    ) -> WorkQueueStatusDetail:\n        \"\"\"\n        Read a work queue status.\n\n        Args:\n            id: the id of the work queue to load\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        Returns:\n            WorkQueueStatus: an instantiated WorkQueueStatus object\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/work_queues/{id}/status\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return WorkQueueStatusDetail.parse_obj(response.json())\n\n    async def match_work_queues(\n        self,\n        prefixes: List[str],\n        work_pool_name: Optional[str] = None,\n    ) -> List[WorkQueue]:\n        \"\"\"\n        Query the Prefect API for work queues with names with a specific prefix.\n\n        Args:\n            prefixes: a list of strings used to match work queue name prefixes\n            work_pool_name: an optional work pool name to scope the query to\n\n        Returns:\n            a list of WorkQueue model representations\n                of the work queues\n        \"\"\"\n        page_length = 100\n        current_page = 0\n        work_queues = []\n\n        while True:\n            new_queues = await self.read_work_queues(\n                work_pool_name=work_pool_name,\n                offset=current_page * page_length,\n                limit=page_length,\n                work_queue_filter=WorkQueueFilter(\n                    name=WorkQueueFilterName(startswith_=prefixes)\n                ),\n            )\n            if not new_queues:\n                break\n            work_queues += new_queues\n            current_page += 1\n\n        return work_queues\n\n    async def delete_work_queue_by_id(\n        self,\n        id: UUID,\n    ):\n        \"\"\"\n        Delete a work queue by its ID.\n\n        Args:\n            id: the id of the work queue to delete\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If requests fails\n        \"\"\"\n        try:\n            await self._client.delete(\n                f\"/work_queues/{id}\",\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def create_block_type(self, block_type: BlockTypeCreate) -> BlockType:\n        \"\"\"\n        Create a block type in the Prefect API.\n        \"\"\"\n        try:\n            response = await self._client.post(\n                \"/block_types/\",\n                json=block_type.dict(\n                    json_compatible=True, exclude_unset=True, exclude={\"id\"}\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            else:\n                raise\n        return BlockType.parse_obj(response.json())\n\n    async def create_block_schema(self, block_schema: BlockSchemaCreate) -> BlockSchema:\n        \"\"\"\n        Create a block schema in the Prefect API.\n        \"\"\"\n        try:\n            response = await self._client.post(\n                \"/block_schemas/\",\n                json=block_schema.dict(\n                    json_compatible=True,\n                    exclude_unset=True,\n                    exclude={\"id\", \"block_type\", \"checksum\"},\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            else:\n                raise\n        return BlockSchema.parse_obj(response.json())\n\n    async def create_block_document(\n        self,\n        block_document: Union[BlockDocument, BlockDocumentCreate],\n        include_secrets: bool = True,\n    ) -> BlockDocument:\n        \"\"\"\n        Create a block document in the Prefect API. This data is used to configure a\n        corresponding Block.\n\n        Args:\n            include_secrets (bool): whether to include secret values\n                on the stored Block, corresponding to Pydantic's `SecretStr` and\n                `SecretBytes` fields. Note Blocks may not work as expected if\n                this is set to `False`.\n        \"\"\"\n        if isinstance(block_document, BlockDocument):\n            block_document = BlockDocumentCreate.parse_obj(\n                block_document.dict(\n                    json_compatible=True,\n                    include_secrets=include_secrets,\n                    exclude_unset=True,\n                    exclude={\"id\", \"block_schema\", \"block_type\"},\n                ),\n            )\n\n        try:\n            response = await self._client.post(\n                \"/block_documents/\",\n                json=block_document.dict(\n                    json_compatible=True,\n                    include_secrets=include_secrets,\n                    exclude_unset=True,\n                    exclude={\"id\", \"block_schema\", \"block_type\"},\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            else:\n                raise\n        return BlockDocument.parse_obj(response.json())\n\n    async def update_block_document(\n        self,\n        block_document_id: UUID,\n        block_document: BlockDocumentUpdate,\n    ):\n        \"\"\"\n        Update a block document in the Prefect API.\n        \"\"\"\n        try:\n            await self._client.patch(\n                f\"/block_documents/{block_document_id}\",\n                json=block_document.dict(\n                    json_compatible=True,\n                    exclude_unset=True,\n                    include={\"data\", \"merge_existing_data\", \"block_schema_id\"},\n                    include_secrets=True,\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_block_document(self, block_document_id: UUID):\n        \"\"\"\n        Delete a block document.\n        \"\"\"\n        try:\n            await self._client.delete(f\"/block_documents/{block_document_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_block_type_by_slug(self, slug: str) -> BlockType:\n        \"\"\"\n        Read a block type by its slug.\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/block_types/slug/{slug}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return BlockType.parse_obj(response.json())\n\n    async def read_block_schema_by_checksum(\n        self, checksum: str, version: Optional[str] = None\n    ) -> BlockSchema:\n        \"\"\"\n        Look up a block schema checksum\n        \"\"\"\n        try:\n            url = f\"/block_schemas/checksum/{checksum}\"\n            if version is not None:\n                url = f\"{url}?version={version}\"\n            response = await self._client.get(url)\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return BlockSchema.parse_obj(response.json())\n\n    async def update_block_type(self, block_type_id: UUID, block_type: BlockTypeUpdate):\n        \"\"\"\n        Update a block document in the Prefect API.\n        \"\"\"\n        try:\n            await self._client.patch(\n                f\"/block_types/{block_type_id}\",\n                json=block_type.dict(\n                    json_compatible=True,\n                    exclude_unset=True,\n                    include=BlockTypeUpdate.updatable_fields(),\n                    include_secrets=True,\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_block_type(self, block_type_id: UUID):\n        \"\"\"\n        Delete a block type.\n        \"\"\"\n        try:\n            await self._client.delete(f\"/block_types/{block_type_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            elif (\n                e.response.status_code == status.HTTP_403_FORBIDDEN\n                and e.response.json()[\"detail\"]\n                == \"protected block types cannot be deleted.\"\n            ):\n                raise prefect.exceptions.ProtectedBlockError(\n                    \"Protected block types cannot be deleted.\"\n                ) from e\n            else:\n                raise\n\n    async def read_block_types(self) -> List[BlockType]:\n        \"\"\"\n        Read all block types\n        Raises:\n            httpx.RequestError: if the block types were not found\n\n        Returns:\n            List of BlockTypes.\n        \"\"\"\n        response = await self._client.post(\"/block_types/filter\", json={})\n        return pydantic.parse_obj_as(List[BlockType], response.json())\n\n    async def read_block_schemas(self) -> List[BlockSchema]:\n        \"\"\"\n        Read all block schemas\n        Raises:\n            httpx.RequestError: if a valid block schema was not found\n\n        Returns:\n            A BlockSchema.\n        \"\"\"\n        response = await self._client.post(\"/block_schemas/filter\", json={})\n        return pydantic.parse_obj_as(List[BlockSchema], response.json())\n\n    async def get_most_recent_block_schema_for_block_type(\n        self,\n        block_type_id: UUID,\n    ) -> Optional[BlockSchema]:\n        \"\"\"\n        Fetches the most recent block schema for a specified block type ID.\n\n        Args:\n            block_type_id: The ID of the block type.\n\n        Raises:\n            httpx.RequestError: If the request fails for any reason.\n\n        Returns:\n            The most recent block schema or None.\n        \"\"\"\n        try:\n            response = await self._client.post(\n                \"/block_schemas/filter\",\n                json={\n                    \"block_schemas\": {\"block_type_id\": {\"any_\": [str(block_type_id)]}},\n                    \"limit\": 1,\n                },\n            )\n        except httpx.HTTPStatusError:\n            raise\n        return BlockSchema.parse_obj(response.json()[0]) if response.json() else None\n\n    async def read_block_document(\n        self,\n        block_document_id: UUID,\n        include_secrets: bool = True,\n    ):\n        \"\"\"\n        Read the block document with the specified ID.\n\n        Args:\n            block_document_id: the block document id\n            include_secrets (bool): whether to include secret values\n                on the Block, corresponding to Pydantic's `SecretStr` and\n                `SecretBytes` fields. These fields are automatically obfuscated\n                by Pydantic, but users can additionally choose not to receive\n                their values from the API. Note that any business logic on the\n                Block may not work if this is `False`.\n\n        Raises:\n            httpx.RequestError: if the block document was not found for any reason\n\n        Returns:\n            A block document or None.\n        \"\"\"\n        assert (\n            block_document_id is not None\n        ), \"Unexpected ID on block document. Was it persisted?\"\n        try:\n            response = await self._client.get(\n                f\"/block_documents/{block_document_id}\",\n                params=dict(include_secrets=include_secrets),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return BlockDocument.parse_obj(response.json())\n\n    async def read_block_document_by_name(\n        self,\n        name: str,\n        block_type_slug: str,\n        include_secrets: bool = True,\n    ) -> BlockDocument:\n        \"\"\"\n        Read the block document with the specified name that corresponds to a\n        specific block type name.\n\n        Args:\n            name: The block document name.\n            block_type_slug: The block type slug.\n            include_secrets (bool): whether to include secret values\n                on the Block, corresponding to Pydantic's `SecretStr` and\n                `SecretBytes` fields. These fields are automatically obfuscated\n                by Pydantic, but users can additionally choose not to receive\n                their values from the API. Note that any business logic on the\n                Block may not work if this is `False`.\n\n        Raises:\n            httpx.RequestError: if the block document was not found for any reason\n\n        Returns:\n            A block document or None.\n        \"\"\"\n        try:\n            response = await self._client.get(\n                f\"/block_types/slug/{block_type_slug}/block_documents/name/{name}\",\n                params=dict(include_secrets=include_secrets),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return BlockDocument.parse_obj(response.json())\n\n    async def read_block_documents(\n        self,\n        block_schema_type: Optional[str] = None,\n        offset: Optional[int] = None,\n        limit: Optional[int] = None,\n        include_secrets: bool = True,\n    ):\n        \"\"\"\n        Read block documents\n\n        Args:\n            block_schema_type: an optional block schema type\n            offset: an offset\n            limit: the number of blocks to return\n            include_secrets (bool): whether to include secret values\n                on the Block, corresponding to Pydantic's `SecretStr` and\n                `SecretBytes` fields. These fields are automatically obfuscated\n                by Pydantic, but users can additionally choose not to receive\n                their values from the API. Note that any business logic on the\n                Block may not work if this is `False`.\n\n        Returns:\n            A list of block documents\n        \"\"\"\n        response = await self._client.post(\n            \"/block_documents/filter\",\n            json=dict(\n                block_schema_type=block_schema_type,\n                offset=offset,\n                limit=limit,\n                include_secrets=include_secrets,\n            ),\n        )\n        return pydantic.parse_obj_as(List[BlockDocument], response.json())\n\n    async def read_block_documents_by_type(\n        self,\n        block_type_slug: str,\n        offset: Optional[int] = None,\n        limit: Optional[int] = None,\n        include_secrets: bool = True,\n    ) -> List[BlockDocument]:\n        \"\"\"Retrieve block documents by block type slug.\n\n        Args:\n            block_type_slug: The block type slug.\n            offset: an offset\n            limit: the number of blocks to return\n            include_secrets: whether to include secret values\n\n        Returns:\n            A list of block documents\n        \"\"\"\n        response = await self._client.get(\n            f\"/block_types/slug/{block_type_slug}/block_documents\",\n            params=dict(\n                offset=offset,\n                limit=limit,\n                include_secrets=include_secrets,\n            ),\n        )\n\n        return pydantic.parse_obj_as(List[BlockDocument], response.json())\n\n    async def create_deployment(\n        self,\n        flow_id: UUID,\n        name: str,\n        version: str = None,\n        schedule: SCHEDULE_TYPES = None,\n        parameters: Dict[str, Any] = None,\n        description: str = None,\n        work_queue_name: str = None,\n        work_pool_name: str = None,\n        tags: List[str] = None,\n        storage_document_id: UUID = None,\n        manifest_path: str = None,\n        path: str = None,\n        entrypoint: str = None,\n        infrastructure_document_id: UUID = None,\n        infra_overrides: Dict[str, Any] = None,\n        parameter_openapi_schema: dict = None,\n        is_schedule_active: Optional[bool] = None,\n        pull_steps: Optional[List[dict]] = None,\n        enforce_parameter_schema: Optional[bool] = None,\n    ) -> UUID:\n        \"\"\"\n        Create a deployment.\n\n        Args:\n            flow_id: the flow ID to create a deployment for\n            name: the name of the deployment\n            version: an optional version string for the deployment\n            schedule: an optional schedule to apply to the deployment\n            tags: an optional list of tags to apply to the deployment\n            storage_document_id: an reference to the storage block document\n                used for the deployed flow\n            infrastructure_document_id: an reference to the infrastructure block document\n                to use for this deployment\n\n        Raises:\n            httpx.RequestError: if the deployment was not created for any reason\n\n        Returns:\n            the ID of the deployment in the backend\n        \"\"\"\n        deployment_create = DeploymentCreate(\n            flow_id=flow_id,\n            name=name,\n            version=version,\n            schedule=schedule,\n            parameters=dict(parameters or {}),\n            tags=list(tags or []),\n            work_queue_name=work_queue_name,\n            description=description,\n            storage_document_id=storage_document_id,\n            path=path,\n            entrypoint=entrypoint,\n            manifest_path=manifest_path,  # for backwards compat\n            infrastructure_document_id=infrastructure_document_id,\n            infra_overrides=infra_overrides or {},\n            parameter_openapi_schema=parameter_openapi_schema,\n            is_schedule_active=is_schedule_active,\n            pull_steps=pull_steps,\n            enforce_parameter_schema=enforce_parameter_schema,\n        )\n\n        if work_pool_name is not None:\n            deployment_create.work_pool_name = work_pool_name\n\n        # Exclude newer fields that are not set to avoid compatibility issues\n        exclude = {\n            field\n            for field in [\"work_pool_name\", \"work_queue_name\"]\n            if field not in deployment_create.__fields_set__\n        }\n\n        if deployment_create.is_schedule_active is None:\n            exclude.add(\"is_schedule_active\")\n\n        if deployment_create.pull_steps is None:\n            exclude.add(\"pull_steps\")\n\n        if deployment_create.enforce_parameter_schema is None:\n            exclude.add(\"enforce_parameter_schema\")\n\n        json = deployment_create.dict(json_compatible=True, exclude=exclude)\n        response = await self._client.post(\n            \"/deployments/\",\n            json=json,\n        )\n        deployment_id = response.json().get(\"id\")\n        if not deployment_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        return UUID(deployment_id)\n\n    async def update_schedule(self, deployment_id: UUID, active: bool = True):\n        path = \"set_schedule_active\" if active else \"set_schedule_inactive\"\n        await self._client.post(\n            f\"/deployments/{deployment_id}/{path}\",\n        )\n\n    async def update_deployment(\n        self,\n        deployment: Deployment,\n        schedule: SCHEDULE_TYPES = None,\n        is_schedule_active: bool = None,\n    ):\n        deployment_update = DeploymentUpdate(\n            version=deployment.version,\n            schedule=schedule if schedule is not None else deployment.schedule,\n            is_schedule_active=(\n                is_schedule_active\n                if is_schedule_active is not None\n                else deployment.is_schedule_active\n            ),\n            description=deployment.description,\n            work_queue_name=deployment.work_queue_name,\n            tags=deployment.tags,\n            manifest_path=deployment.manifest_path,\n            path=deployment.path,\n            entrypoint=deployment.entrypoint,\n            parameters=deployment.parameters,\n            storage_document_id=deployment.storage_document_id,\n            infrastructure_document_id=deployment.infrastructure_document_id,\n            infra_overrides=deployment.infra_overrides,\n            enforce_parameter_schema=deployment.enforce_parameter_schema,\n        )\n\n        if getattr(deployment, \"work_pool_name\", None) is not None:\n            deployment_update.work_pool_name = deployment.work_pool_name\n\n        exclude = set()\n        if deployment.enforce_parameter_schema is None:\n            exclude.add(\"enforce_parameter_schema\")\n\n        await self._client.patch(\n            f\"/deployments/{deployment.id}\",\n            json=deployment_update.dict(json_compatible=True, exclude=exclude),\n        )\n\n    async def _create_deployment_from_schema(self, schema: DeploymentCreate) -> UUID:\n        \"\"\"\n        Create a deployment from a prepared `DeploymentCreate` schema.\n        \"\"\"\n        # TODO: We are likely to remove this method once we have considered the\n        #       packaging interface for deployments further.\n        response = await self._client.post(\n            \"/deployments/\", json=schema.dict(json_compatible=True)\n        )\n        deployment_id = response.json().get(\"id\")\n        if not deployment_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        return UUID(deployment_id)\n\n    async def read_deployment(\n        self,\n        deployment_id: UUID,\n    ) -> DeploymentResponse:\n        \"\"\"\n        Query the Prefect API for a deployment by id.\n\n        Args:\n            deployment_id: the deployment ID of interest\n\n        Returns:\n            a [Deployment model][prefect.client.schemas.objects.Deployment] representation of the deployment\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/deployments/{deployment_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return DeploymentResponse.parse_obj(response.json())\n\n    async def read_deployment_by_name(\n        self,\n        name: str,\n    ) -> DeploymentResponse:\n        \"\"\"\n        Query the Prefect API for a deployment by name.\n\n        Args:\n            name: A deployed flow's name: <FLOW_NAME>/<DEPLOYMENT_NAME>\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        Returns:\n            a Deployment model representation of the deployment\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/deployments/name/{name}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n        return DeploymentResponse.parse_obj(response.json())\n\n    async def read_deployments(\n        self,\n        *,\n        flow_filter: FlowFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        deployment_filter: DeploymentFilter = None,\n        work_pool_filter: WorkPoolFilter = None,\n        work_queue_filter: WorkQueueFilter = None,\n        limit: int = None,\n        sort: DeploymentSort = None,\n        offset: int = 0,\n    ) -> List[DeploymentResponse]:\n        \"\"\"\n        Query the Prefect API for deployments. Only deployments matching all\n        the provided criteria will be returned.\n\n        Args:\n            flow_filter: filter criteria for flows\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            deployment_filter: filter criteria for deployments\n            work_pool_filter: filter criteria for work pools\n            work_queue_filter: filter criteria for work pool queues\n            limit: a limit for the deployment query\n            offset: an offset for the deployment query\n\n        Returns:\n            a list of Deployment model representations\n                of the deployments\n        \"\"\"\n        body = {\n            \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n                if flow_run_filter\n                else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"deployments\": (\n                deployment_filter.dict(json_compatible=True)\n                if deployment_filter\n                else None\n            ),\n            \"work_pools\": (\n                work_pool_filter.dict(json_compatible=True)\n                if work_pool_filter\n                else None\n            ),\n            \"work_pool_queues\": (\n                work_queue_filter.dict(json_compatible=True)\n                if work_queue_filter\n                else None\n            ),\n            \"limit\": limit,\n            \"offset\": offset,\n            \"sort\": sort,\n        }\n\n        response = await self._client.post(\"/deployments/filter\", json=body)\n        return pydantic.parse_obj_as(List[DeploymentResponse], response.json())\n\n    async def delete_deployment(\n        self,\n        deployment_id: UUID,\n    ):\n        \"\"\"\n        Delete deployment by id.\n\n        Args:\n            deployment_id: The deployment id of interest.\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If requests fails\n        \"\"\"\n        try:\n            await self._client.delete(f\"/deployments/{deployment_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_flow_run(self, flow_run_id: UUID) -> FlowRun:\n        \"\"\"\n        Query the Prefect API for a flow run by id.\n\n        Args:\n            flow_run_id: the flow run ID of interest\n\n        Returns:\n            a Flow Run model representation of the flow run\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/flow_runs/{flow_run_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return FlowRun.parse_obj(response.json())\n\n    async def resume_flow_run(\n        self, flow_run_id: UUID, run_input: Optional[Dict] = None\n    ) -> OrchestrationResult:\n        \"\"\"\n        Resumes a paused flow run.\n\n        Args:\n            flow_run_id: the flow run ID of interest\n            run_input: the input to resume the flow run with\n\n        Returns:\n            an OrchestrationResult model representation of state orchestration output\n        \"\"\"\n        try:\n            response = await self._client.post(\n                f\"/flow_runs/{flow_run_id}/resume\", json={\"run_input\": run_input}\n            )\n        except httpx.HTTPStatusError:\n            raise\n\n        return OrchestrationResult.parse_obj(response.json())\n\n    async def read_flow_runs(\n        self,\n        *,\n        flow_filter: FlowFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        deployment_filter: DeploymentFilter = None,\n        work_pool_filter: WorkPoolFilter = None,\n        work_queue_filter: WorkQueueFilter = None,\n        sort: FlowRunSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[FlowRun]:\n        \"\"\"\n        Query the Prefect API for flow runs. Only flow runs matching all criteria will\n        be returned.\n\n        Args:\n            flow_filter: filter criteria for flows\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            deployment_filter: filter criteria for deployments\n            work_pool_filter: filter criteria for work pools\n            work_queue_filter: filter criteria for work pool queues\n            sort: sort criteria for the flow runs\n            limit: limit for the flow run query\n            offset: offset for the flow run query\n\n        Returns:\n            a list of Flow Run model representations\n                of the flow runs\n        \"\"\"\n        body = {\n            \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n                if flow_run_filter\n                else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"deployments\": (\n                deployment_filter.dict(json_compatible=True)\n                if deployment_filter\n                else None\n            ),\n            \"work_pools\": (\n                work_pool_filter.dict(json_compatible=True)\n                if work_pool_filter\n                else None\n            ),\n            \"work_pool_queues\": (\n                work_queue_filter.dict(json_compatible=True)\n                if work_queue_filter\n                else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n\n        response = await self._client.post(\"/flow_runs/filter\", json=body)\n        return pydantic.parse_obj_as(List[FlowRun], response.json())\n\n    async def set_flow_run_state(\n        self,\n        flow_run_id: UUID,\n        state: \"prefect.states.State\",\n        force: bool = False,\n    ) -> OrchestrationResult:\n        \"\"\"\n        Set the state of a flow run.\n\n        Args:\n            flow_run_id: the id of the flow run\n            state: the state to set\n            force: if True, disregard orchestration logic when setting the state,\n                forcing the Prefect API to accept the state\n\n        Returns:\n            an OrchestrationResult model representation of state orchestration output\n        \"\"\"\n        state_create = state.to_state_create()\n        state_create.state_details.flow_run_id = flow_run_id\n        try:\n            response = await self._client.post(\n                f\"/flow_runs/{flow_run_id}/set_state\",\n                json=dict(state=state_create.dict(json_compatible=True), force=force),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n        return OrchestrationResult.parse_obj(response.json())\n\n    async def read_flow_run_states(\n        self, flow_run_id: UUID\n    ) -> List[prefect.states.State]:\n        \"\"\"\n        Query for the states of a flow run\n\n        Args:\n            flow_run_id: the id of the flow run\n\n        Returns:\n            a list of State model representations\n                of the flow run states\n        \"\"\"\n        response = await self._client.get(\n            \"/flow_run_states/\", params=dict(flow_run_id=str(flow_run_id))\n        )\n        return pydantic.parse_obj_as(List[prefect.states.State], response.json())\n\n    async def set_task_run_name(self, task_run_id: UUID, name: str):\n        task_run_data = TaskRunUpdate(name=name)\n        return await self._client.patch(\n            f\"/task_runs/{task_run_id}\",\n            json=task_run_data.dict(json_compatible=True, exclude_unset=True),\n        )\n\n    async def create_task_run(\n        self,\n        task: \"TaskObject\",\n        flow_run_id: UUID,\n        dynamic_key: str,\n        name: str = None,\n        extra_tags: Iterable[str] = None,\n        state: prefect.states.State = None,\n        task_inputs: Dict[\n            str,\n            List[\n                Union[\n                    TaskRunResult,\n                    Parameter,\n                    Constant,\n                ]\n            ],\n        ] = None,\n    ) -> TaskRun:\n        \"\"\"\n        Create a task run\n\n        Args:\n            task: The Task to run\n            flow_run_id: The flow run id with which to associate the task run\n            dynamic_key: A key unique to this particular run of a Task within the flow\n            name: An optional name for the task run\n            extra_tags: an optional list of extra tags to apply to the task run in\n                addition to `task.tags`\n            state: The initial state for the run. If not provided, defaults to\n                `Pending` for now. Should always be a `Scheduled` type.\n            task_inputs: the set of inputs passed to the task\n\n        Returns:\n            The created task run.\n        \"\"\"\n        tags = set(task.tags).union(extra_tags or [])\n\n        if state is None:\n            state = prefect.states.Pending()\n\n        task_run_data = TaskRunCreate(\n            name=name,\n            flow_run_id=flow_run_id,\n            task_key=task.task_key,\n            dynamic_key=dynamic_key,\n            tags=list(tags),\n            task_version=task.version,\n            empirical_policy=TaskRunPolicy(\n                retries=task.retries,\n                retry_delay=task.retry_delay_seconds,\n                retry_jitter_factor=task.retry_jitter_factor,\n            ),\n            state=state.to_state_create(),\n            task_inputs=task_inputs or {},\n        )\n\n        response = await self._client.post(\n            \"/task_runs/\", json=task_run_data.dict(json_compatible=True)\n        )\n        return TaskRun.parse_obj(response.json())\n\n    async def read_task_run(self, task_run_id: UUID) -> TaskRun:\n        \"\"\"\n        Query the Prefect API for a task run by id.\n\n        Args:\n            task_run_id: the task run ID of interest\n\n        Returns:\n            a Task Run model representation of the task run\n        \"\"\"\n        response = await self._client.get(f\"/task_runs/{task_run_id}\")\n        return TaskRun.parse_obj(response.json())\n\n    async def read_task_runs(\n        self,\n        *,\n        flow_filter: FlowFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        deployment_filter: DeploymentFilter = None,\n        sort: TaskRunSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[TaskRun]:\n        \"\"\"\n        Query the Prefect API for task runs. Only task runs matching all criteria will\n        be returned.\n\n        Args:\n            flow_filter: filter criteria for flows\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            deployment_filter: filter criteria for deployments\n            sort: sort criteria for the task runs\n            limit: a limit for the task run query\n            offset: an offset for the task run query\n\n        Returns:\n            a list of Task Run model representations\n                of the task runs\n        \"\"\"\n        body = {\n            \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n                if flow_run_filter\n                else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"deployments\": (\n                deployment_filter.dict(json_compatible=True)\n                if deployment_filter\n                else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n        response = await self._client.post(\"/task_runs/filter\", json=body)\n        return pydantic.parse_obj_as(List[TaskRun], response.json())\n\n    async def set_task_run_state(\n        self,\n        task_run_id: UUID,\n        state: prefect.states.State,\n        force: bool = False,\n    ) -> OrchestrationResult:\n        \"\"\"\n        Set the state of a task run.\n\n        Args:\n            task_run_id: the id of the task run\n            state: the state to set\n            force: if True, disregard orchestration logic when setting the state,\n                forcing the Prefect API to accept the state\n\n        Returns:\n            an OrchestrationResult model representation of state orchestration output\n        \"\"\"\n        state_create = state.to_state_create()\n        state_create.state_details.task_run_id = task_run_id\n        response = await self._client.post(\n            f\"/task_runs/{task_run_id}/set_state\",\n            json=dict(state=state_create.dict(json_compatible=True), force=force),\n        )\n        return OrchestrationResult.parse_obj(response.json())\n\n    async def read_task_run_states(\n        self, task_run_id: UUID\n    ) -> List[prefect.states.State]:\n        \"\"\"\n        Query for the states of a task run\n\n        Args:\n            task_run_id: the id of the task run\n\n        Returns:\n            a list of State model representations of the task run states\n        \"\"\"\n        response = await self._client.get(\n            \"/task_run_states/\", params=dict(task_run_id=str(task_run_id))\n        )\n        return pydantic.parse_obj_as(List[prefect.states.State], response.json())\n\n    async def create_logs(self, logs: Iterable[Union[LogCreate, dict]]) -> None:\n        \"\"\"\n        Create logs for a flow or task run\n\n        Args:\n            logs: An iterable of `LogCreate` objects or already json-compatible dicts\n        \"\"\"\n        serialized_logs = [\n            log.dict(json_compatible=True) if isinstance(log, LogCreate) else log\n            for log in logs\n        ]\n        await self._client.post(\"/logs/\", json=serialized_logs)\n\n    async def create_flow_run_notification_policy(\n        self,\n        block_document_id: UUID,\n        is_active: bool = True,\n        tags: List[str] = None,\n        state_names: List[str] = None,\n        message_template: Optional[str] = None,\n    ) -> UUID:\n        \"\"\"\n        Create a notification policy for flow runs\n\n        Args:\n            block_document_id: The block document UUID\n            is_active: Whether the notification policy is active\n            tags: List of flow tags\n            state_names: List of state names\n            message_template: Notification message template\n        \"\"\"\n        if tags is None:\n            tags = []\n        if state_names is None:\n            state_names = []\n\n        policy = FlowRunNotificationPolicyCreate(\n            block_document_id=block_document_id,\n            is_active=is_active,\n            tags=tags,\n            state_names=state_names,\n            message_template=message_template,\n        )\n        response = await self._client.post(\n            \"/flow_run_notification_policies/\",\n            json=policy.dict(json_compatible=True),\n        )\n\n        policy_id = response.json().get(\"id\")\n        if not policy_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        return UUID(policy_id)\n\n    async def read_flow_run_notification_policies(\n        self,\n        flow_run_notification_policy_filter: FlowRunNotificationPolicyFilter,\n        limit: Optional[int] = None,\n        offset: int = 0,\n    ) -> List[FlowRunNotificationPolicy]:\n        \"\"\"\n        Query the Prefect API for flow run notification policies. Only policies matching all criteria will\n        be returned.\n\n        Args:\n            flow_run_notification_policy_filter: filter criteria for notification policies\n            limit: a limit for the notification policies query\n            offset: an offset for the notification policies query\n\n        Returns:\n            a list of FlowRunNotificationPolicy model representations\n                of the notification policies\n        \"\"\"\n        body = {\n            \"flow_run_notification_policy_filter\": (\n                flow_run_notification_policy_filter.dict(json_compatible=True)\n                if flow_run_notification_policy_filter\n                else None\n            ),\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n        response = await self._client.post(\n            \"/flow_run_notification_policies/filter\", json=body\n        )\n        return pydantic.parse_obj_as(List[FlowRunNotificationPolicy], response.json())\n\n    async def read_logs(\n        self,\n        log_filter: LogFilter = None,\n        limit: int = None,\n        offset: int = None,\n        sort: LogSort = LogSort.TIMESTAMP_ASC,\n    ) -> List[Log]:\n        \"\"\"\n        Read flow and task run logs.\n        \"\"\"\n        body = {\n            \"logs\": log_filter.dict(json_compatible=True) if log_filter else None,\n            \"limit\": limit,\n            \"offset\": offset,\n            \"sort\": sort,\n        }\n\n        response = await self._client.post(\"/logs/filter\", json=body)\n        return pydantic.parse_obj_as(List[Log], response.json())\n\n    async def resolve_datadoc(self, datadoc: DataDocument) -> Any:\n        \"\"\"\n        Recursively decode possibly nested data documents.\n\n        \"server\" encoded documents will be retrieved from the server.\n\n        Args:\n            datadoc: The data document to resolve\n\n        Returns:\n            a decoded object, the innermost data\n        \"\"\"\n        if not isinstance(datadoc, DataDocument):\n            raise TypeError(\n                f\"`resolve_datadoc` received invalid type {type(datadoc).__name__}\"\n            )\n\n        async def resolve_inner(data):\n            if isinstance(data, bytes):\n                try:\n                    data = DataDocument.parse_raw(data)\n                except pydantic.ValidationError:\n                    return data\n\n            if isinstance(data, DataDocument):\n                return await resolve_inner(data.decode())\n\n            return data\n\n        return await resolve_inner(datadoc)\n\n    async def send_worker_heartbeat(\n        self,\n        work_pool_name: str,\n        worker_name: str,\n        heartbeat_interval_seconds: Optional[float] = None,\n    ):\n        \"\"\"\n        Sends a worker heartbeat for a given work pool.\n\n        Args:\n            work_pool_name: The name of the work pool to heartbeat against.\n            worker_name: The name of the worker sending the heartbeat.\n        \"\"\"\n        await self._client.post(\n            f\"/work_pools/{work_pool_name}/workers/heartbeat\",\n            json={\n                \"name\": worker_name,\n                \"heartbeat_interval_seconds\": heartbeat_interval_seconds,\n            },\n        )\n\n    async def read_workers_for_work_pool(\n        self,\n        work_pool_name: str,\n        worker_filter: Optional[WorkerFilter] = None,\n        offset: Optional[int] = None,\n        limit: Optional[int] = None,\n    ) -> List[Worker]:\n        \"\"\"\n        Reads workers for a given work pool.\n\n        Args:\n            work_pool_name: The name of the work pool for which to get\n                member workers.\n            worker_filter: Criteria by which to filter workers.\n            limit: Limit for the worker query.\n            offset: Limit for the worker query.\n        \"\"\"\n        response = await self._client.post(\n            f\"/work_pools/{work_pool_name}/workers/filter\",\n            json={\n                \"worker_filter\": (\n                    worker_filter.dict(json_compatible=True, exclude_unset=True)\n                    if worker_filter\n                    else None\n                ),\n                \"offset\": offset,\n                \"limit\": limit,\n            },\n        )\n\n        return pydantic.parse_obj_as(List[Worker], response.json())\n\n    async def read_work_pool(self, work_pool_name: str) -> WorkPool:\n        \"\"\"\n        Reads information for a given work pool\n\n        Args:\n            work_pool_name: The name of the work pool to for which to get\n                information.\n\n        Returns:\n            Information about the requested work pool.\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/work_pools/{work_pool_name}\")\n            return pydantic.parse_obj_as(WorkPool, response.json())\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_work_pools(\n        self,\n        limit: Optional[int] = None,\n        offset: int = 0,\n        work_pool_filter: Optional[WorkPoolFilter] = None,\n    ) -> List[WorkPool]:\n        \"\"\"\n        Reads work pools.\n\n        Args:\n            limit: Limit for the work pool query.\n            offset: Offset for the work pool query.\n            work_pool_filter: Criteria by which to filter work pools.\n\n        Returns:\n            A list of work pools.\n        \"\"\"\n\n        body = {\n            \"limit\": limit,\n            \"offset\": offset,\n            \"work_pools\": (\n                work_pool_filter.dict(json_compatible=True)\n                if work_pool_filter\n                else None\n            ),\n        }\n        response = await self._client.post(\"/work_pools/filter\", json=body)\n        return pydantic.parse_obj_as(List[WorkPool], response.json())\n\n    async def create_work_pool(\n        self,\n        work_pool: WorkPoolCreate,\n    ) -> WorkPool:\n        \"\"\"\n        Creates a work pool with the provided configuration.\n\n        Args:\n            work_pool: Desired configuration for the new work pool.\n\n        Returns:\n            Information about the newly created work pool.\n        \"\"\"\n        try:\n            response = await self._client.post(\n                \"/work_pools/\",\n                json=work_pool.dict(json_compatible=True, exclude_unset=True),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            else:\n                raise\n\n        return pydantic.parse_obj_as(WorkPool, response.json())\n\n    async def update_work_pool(\n        self,\n        work_pool_name: str,\n        work_pool: WorkPoolUpdate,\n    ):\n        \"\"\"\n        Updates a work pool.\n\n        Args:\n            work_pool_name: Name of the work pool to update.\n            work_pool: Fields to update in the work pool.\n        \"\"\"\n        try:\n            await self._client.patch(\n                f\"/work_pools/{work_pool_name}\",\n                json=work_pool.dict(json_compatible=True, exclude_unset=True),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_work_pool(\n        self,\n        work_pool_name: str,\n    ):\n        \"\"\"\n        Deletes a work pool.\n\n        Args:\n            work_pool_name: Name of the work pool to delete.\n        \"\"\"\n        try:\n            await self._client.delete(f\"/work_pools/{work_pool_name}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_work_queues(\n        self,\n        work_pool_name: Optional[str] = None,\n        work_queue_filter: Optional[WorkQueueFilter] = None,\n        limit: Optional[int] = None,\n        offset: Optional[int] = None,\n    ) -> List[WorkQueue]:\n        \"\"\"\n        Retrieves queues for a work pool.\n\n        Args:\n            work_pool_name: Name of the work pool for which to get queues.\n            work_queue_filter: Criteria by which to filter queues.\n            limit: Limit for the queue query.\n            offset: Limit for the queue query.\n\n        Returns:\n            List of queues for the specified work pool.\n        \"\"\"\n        json = {\n            \"work_queues\": (\n                work_queue_filter.dict(json_compatible=True, exclude_unset=True)\n                if work_queue_filter\n                else None\n            ),\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n\n        if work_pool_name:\n            try:\n                response = await self._client.post(\n                    f\"/work_pools/{work_pool_name}/queues/filter\",\n                    json=json,\n                )\n            except httpx.HTTPStatusError as e:\n                if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                    raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n                else:\n                    raise\n        else:\n            response = await self._client.post(\"/work_queues/filter\", json=json)\n\n        return pydantic.parse_obj_as(List[WorkQueue], response.json())\n\n    async def get_scheduled_flow_runs_for_deployments(\n        self,\n        deployment_ids: List[UUID],\n        scheduled_before: Optional[datetime.datetime] = None,\n        limit: Optional[int] = None,\n    ):\n        body: Dict[str, Any] = dict(deployment_ids=[str(id) for id in deployment_ids])\n        if scheduled_before:\n            body[\"scheduled_before\"] = str(scheduled_before)\n        if limit:\n            body[\"limit\"] = limit\n\n        response = await self._client.post(\n            \"/deployments/get_scheduled_flow_runs\",\n            json=body,\n        )\n\n        return pydantic.parse_obj_as(List[FlowRunResponse], response.json())\n\n    async def get_scheduled_flow_runs_for_work_pool(\n        self,\n        work_pool_name: str,\n        work_queue_names: Optional[List[str]] = None,\n        scheduled_before: Optional[datetime.datetime] = None,\n    ) -> List[WorkerFlowRunResponse]:\n        \"\"\"\n        Retrieves scheduled flow runs for the provided set of work pool queues.\n\n        Args:\n            work_pool_name: The name of the work pool that the work pool\n                queues are associated with.\n            work_queue_names: The names of the work pool queues from which\n                to get scheduled flow runs.\n            scheduled_before: Datetime used to filter returned flow runs. Flow runs\n                scheduled for after the given datetime string will not be returned.\n\n        Returns:\n            A list of worker flow run responses containing information about the\n            retrieved flow runs.\n        \"\"\"\n        body: Dict[str, Any] = {}\n        if work_queue_names is not None:\n            body[\"work_queue_names\"] = list(work_queue_names)\n        if scheduled_before:\n            body[\"scheduled_before\"] = str(scheduled_before)\n\n        response = await self._client.post(\n            f\"/work_pools/{work_pool_name}/get_scheduled_flow_runs\",\n            json=body,\n        )\n        return pydantic.parse_obj_as(List[WorkerFlowRunResponse], response.json())\n\n    async def create_artifact(\n        self,\n        artifact: ArtifactCreate,\n    ) -> Artifact:\n        \"\"\"\n        Creates an artifact with the provided configuration.\n\n        Args:\n            artifact: Desired configuration for the new artifact.\n        Returns:\n            Information about the newly created artifact.\n        \"\"\"\n\n        response = await self._client.post(\n            \"/artifacts/\",\n            json=artifact.dict(json_compatible=True, exclude_unset=True),\n        )\n\n        return pydantic.parse_obj_as(Artifact, response.json())\n\n    async def read_artifacts(\n        self,\n        *,\n        artifact_filter: ArtifactFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        sort: ArtifactSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[Artifact]:\n        \"\"\"\n        Query the Prefect API for artifacts. Only artifacts matching all criteria will\n        be returned.\n        Args:\n            artifact_filter: filter criteria for artifacts\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            sort: sort criteria for the artifacts\n            limit: limit for the artifact query\n            offset: offset for the artifact query\n        Returns:\n            a list of Artifact model representations of the artifacts\n        \"\"\"\n        body = {\n            \"artifacts\": (\n                artifact_filter.dict(json_compatible=True) if artifact_filter else None\n            ),\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True) if flow_run_filter else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n        response = await self._client.post(\"/artifacts/filter\", json=body)\n        return pydantic.parse_obj_as(List[Artifact], response.json())\n\n    async def read_latest_artifacts(\n        self,\n        *,\n        artifact_filter: ArtifactCollectionFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        sort: ArtifactCollectionSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[ArtifactCollection]:\n        \"\"\"\n        Query the Prefect API for artifacts. Only artifacts matching all criteria will\n        be returned.\n        Args:\n            artifact_filter: filter criteria for artifacts\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            sort: sort criteria for the artifacts\n            limit: limit for the artifact query\n            offset: offset for the artifact query\n        Returns:\n            a list of Artifact model representations of the artifacts\n        \"\"\"\n        body = {\n            \"artifacts\": (\n                artifact_filter.dict(json_compatible=True) if artifact_filter else None\n            ),\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True) if flow_run_filter else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n        response = await self._client.post(\"/artifacts/latest/filter\", json=body)\n        return pydantic.parse_obj_as(List[ArtifactCollection], response.json())\n\n    async def delete_artifact(self, artifact_id: UUID) -> None:\n        \"\"\"\n        Deletes an artifact with the provided id.\n\n        Args:\n            artifact_id: The id of the artifact to delete.\n        \"\"\"\n        try:\n            await self._client.delete(f\"/artifacts/{artifact_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_variable_by_name(self, name: str) -> Optional[Variable]:\n        \"\"\"Reads a variable by name. Returns None if no variable is found.\"\"\"\n        try:\n            response = await self._client.get(f\"/variables/name/{name}\")\n            return pydantic.parse_obj_as(Variable, response.json())\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                return None\n            else:\n                raise\n\n    async def delete_variable_by_name(self, name: str):\n        \"\"\"Deletes a variable by name.\"\"\"\n        try:\n            await self._client.delete(f\"/variables/name/{name}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_variables(self, limit: int = None) -> List[Variable]:\n        \"\"\"Reads all variables.\"\"\"\n        response = await self._client.post(\"/variables/filter\", json={\"limit\": limit})\n        return pydantic.parse_obj_as(List[Variable], response.json())\n\n    async def read_worker_metadata(self) -> Dict[str, Any]:\n        \"\"\"Reads worker metadata stored in Prefect collection registry.\"\"\"\n        response = await self._client.get(\"collections/views/aggregate-worker-metadata\")\n        response.raise_for_status()\n        return response.json()\n\n    async def create_automation(self, automation: Automation) -> UUID:\n        \"\"\"Creates an automation in Prefect Cloud.\"\"\"\n        if self.server_type != ServerType.CLOUD:\n            raise RuntimeError(\"Automations are only supported for Prefect Cloud.\")\n\n        response = await self._client.post(\n            \"/automations/\",\n            json=automation.dict(json_compatible=True),\n        )\n\n        return UUID(response.json()[\"id\"])\n\n    async def read_resource_related_automations(\n        self, resource_id: str\n    ) -> List[ExistingAutomation]:\n        if self.server_type != ServerType.CLOUD:\n            raise RuntimeError(\"Automations are only supported for Prefect Cloud.\")\n\n        response = await self._client.get(f\"/automations/related-to/{resource_id}\")\n        response.raise_for_status()\n        return pydantic.parse_obj_as(List[ExistingAutomation], response.json())\n\n    async def delete_resource_owned_automations(self, resource_id: str):\n        if self.server_type != ServerType.CLOUD:\n            raise RuntimeError(\"Automations are only supported for Prefect Cloud.\")\n\n        await self._client.delete(f\"/automations/owned-by/{resource_id}\")\n\n    async def increment_concurrency_slots(\n        self, names: List[str], slots: int, mode: str\n    ) -> httpx.Response:\n        return await self._client.post(\n            \"/v2/concurrency_limits/increment\",\n            json={\"names\": names, \"slots\": slots, \"mode\": mode},\n        )\n\n    async def release_concurrency_slots(\n        self, names: List[str], slots: int, occupancy_seconds: float\n    ) -> httpx.Response:\n        return await self._client.post(\n            \"/v2/concurrency_limits/decrement\",\n            json={\n                \"names\": names,\n                \"slots\": slots,\n                \"occupancy_seconds\": occupancy_seconds,\n            },\n        )\n\n    async def create_flow_run_input(\n        self, flow_run_id: UUID, key: str, value: str, sender: Optional[str] = None\n    ):\n        \"\"\"\n        Creates a flow run input.\n\n        Args:\n            flow_run_id: The flow run id.\n            key: The input key.\n            value: The input value.\n            sender: The sender of the input.\n        \"\"\"\n\n        # Initialize the input to ensure that the key is valid.\n        FlowRunInput(flow_run_id=flow_run_id, key=key, value=value)\n\n        response = await self._client.post(\n            f\"/flow_runs/{flow_run_id}/input\",\n            json={\"key\": key, \"value\": value, \"sender\": sender},\n        )\n        response.raise_for_status()\n\n    async def filter_flow_run_input(\n        self, flow_run_id: UUID, key_prefix: str, limit: int, exclude_keys: Set[str]\n    ) -> List[FlowRunInput]:\n        response = await self._client.post(\n            f\"/flow_runs/{flow_run_id}/input/filter\",\n            json={\n                \"prefix\": key_prefix,\n                \"limit\": limit,\n                \"exclude_keys\": list(exclude_keys),\n            },\n        )\n        response.raise_for_status()\n        return pydantic.parse_obj_as(List[FlowRunInput], response.json())\n\n    async def read_flow_run_input(self, flow_run_id: UUID, key: str) -> str:\n        \"\"\"\n        Reads a flow run input.\n\n        Args:\n            flow_run_id: The flow run id.\n            key: The input key.\n        \"\"\"\n        response = await self._client.get(f\"/flow_runs/{flow_run_id}/input/{key}\")\n        response.raise_for_status()\n        return response.content.decode()\n\n    async def delete_flow_run_input(self, flow_run_id: UUID, key: str):\n        \"\"\"\n        Deletes a flow run input.\n\n        Args:\n            flow_run_id: The flow run id.\n            key: The input key.\n        \"\"\"\n        response = await self._client.delete(f\"/flow_runs/{flow_run_id}/input/{key}\")\n        response.raise_for_status()\n\n    async def __aenter__(self):\n        \"\"\"\n        Start the client.\n\n        If the client is already started, this will raise an exception.\n\n        If the client is already closed, this will raise an exception. Use a new client\n        instance instead.\n        \"\"\"\n        if self._closed:\n            # httpx.AsyncClient does not allow reuse so we will not either.\n            raise RuntimeError(\n                \"The client cannot be started again after closing. \"\n                \"Retrieve a new client with `get_client()` instead.\"\n            )\n\n        if self._started:\n            # httpx.AsyncClient does not allow reentrancy so we will not either.\n            raise RuntimeError(\"The client cannot be started more than once.\")\n\n        self._loop = asyncio.get_running_loop()\n        await self._exit_stack.__aenter__()\n\n        # Enter a lifespan context if using an ephemeral application.\n        # See https://github.com/encode/httpx/issues/350\n        if self._ephemeral_app and self.manage_lifespan:\n            self._ephemeral_lifespan = await self._exit_stack.enter_async_context(\n                app_lifespan_context(self._ephemeral_app)\n            )\n\n        if self._ephemeral_app:\n            self.logger.debug(\n                \"Using ephemeral application with database at \"\n                f\"{PREFECT_API_DATABASE_CONNECTION_URL.value()}\"\n            )\n        else:\n            self.logger.debug(f\"Connecting to API at {self.api_url}\")\n\n        # Enter the httpx client's context\n        await self._exit_stack.enter_async_context(self._client)\n\n        self._started = True\n\n        return self\n\n    async def __aexit__(self, *exc_info):\n        \"\"\"\n        Shutdown the client.\n        \"\"\"\n        self._closed = True\n        return await self._exit_stack.__aexit__(*exc_info)\n\n    def __enter__(self):\n        raise RuntimeError(\n            \"The `PrefectClient` must be entered with an async context. Use 'async \"\n            \"with PrefectClient(...)' not 'with PrefectClient(...)'\"\n        )\n\n    def __exit__(self, *_):\n        assert False, \"This should never be called but must be defined for __enter__\"\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.api_url","title":"api_url: httpx.URL property","text":"

    Get the base URL for the API.

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.api_healthcheck","title":"api_healthcheck async","text":"

    Attempts to connect to the API and returns the encountered exception if not successful.

    If successful, returns None.

    Source code in prefect/client/orchestration.py
    async def api_healthcheck(self) -> Optional[Exception]:\n    \"\"\"\n    Attempts to connect to the API and returns the encountered exception if not\n    successful.\n\n    If successful, returns `None`.\n    \"\"\"\n    try:\n        await self._client.get(\"/health\")\n        return None\n    except Exception as exc:\n        return exc\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.hello","title":"hello async","text":"

    Send a GET request to /hello for testing purposes.

    Source code in prefect/client/orchestration.py
    async def hello(self) -> httpx.Response:\n    \"\"\"\n    Send a GET request to /hello for testing purposes.\n    \"\"\"\n    return await self._client.get(\"/hello\")\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow","title":"create_flow async","text":"

    Create a flow in the Prefect API.

    Parameters:

    Name Type Description Default flow Flow

    a Flow object

    required

    Raises:

    Type Description RequestError

    if a flow was not created for any reason

    Returns:

    Type Description UUID

    the ID of the flow in the backend

    Source code in prefect/client/orchestration.py
    async def create_flow(self, flow: \"FlowObject\") -> UUID:\n    \"\"\"\n    Create a flow in the Prefect API.\n\n    Args:\n        flow: a [Flow][prefect.flows.Flow] object\n\n    Raises:\n        httpx.RequestError: if a flow was not created for any reason\n\n    Returns:\n        the ID of the flow in the backend\n    \"\"\"\n    return await self.create_flow_from_name(flow.name)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_from_name","title":"create_flow_from_name async","text":"

    Create a flow in the Prefect API.

    Parameters:

    Name Type Description Default flow_name str

    the name of the new flow

    required

    Raises:

    Type Description RequestError

    if a flow was not created for any reason

    Returns:

    Type Description UUID

    the ID of the flow in the backend

    Source code in prefect/client/orchestration.py
    async def create_flow_from_name(self, flow_name: str) -> UUID:\n    \"\"\"\n    Create a flow in the Prefect API.\n\n    Args:\n        flow_name: the name of the new flow\n\n    Raises:\n        httpx.RequestError: if a flow was not created for any reason\n\n    Returns:\n        the ID of the flow in the backend\n    \"\"\"\n    flow_data = FlowCreate(name=flow_name)\n    response = await self._client.post(\n        \"/flows/\", json=flow_data.dict(json_compatible=True)\n    )\n\n    flow_id = response.json().get(\"id\")\n    if not flow_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    # Return the id of the created flow\n    return UUID(flow_id)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow","title":"read_flow async","text":"

    Query the Prefect API for a flow by id.

    Parameters:

    Name Type Description Default flow_id UUID

    the flow ID of interest

    required

    Returns:

    Type Description Flow

    a Flow model representation of the flow

    Source code in prefect/client/orchestration.py
    async def read_flow(self, flow_id: UUID) -> Flow:\n    \"\"\"\n    Query the Prefect API for a flow by id.\n\n    Args:\n        flow_id: the flow ID of interest\n\n    Returns:\n        a [Flow model][prefect.client.schemas.objects.Flow] representation of the flow\n    \"\"\"\n    response = await self._client.get(f\"/flows/{flow_id}\")\n    return Flow.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flows","title":"read_flows async","text":"

    Query the Prefect API for flows. Only flows matching all criteria will be returned.

    Parameters:

    Name Type Description Default flow_filter FlowFilter

    filter criteria for flows

    None flow_run_filter FlowRunFilter

    filter criteria for flow runs

    None task_run_filter TaskRunFilter

    filter criteria for task runs

    None deployment_filter DeploymentFilter

    filter criteria for deployments

    None work_pool_filter WorkPoolFilter

    filter criteria for work pools

    None work_queue_filter WorkQueueFilter

    filter criteria for work pool queues

    None sort FlowSort

    sort criteria for the flows

    None limit int

    limit for the flow query

    None offset int

    offset for the flow query

    0

    Returns:

    Type Description List[Flow]

    a list of Flow model representations of the flows

    Source code in prefect/client/orchestration.py
    async def read_flows(\n    self,\n    *,\n    flow_filter: FlowFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    deployment_filter: DeploymentFilter = None,\n    work_pool_filter: WorkPoolFilter = None,\n    work_queue_filter: WorkQueueFilter = None,\n    sort: FlowSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[Flow]:\n    \"\"\"\n    Query the Prefect API for flows. Only flows matching all criteria will\n    be returned.\n\n    Args:\n        flow_filter: filter criteria for flows\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        deployment_filter: filter criteria for deployments\n        work_pool_filter: filter criteria for work pools\n        work_queue_filter: filter criteria for work pool queues\n        sort: sort criteria for the flows\n        limit: limit for the flow query\n        offset: offset for the flow query\n\n    Returns:\n        a list of Flow model representations of the flows\n    \"\"\"\n    body = {\n        \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n            if flow_run_filter\n            else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"deployments\": (\n            deployment_filter.dict(json_compatible=True)\n            if deployment_filter\n            else None\n        ),\n        \"work_pools\": (\n            work_pool_filter.dict(json_compatible=True)\n            if work_pool_filter\n            else None\n        ),\n        \"work_queues\": (\n            work_queue_filter.dict(json_compatible=True)\n            if work_queue_filter\n            else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n\n    response = await self._client.post(\"/flows/filter\", json=body)\n    return pydantic.parse_obj_as(List[Flow], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_by_name","title":"read_flow_by_name async","text":"

    Query the Prefect API for a flow by name.

    Parameters:

    Name Type Description Default flow_name str

    the name of a flow

    required

    Returns:

    Type Description Flow

    a fully hydrated Flow model

    Source code in prefect/client/orchestration.py
    async def read_flow_by_name(\n    self,\n    flow_name: str,\n) -> Flow:\n    \"\"\"\n    Query the Prefect API for a flow by name.\n\n    Args:\n        flow_name: the name of a flow\n\n    Returns:\n        a fully hydrated Flow model\n    \"\"\"\n    response = await self._client.get(f\"/flows/name/{flow_name}\")\n    return Flow.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_run_from_deployment","title":"create_flow_run_from_deployment async","text":"

    Create a flow run for a deployment.

    Parameters:

    Name Type Description Default deployment_id UUID

    The deployment ID to create the flow run from

    required parameters Dict[str, Any]

    Parameter overrides for this flow run. Merged with the deployment defaults

    None context dict

    Optional run context data

    None state State

    The initial state for the run. If not provided, defaults to Scheduled for now. Should always be a Scheduled type.

    None name str

    An optional name for the flow run. If not provided, the server will generate a name.

    None tags Iterable[str]

    An optional iterable of tags to apply to the flow run; these tags are merged with the deployment's tags.

    None idempotency_key str

    Optional idempotency key for creation of the flow run. If the key matches the key of an existing flow run, the existing run will be returned instead of creating a new one.

    None parent_task_run_id UUID

    if a subflow run is being created, the placeholder task run identifier in the parent flow

    None work_queue_name str

    An optional work queue name to add this run to. If not provided, will default to the deployment's set work queue. If one is provided that does not exist, a new work queue will be created within the deployment's work pool.

    None

    Raises:

    Type Description RequestError

    if the Prefect API does not successfully create a run for any reason

    Returns:

    Type Description FlowRun

    The flow run model

    Source code in prefect/client/orchestration.py
    async def create_flow_run_from_deployment(\n    self,\n    deployment_id: UUID,\n    *,\n    parameters: Dict[str, Any] = None,\n    context: dict = None,\n    state: prefect.states.State = None,\n    name: str = None,\n    tags: Iterable[str] = None,\n    idempotency_key: str = None,\n    parent_task_run_id: UUID = None,\n    work_queue_name: str = None,\n) -> FlowRun:\n    \"\"\"\n    Create a flow run for a deployment.\n\n    Args:\n        deployment_id: The deployment ID to create the flow run from\n        parameters: Parameter overrides for this flow run. Merged with the\n            deployment defaults\n        context: Optional run context data\n        state: The initial state for the run. If not provided, defaults to\n            `Scheduled` for now. Should always be a `Scheduled` type.\n        name: An optional name for the flow run. If not provided, the server will\n            generate a name.\n        tags: An optional iterable of tags to apply to the flow run; these tags\n            are merged with the deployment's tags.\n        idempotency_key: Optional idempotency key for creation of the flow run.\n            If the key matches the key of an existing flow run, the existing run will\n            be returned instead of creating a new one.\n        parent_task_run_id: if a subflow run is being created, the placeholder task\n            run identifier in the parent flow\n        work_queue_name: An optional work queue name to add this run to. If not provided,\n            will default to the deployment's set work queue.  If one is provided that does not\n            exist, a new work queue will be created within the deployment's work pool.\n\n    Raises:\n        httpx.RequestError: if the Prefect API does not successfully create a run for any reason\n\n    Returns:\n        The flow run model\n    \"\"\"\n    parameters = parameters or {}\n    context = context or {}\n    state = state or prefect.states.Scheduled()\n    tags = tags or []\n\n    flow_run_create = DeploymentFlowRunCreate(\n        parameters=parameters,\n        context=context,\n        state=state.to_state_create(),\n        tags=tags,\n        name=name,\n        idempotency_key=idempotency_key,\n        parent_task_run_id=parent_task_run_id,\n    )\n\n    # done separately to avoid including this field in payloads sent to older API versions\n    if work_queue_name:\n        flow_run_create.work_queue_name = work_queue_name\n\n    response = await self._client.post(\n        f\"/deployments/{deployment_id}/create_flow_run\",\n        json=flow_run_create.dict(json_compatible=True, exclude_unset=True),\n    )\n    return FlowRun.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_run","title":"create_flow_run async","text":"

    Create a flow run for a flow.

    Parameters:

    Name Type Description Default flow Flow

    The flow model to create the flow run for

    required name str

    An optional name for the flow run

    None parameters Dict[str, Any]

    Parameter overrides for this flow run.

    None context dict

    Optional run context data

    None tags Iterable[str]

    a list of tags to apply to this flow run

    None parent_task_run_id UUID

    if a subflow run is being created, the placeholder task run identifier in the parent flow

    None state State

    The initial state for the run. If not provided, defaults to Scheduled for now. Should always be a Scheduled type.

    None

    Raises:

    Type Description RequestError

    if the Prefect API does not successfully create a run for any reason

    Returns:

    Type Description FlowRun

    The flow run model

    Source code in prefect/client/orchestration.py
    async def create_flow_run(\n    self,\n    flow: \"FlowObject\",\n    name: str = None,\n    parameters: Dict[str, Any] = None,\n    context: dict = None,\n    tags: Iterable[str] = None,\n    parent_task_run_id: UUID = None,\n    state: \"prefect.states.State\" = None,\n) -> FlowRun:\n    \"\"\"\n    Create a flow run for a flow.\n\n    Args:\n        flow: The flow model to create the flow run for\n        name: An optional name for the flow run\n        parameters: Parameter overrides for this flow run.\n        context: Optional run context data\n        tags: a list of tags to apply to this flow run\n        parent_task_run_id: if a subflow run is being created, the placeholder task\n            run identifier in the parent flow\n        state: The initial state for the run. If not provided, defaults to\n            `Scheduled` for now. Should always be a `Scheduled` type.\n\n    Raises:\n        httpx.RequestError: if the Prefect API does not successfully create a run for any reason\n\n    Returns:\n        The flow run model\n    \"\"\"\n    parameters = parameters or {}\n    context = context or {}\n\n    if state is None:\n        state = prefect.states.Pending()\n\n    # Retrieve the flow id\n    flow_id = await self.create_flow(flow)\n\n    flow_run_create = FlowRunCreate(\n        flow_id=flow_id,\n        flow_version=flow.version,\n        name=name,\n        parameters=parameters,\n        context=context,\n        tags=list(tags or []),\n        parent_task_run_id=parent_task_run_id,\n        state=state.to_state_create(),\n        empirical_policy=FlowRunPolicy(\n            retries=flow.retries,\n            retry_delay=flow.retry_delay_seconds,\n        ),\n    )\n\n    flow_run_create_json = flow_run_create.dict(json_compatible=True)\n    response = await self._client.post(\"/flow_runs/\", json=flow_run_create_json)\n    flow_run = FlowRun.parse_obj(response.json())\n\n    # Restore the parameters to the local objects to retain expectations about\n    # Python objects\n    flow_run.parameters = parameters\n\n    return flow_run\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_flow_run","title":"update_flow_run async","text":"

    Update a flow run's details.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The identifier for the flow run to update.

    required flow_version Optional[str]

    A new version string for the flow run.

    None parameters Optional[dict]

    A dictionary of parameter values for the flow run. This will not be merged with any existing parameters.

    None name Optional[str]

    A new name for the flow run.

    None empirical_policy Optional[FlowRunPolicy]

    A new flow run orchestration policy. This will not be merged with any existing policy.

    None tags Optional[Iterable[str]]

    An iterable of new tags for the flow run. These will not be merged with any existing tags.

    None infrastructure_pid Optional[str]

    The id of flow run as returned by an infrastructure block.

    None

    Returns:

    Type Description Response

    an httpx.Response object from the PATCH request

    Source code in prefect/client/orchestration.py
    async def update_flow_run(\n    self,\n    flow_run_id: UUID,\n    flow_version: Optional[str] = None,\n    parameters: Optional[dict] = None,\n    name: Optional[str] = None,\n    tags: Optional[Iterable[str]] = None,\n    empirical_policy: Optional[FlowRunPolicy] = None,\n    infrastructure_pid: Optional[str] = None,\n) -> httpx.Response:\n    \"\"\"\n    Update a flow run's details.\n\n    Args:\n        flow_run_id: The identifier for the flow run to update.\n        flow_version: A new version string for the flow run.\n        parameters: A dictionary of parameter values for the flow run. This will not\n            be merged with any existing parameters.\n        name: A new name for the flow run.\n        empirical_policy: A new flow run orchestration policy. This will not be\n            merged with any existing policy.\n        tags: An iterable of new tags for the flow run. These will not be merged with\n            any existing tags.\n        infrastructure_pid: The id of flow run as returned by an\n            infrastructure block.\n\n    Returns:\n        an `httpx.Response` object from the PATCH request\n    \"\"\"\n    params = {}\n    if flow_version is not None:\n        params[\"flow_version\"] = flow_version\n    if parameters is not None:\n        params[\"parameters\"] = parameters\n    if name is not None:\n        params[\"name\"] = name\n    if tags is not None:\n        params[\"tags\"] = tags\n    if empirical_policy is not None:\n        params[\"empirical_policy\"] = empirical_policy\n    if infrastructure_pid:\n        params[\"infrastructure_pid\"] = infrastructure_pid\n\n    flow_run_data = FlowRunUpdate(**params)\n\n    return await self._client.patch(\n        f\"/flow_runs/{flow_run_id}\",\n        json=flow_run_data.dict(json_compatible=True, exclude_unset=True),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_flow_run","title":"delete_flow_run async","text":"

    Delete a flow run by UUID.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The flow run UUID of interest.

    required Source code in prefect/client/orchestration.py
    async def delete_flow_run(\n    self,\n    flow_run_id: UUID,\n) -> None:\n    \"\"\"\n    Delete a flow run by UUID.\n\n    Args:\n        flow_run_id: The flow run UUID of interest.\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If requests fails\n    \"\"\"\n    try:\n        await self._client.delete(f\"/flow_runs/{flow_run_id}\"),\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_concurrency_limit","title":"create_concurrency_limit async","text":"

    Create a tag concurrency limit in the Prefect API. These limits govern concurrently running tasks.

    Parameters:

    Name Type Description Default tag str

    a tag the concurrency limit is applied to

    required concurrency_limit int

    the maximum number of concurrent task runs for a given tag

    required

    Raises:

    Type Description RequestError

    if the concurrency limit was not created for any reason

    Returns:

    Type Description UUID

    the ID of the concurrency limit in the backend

    Source code in prefect/client/orchestration.py
    async def create_concurrency_limit(\n    self,\n    tag: str,\n    concurrency_limit: int,\n) -> UUID:\n    \"\"\"\n    Create a tag concurrency limit in the Prefect API. These limits govern concurrently\n    running tasks.\n\n    Args:\n        tag: a tag the concurrency limit is applied to\n        concurrency_limit: the maximum number of concurrent task runs for a given tag\n\n    Raises:\n        httpx.RequestError: if the concurrency limit was not created for any reason\n\n    Returns:\n        the ID of the concurrency limit in the backend\n    \"\"\"\n\n    concurrency_limit_create = ConcurrencyLimitCreate(\n        tag=tag,\n        concurrency_limit=concurrency_limit,\n    )\n    response = await self._client.post(\n        \"/concurrency_limits/\",\n        json=concurrency_limit_create.dict(json_compatible=True),\n    )\n\n    concurrency_limit_id = response.json().get(\"id\")\n\n    if not concurrency_limit_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    return UUID(concurrency_limit_id)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_concurrency_limit_by_tag","title":"read_concurrency_limit_by_tag async","text":"

    Read the concurrency limit set on a specific tag.

    Parameters:

    Name Type Description Default tag str

    a tag the concurrency limit is applied to

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    if the concurrency limit was not created for any reason

    Returns:

    Type Description

    the concurrency limit set on a specific tag

    Source code in prefect/client/orchestration.py
    async def read_concurrency_limit_by_tag(\n    self,\n    tag: str,\n):\n    \"\"\"\n    Read the concurrency limit set on a specific tag.\n\n    Args:\n        tag: a tag the concurrency limit is applied to\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: if the concurrency limit was not created for any reason\n\n    Returns:\n        the concurrency limit set on a specific tag\n    \"\"\"\n    try:\n        response = await self._client.get(\n            f\"/concurrency_limits/tag/{tag}\",\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n\n    concurrency_limit_id = response.json().get(\"id\")\n\n    if not concurrency_limit_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    concurrency_limit = ConcurrencyLimit.parse_obj(response.json())\n    return concurrency_limit\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_concurrency_limits","title":"read_concurrency_limits async","text":"

    Lists concurrency limits set on task run tags.

    Parameters:

    Name Type Description Default limit int

    the maximum number of concurrency limits returned

    required offset int

    the concurrency limit query offset

    required

    Returns:

    Type Description

    a list of concurrency limits

    Source code in prefect/client/orchestration.py
    async def read_concurrency_limits(\n    self,\n    limit: int,\n    offset: int,\n):\n    \"\"\"\n    Lists concurrency limits set on task run tags.\n\n    Args:\n        limit: the maximum number of concurrency limits returned\n        offset: the concurrency limit query offset\n\n    Returns:\n        a list of concurrency limits\n    \"\"\"\n\n    body = {\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n\n    response = await self._client.post(\"/concurrency_limits/filter\", json=body)\n    return pydantic.parse_obj_as(List[ConcurrencyLimit], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.reset_concurrency_limit_by_tag","title":"reset_concurrency_limit_by_tag async","text":"

    Resets the concurrency limit slots set on a specific tag.

    Parameters:

    Name Type Description Default tag str

    a tag the concurrency limit is applied to

    required slot_override Optional[List[Union[UUID, str]]]

    a list of task run IDs that are currently using a concurrency slot, please check that any task run IDs included in slot_override are currently running, otherwise those concurrency slots will never be released.

    None

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Source code in prefect/client/orchestration.py
    async def reset_concurrency_limit_by_tag(\n    self,\n    tag: str,\n    slot_override: Optional[List[Union[UUID, str]]] = None,\n):\n    \"\"\"\n    Resets the concurrency limit slots set on a specific tag.\n\n    Args:\n        tag: a tag the concurrency limit is applied to\n        slot_override: a list of task run IDs that are currently using a\n            concurrency slot, please check that any task run IDs included in\n            `slot_override` are currently running, otherwise those concurrency\n            slots will never be released.\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    \"\"\"\n    if slot_override is not None:\n        slot_override = [str(slot) for slot in slot_override]\n\n    try:\n        await self._client.post(\n            f\"/concurrency_limits/tag/{tag}/reset\",\n            json=dict(slot_override=slot_override),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_concurrency_limit_by_tag","title":"delete_concurrency_limit_by_tag async","text":"

    Delete the concurrency limit set on a specific tag.

    Parameters:

    Name Type Description Default tag str

    a tag the concurrency limit is applied to

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Source code in prefect/client/orchestration.py
    async def delete_concurrency_limit_by_tag(\n    self,\n    tag: str,\n):\n    \"\"\"\n    Delete the concurrency limit set on a specific tag.\n\n    Args:\n        tag: a tag the concurrency limit is applied to\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    \"\"\"\n    try:\n        await self._client.delete(\n            f\"/concurrency_limits/tag/{tag}\",\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_work_queue","title":"create_work_queue async","text":"

    Create a work queue.

    Parameters:

    Name Type Description Default name str

    a unique name for the work queue

    required tags Optional[List[str]]

    will be included in the queue. This option will be removed on 2023-02-23.

    None description Optional[str]

    An optional description for the work queue.

    None is_paused Optional[bool]

    Whether or not the work queue is paused.

    None concurrency_limit Optional[int]

    An optional concurrency limit for the work queue.

    None priority Optional[int]

    The queue's priority. Lower values are higher priority (1 is the highest).

    None work_pool_name Optional[str]

    The name of the work pool to use for this queue.

    None

    Raises:

    Type Description ObjectAlreadyExists

    If request returns 409

    RequestError

    If request fails

    Returns:

    Type Description WorkQueue

    The created work queue

    Source code in prefect/client/orchestration.py
    async def create_work_queue(\n    self,\n    name: str,\n    tags: Optional[List[str]] = None,\n    description: Optional[str] = None,\n    is_paused: Optional[bool] = None,\n    concurrency_limit: Optional[int] = None,\n    priority: Optional[int] = None,\n    work_pool_name: Optional[str] = None,\n) -> WorkQueue:\n    \"\"\"\n    Create a work queue.\n\n    Args:\n        name: a unique name for the work queue\n        tags: DEPRECATED: an optional list of tags to filter on; only work scheduled with these tags\n            will be included in the queue. This option will be removed on 2023-02-23.\n        description: An optional description for the work queue.\n        is_paused: Whether or not the work queue is paused.\n        concurrency_limit: An optional concurrency limit for the work queue.\n        priority: The queue's priority. Lower values are higher priority (1 is the highest).\n        work_pool_name: The name of the work pool to use for this queue.\n\n    Raises:\n        prefect.exceptions.ObjectAlreadyExists: If request returns 409\n        httpx.RequestError: If request fails\n\n    Returns:\n        The created work queue\n    \"\"\"\n    if tags:\n        warnings.warn(\n            (\n                \"The use of tags for creating work queue filters is deprecated.\"\n                \" This option will be removed on 2023-02-23.\"\n            ),\n            DeprecationWarning,\n        )\n        filter = QueueFilter(tags=tags)\n    else:\n        filter = None\n    create_model = WorkQueueCreate(name=name, filter=filter)\n    if description is not None:\n        create_model.description = description\n    if is_paused is not None:\n        create_model.is_paused = is_paused\n    if concurrency_limit is not None:\n        create_model.concurrency_limit = concurrency_limit\n    if priority is not None:\n        create_model.priority = priority\n\n    data = create_model.dict(json_compatible=True)\n    try:\n        if work_pool_name is not None:\n            response = await self._client.post(\n                f\"/work_pools/{work_pool_name}/queues\", json=data\n            )\n        else:\n            response = await self._client.post(\"/work_queues/\", json=data)\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        elif e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return WorkQueue.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_queue_by_name","title":"read_work_queue_by_name async","text":"

    Read a work queue by name.

    Parameters:

    Name Type Description Default name str

    a unique name for the work queue

    required work_pool_name str

    the name of the work pool the queue belongs to.

    None

    Raises:

    Type Description ObjectNotFound

    if no work queue is found

    HTTPStatusError

    other status errors

    Returns:

    Name Type Description WorkQueue WorkQueue

    a work queue API object

    Source code in prefect/client/orchestration.py
    async def read_work_queue_by_name(\n    self,\n    name: str,\n    work_pool_name: Optional[str] = None,\n) -> WorkQueue:\n    \"\"\"\n    Read a work queue by name.\n\n    Args:\n        name (str): a unique name for the work queue\n        work_pool_name (str, optional): the name of the work pool\n            the queue belongs to.\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: if no work queue is found\n        httpx.HTTPStatusError: other status errors\n\n    Returns:\n        WorkQueue: a work queue API object\n    \"\"\"\n    try:\n        if work_pool_name is not None:\n            response = await self._client.get(\n                f\"/work_pools/{work_pool_name}/queues/{name}\"\n            )\n        else:\n            response = await self._client.get(f\"/work_queues/name/{name}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n\n    return WorkQueue.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_work_queue","title":"update_work_queue async","text":"

    Update properties of a work queue.

    Parameters:

    Name Type Description Default id UUID

    the ID of the work queue to update

    required **kwargs

    the fields to update

    {}

    Raises:

    Type Description ValueError

    if no kwargs are provided

    ObjectNotFound

    if request returns 404

    RequestError

    if the request fails

    Source code in prefect/client/orchestration.py
    async def update_work_queue(self, id: UUID, **kwargs):\n    \"\"\"\n    Update properties of a work queue.\n\n    Args:\n        id: the ID of the work queue to update\n        **kwargs: the fields to update\n\n    Raises:\n        ValueError: if no kwargs are provided\n        prefect.exceptions.ObjectNotFound: if request returns 404\n        httpx.RequestError: if the request fails\n\n    \"\"\"\n    if not kwargs:\n        raise ValueError(\"No fields provided to update.\")\n\n    data = WorkQueueUpdate(**kwargs).dict(json_compatible=True, exclude_unset=True)\n    try:\n        await self._client.patch(f\"/work_queues/{id}\", json=data)\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.get_runs_in_work_queue","title":"get_runs_in_work_queue async","text":"

    Read flow runs off a work queue.

    Parameters:

    Name Type Description Default id UUID

    the id of the work queue to read from

    required limit int

    a limit on the number of runs to return

    10 scheduled_before datetime

    a timestamp; only runs scheduled before this time will be returned. Defaults to now.

    None

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Returns:

    Type Description List[FlowRun]

    List[FlowRun]: a list of FlowRun objects read from the queue

    Source code in prefect/client/orchestration.py
    async def get_runs_in_work_queue(\n    self,\n    id: UUID,\n    limit: int = 10,\n    scheduled_before: datetime.datetime = None,\n) -> List[FlowRun]:\n    \"\"\"\n    Read flow runs off a work queue.\n\n    Args:\n        id: the id of the work queue to read from\n        limit: a limit on the number of runs to return\n        scheduled_before: a timestamp; only runs scheduled before this time will be returned.\n            Defaults to now.\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    Returns:\n        List[FlowRun]: a list of FlowRun objects read from the queue\n    \"\"\"\n    if scheduled_before is None:\n        scheduled_before = pendulum.now(\"UTC\")\n\n    try:\n        response = await self._client.post(\n            f\"/work_queues/{id}/get_runs\",\n            json={\n                \"limit\": limit,\n                \"scheduled_before\": scheduled_before.isoformat(),\n            },\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return pydantic.parse_obj_as(List[FlowRun], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_queue","title":"read_work_queue async","text":"

    Read a work queue.

    Parameters:

    Name Type Description Default id UUID

    the id of the work queue to load

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Returns:

    Name Type Description WorkQueue WorkQueue

    an instantiated WorkQueue object

    Source code in prefect/client/orchestration.py
    async def read_work_queue(\n    self,\n    id: UUID,\n) -> WorkQueue:\n    \"\"\"\n    Read a work queue.\n\n    Args:\n        id: the id of the work queue to load\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    Returns:\n        WorkQueue: an instantiated WorkQueue object\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/work_queues/{id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return WorkQueue.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_queue_status","title":"read_work_queue_status async","text":"

    Read a work queue status.

    Parameters:

    Name Type Description Default id UUID

    the id of the work queue to load

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Returns:

    Name Type Description WorkQueueStatus WorkQueueStatusDetail

    an instantiated WorkQueueStatus object

    Source code in prefect/client/orchestration.py
    async def read_work_queue_status(\n    self,\n    id: UUID,\n) -> WorkQueueStatusDetail:\n    \"\"\"\n    Read a work queue status.\n\n    Args:\n        id: the id of the work queue to load\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    Returns:\n        WorkQueueStatus: an instantiated WorkQueueStatus object\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/work_queues/{id}/status\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return WorkQueueStatusDetail.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.match_work_queues","title":"match_work_queues async","text":"

    Query the Prefect API for work queues with names with a specific prefix.

    Parameters:

    Name Type Description Default prefixes List[str]

    a list of strings used to match work queue name prefixes

    required work_pool_name Optional[str]

    an optional work pool name to scope the query to

    None

    Returns:

    Type Description List[WorkQueue]

    a list of WorkQueue model representations of the work queues

    Source code in prefect/client/orchestration.py
    async def match_work_queues(\n    self,\n    prefixes: List[str],\n    work_pool_name: Optional[str] = None,\n) -> List[WorkQueue]:\n    \"\"\"\n    Query the Prefect API for work queues with names with a specific prefix.\n\n    Args:\n        prefixes: a list of strings used to match work queue name prefixes\n        work_pool_name: an optional work pool name to scope the query to\n\n    Returns:\n        a list of WorkQueue model representations\n            of the work queues\n    \"\"\"\n    page_length = 100\n    current_page = 0\n    work_queues = []\n\n    while True:\n        new_queues = await self.read_work_queues(\n            work_pool_name=work_pool_name,\n            offset=current_page * page_length,\n            limit=page_length,\n            work_queue_filter=WorkQueueFilter(\n                name=WorkQueueFilterName(startswith_=prefixes)\n            ),\n        )\n        if not new_queues:\n            break\n        work_queues += new_queues\n        current_page += 1\n\n    return work_queues\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_work_queue_by_id","title":"delete_work_queue_by_id async","text":"

    Delete a work queue by its ID.

    Parameters:

    Name Type Description Default id UUID

    the id of the work queue to delete

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If requests fails

    Source code in prefect/client/orchestration.py
    async def delete_work_queue_by_id(\n    self,\n    id: UUID,\n):\n    \"\"\"\n    Delete a work queue by its ID.\n\n    Args:\n        id: the id of the work queue to delete\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If requests fails\n    \"\"\"\n    try:\n        await self._client.delete(\n            f\"/work_queues/{id}\",\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_block_type","title":"create_block_type async","text":"

    Create a block type in the Prefect API.

    Source code in prefect/client/orchestration.py
    async def create_block_type(self, block_type: BlockTypeCreate) -> BlockType:\n    \"\"\"\n    Create a block type in the Prefect API.\n    \"\"\"\n    try:\n        response = await self._client.post(\n            \"/block_types/\",\n            json=block_type.dict(\n                json_compatible=True, exclude_unset=True, exclude={\"id\"}\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        else:\n            raise\n    return BlockType.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_block_schema","title":"create_block_schema async","text":"

    Create a block schema in the Prefect API.

    Source code in prefect/client/orchestration.py
    async def create_block_schema(self, block_schema: BlockSchemaCreate) -> BlockSchema:\n    \"\"\"\n    Create a block schema in the Prefect API.\n    \"\"\"\n    try:\n        response = await self._client.post(\n            \"/block_schemas/\",\n            json=block_schema.dict(\n                json_compatible=True,\n                exclude_unset=True,\n                exclude={\"id\", \"block_type\", \"checksum\"},\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        else:\n            raise\n    return BlockSchema.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_block_document","title":"create_block_document async","text":"

    Create a block document in the Prefect API. This data is used to configure a corresponding Block.

    Parameters:

    Name Type Description Default include_secrets bool

    whether to include secret values on the stored Block, corresponding to Pydantic's SecretStr and SecretBytes fields. Note Blocks may not work as expected if this is set to False.

    True Source code in prefect/client/orchestration.py
    async def create_block_document(\n    self,\n    block_document: Union[BlockDocument, BlockDocumentCreate],\n    include_secrets: bool = True,\n) -> BlockDocument:\n    \"\"\"\n    Create a block document in the Prefect API. This data is used to configure a\n    corresponding Block.\n\n    Args:\n        include_secrets (bool): whether to include secret values\n            on the stored Block, corresponding to Pydantic's `SecretStr` and\n            `SecretBytes` fields. Note Blocks may not work as expected if\n            this is set to `False`.\n    \"\"\"\n    if isinstance(block_document, BlockDocument):\n        block_document = BlockDocumentCreate.parse_obj(\n            block_document.dict(\n                json_compatible=True,\n                include_secrets=include_secrets,\n                exclude_unset=True,\n                exclude={\"id\", \"block_schema\", \"block_type\"},\n            ),\n        )\n\n    try:\n        response = await self._client.post(\n            \"/block_documents/\",\n            json=block_document.dict(\n                json_compatible=True,\n                include_secrets=include_secrets,\n                exclude_unset=True,\n                exclude={\"id\", \"block_schema\", \"block_type\"},\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        else:\n            raise\n    return BlockDocument.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_block_document","title":"update_block_document async","text":"

    Update a block document in the Prefect API.

    Source code in prefect/client/orchestration.py
    async def update_block_document(\n    self,\n    block_document_id: UUID,\n    block_document: BlockDocumentUpdate,\n):\n    \"\"\"\n    Update a block document in the Prefect API.\n    \"\"\"\n    try:\n        await self._client.patch(\n            f\"/block_documents/{block_document_id}\",\n            json=block_document.dict(\n                json_compatible=True,\n                exclude_unset=True,\n                include={\"data\", \"merge_existing_data\", \"block_schema_id\"},\n                include_secrets=True,\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_block_document","title":"delete_block_document async","text":"

    Delete a block document.

    Source code in prefect/client/orchestration.py
    async def delete_block_document(self, block_document_id: UUID):\n    \"\"\"\n    Delete a block document.\n    \"\"\"\n    try:\n        await self._client.delete(f\"/block_documents/{block_document_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_type_by_slug","title":"read_block_type_by_slug async","text":"

    Read a block type by its slug.

    Source code in prefect/client/orchestration.py
    async def read_block_type_by_slug(self, slug: str) -> BlockType:\n    \"\"\"\n    Read a block type by its slug.\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/block_types/slug/{slug}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return BlockType.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_schema_by_checksum","title":"read_block_schema_by_checksum async","text":"

    Look up a block schema checksum

    Source code in prefect/client/orchestration.py
    async def read_block_schema_by_checksum(\n    self, checksum: str, version: Optional[str] = None\n) -> BlockSchema:\n    \"\"\"\n    Look up a block schema checksum\n    \"\"\"\n    try:\n        url = f\"/block_schemas/checksum/{checksum}\"\n        if version is not None:\n            url = f\"{url}?version={version}\"\n        response = await self._client.get(url)\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return BlockSchema.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_block_type","title":"update_block_type async","text":"

    Update a block document in the Prefect API.

    Source code in prefect/client/orchestration.py
    async def update_block_type(self, block_type_id: UUID, block_type: BlockTypeUpdate):\n    \"\"\"\n    Update a block document in the Prefect API.\n    \"\"\"\n    try:\n        await self._client.patch(\n            f\"/block_types/{block_type_id}\",\n            json=block_type.dict(\n                json_compatible=True,\n                exclude_unset=True,\n                include=BlockTypeUpdate.updatable_fields(),\n                include_secrets=True,\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_block_type","title":"delete_block_type async","text":"

    Delete a block type.

    Source code in prefect/client/orchestration.py
    async def delete_block_type(self, block_type_id: UUID):\n    \"\"\"\n    Delete a block type.\n    \"\"\"\n    try:\n        await self._client.delete(f\"/block_types/{block_type_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        elif (\n            e.response.status_code == status.HTTP_403_FORBIDDEN\n            and e.response.json()[\"detail\"]\n            == \"protected block types cannot be deleted.\"\n        ):\n            raise prefect.exceptions.ProtectedBlockError(\n                \"Protected block types cannot be deleted.\"\n            ) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_types","title":"read_block_types async","text":"

    Read all block types Raises: httpx.RequestError: if the block types were not found

    Returns:

    Type Description List[BlockType]

    List of BlockTypes.

    Source code in prefect/client/orchestration.py
    async def read_block_types(self) -> List[BlockType]:\n    \"\"\"\n    Read all block types\n    Raises:\n        httpx.RequestError: if the block types were not found\n\n    Returns:\n        List of BlockTypes.\n    \"\"\"\n    response = await self._client.post(\"/block_types/filter\", json={})\n    return pydantic.parse_obj_as(List[BlockType], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_schemas","title":"read_block_schemas async","text":"

    Read all block schemas Raises: httpx.RequestError: if a valid block schema was not found

    Returns:

    Type Description List[BlockSchema]

    A BlockSchema.

    Source code in prefect/client/orchestration.py
    async def read_block_schemas(self) -> List[BlockSchema]:\n    \"\"\"\n    Read all block schemas\n    Raises:\n        httpx.RequestError: if a valid block schema was not found\n\n    Returns:\n        A BlockSchema.\n    \"\"\"\n    response = await self._client.post(\"/block_schemas/filter\", json={})\n    return pydantic.parse_obj_as(List[BlockSchema], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.get_most_recent_block_schema_for_block_type","title":"get_most_recent_block_schema_for_block_type async","text":"

    Fetches the most recent block schema for a specified block type ID.

    Parameters:

    Name Type Description Default block_type_id UUID

    The ID of the block type.

    required

    Raises:

    Type Description RequestError

    If the request fails for any reason.

    Returns:

    Type Description Optional[BlockSchema]

    The most recent block schema or None.

    Source code in prefect/client/orchestration.py
    async def get_most_recent_block_schema_for_block_type(\n    self,\n    block_type_id: UUID,\n) -> Optional[BlockSchema]:\n    \"\"\"\n    Fetches the most recent block schema for a specified block type ID.\n\n    Args:\n        block_type_id: The ID of the block type.\n\n    Raises:\n        httpx.RequestError: If the request fails for any reason.\n\n    Returns:\n        The most recent block schema or None.\n    \"\"\"\n    try:\n        response = await self._client.post(\n            \"/block_schemas/filter\",\n            json={\n                \"block_schemas\": {\"block_type_id\": {\"any_\": [str(block_type_id)]}},\n                \"limit\": 1,\n            },\n        )\n    except httpx.HTTPStatusError:\n        raise\n    return BlockSchema.parse_obj(response.json()[0]) if response.json() else None\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_document","title":"read_block_document async","text":"

    Read the block document with the specified ID.

    Parameters:

    Name Type Description Default block_document_id UUID

    the block document id

    required include_secrets bool

    whether to include secret values on the Block, corresponding to Pydantic's SecretStr and SecretBytes fields. These fields are automatically obfuscated by Pydantic, but users can additionally choose not to receive their values from the API. Note that any business logic on the Block may not work if this is False.

    True

    Raises:

    Type Description RequestError

    if the block document was not found for any reason

    Returns:

    Type Description

    A block document or None.

    Source code in prefect/client/orchestration.py
    async def read_block_document(\n    self,\n    block_document_id: UUID,\n    include_secrets: bool = True,\n):\n    \"\"\"\n    Read the block document with the specified ID.\n\n    Args:\n        block_document_id: the block document id\n        include_secrets (bool): whether to include secret values\n            on the Block, corresponding to Pydantic's `SecretStr` and\n            `SecretBytes` fields. These fields are automatically obfuscated\n            by Pydantic, but users can additionally choose not to receive\n            their values from the API. Note that any business logic on the\n            Block may not work if this is `False`.\n\n    Raises:\n        httpx.RequestError: if the block document was not found for any reason\n\n    Returns:\n        A block document or None.\n    \"\"\"\n    assert (\n        block_document_id is not None\n    ), \"Unexpected ID on block document. Was it persisted?\"\n    try:\n        response = await self._client.get(\n            f\"/block_documents/{block_document_id}\",\n            params=dict(include_secrets=include_secrets),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return BlockDocument.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_document_by_name","title":"read_block_document_by_name async","text":"

    Read the block document with the specified name that corresponds to a specific block type name.

    Parameters:

    Name Type Description Default name str

    The block document name.

    required block_type_slug str

    The block type slug.

    required include_secrets bool

    whether to include secret values on the Block, corresponding to Pydantic's SecretStr and SecretBytes fields. These fields are automatically obfuscated by Pydantic, but users can additionally choose not to receive their values from the API. Note that any business logic on the Block may not work if this is False.

    True

    Raises:

    Type Description RequestError

    if the block document was not found for any reason

    Returns:

    Type Description BlockDocument

    A block document or None.

    Source code in prefect/client/orchestration.py
    async def read_block_document_by_name(\n    self,\n    name: str,\n    block_type_slug: str,\n    include_secrets: bool = True,\n) -> BlockDocument:\n    \"\"\"\n    Read the block document with the specified name that corresponds to a\n    specific block type name.\n\n    Args:\n        name: The block document name.\n        block_type_slug: The block type slug.\n        include_secrets (bool): whether to include secret values\n            on the Block, corresponding to Pydantic's `SecretStr` and\n            `SecretBytes` fields. These fields are automatically obfuscated\n            by Pydantic, but users can additionally choose not to receive\n            their values from the API. Note that any business logic on the\n            Block may not work if this is `False`.\n\n    Raises:\n        httpx.RequestError: if the block document was not found for any reason\n\n    Returns:\n        A block document or None.\n    \"\"\"\n    try:\n        response = await self._client.get(\n            f\"/block_types/slug/{block_type_slug}/block_documents/name/{name}\",\n            params=dict(include_secrets=include_secrets),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return BlockDocument.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_documents","title":"read_block_documents async","text":"

    Read block documents

    Parameters:

    Name Type Description Default block_schema_type Optional[str]

    an optional block schema type

    None offset Optional[int]

    an offset

    None limit Optional[int]

    the number of blocks to return

    None include_secrets bool

    whether to include secret values on the Block, corresponding to Pydantic's SecretStr and SecretBytes fields. These fields are automatically obfuscated by Pydantic, but users can additionally choose not to receive their values from the API. Note that any business logic on the Block may not work if this is False.

    True

    Returns:

    Type Description

    A list of block documents

    Source code in prefect/client/orchestration.py
    async def read_block_documents(\n    self,\n    block_schema_type: Optional[str] = None,\n    offset: Optional[int] = None,\n    limit: Optional[int] = None,\n    include_secrets: bool = True,\n):\n    \"\"\"\n    Read block documents\n\n    Args:\n        block_schema_type: an optional block schema type\n        offset: an offset\n        limit: the number of blocks to return\n        include_secrets (bool): whether to include secret values\n            on the Block, corresponding to Pydantic's `SecretStr` and\n            `SecretBytes` fields. These fields are automatically obfuscated\n            by Pydantic, but users can additionally choose not to receive\n            their values from the API. Note that any business logic on the\n            Block may not work if this is `False`.\n\n    Returns:\n        A list of block documents\n    \"\"\"\n    response = await self._client.post(\n        \"/block_documents/filter\",\n        json=dict(\n            block_schema_type=block_schema_type,\n            offset=offset,\n            limit=limit,\n            include_secrets=include_secrets,\n        ),\n    )\n    return pydantic.parse_obj_as(List[BlockDocument], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_documents_by_type","title":"read_block_documents_by_type async","text":"

    Retrieve block documents by block type slug.

    Parameters:

    Name Type Description Default block_type_slug str

    The block type slug.

    required offset Optional[int]

    an offset

    None limit Optional[int]

    the number of blocks to return

    None include_secrets bool

    whether to include secret values

    True

    Returns:

    Type Description List[BlockDocument]

    A list of block documents

    Source code in prefect/client/orchestration.py
    async def read_block_documents_by_type(\n    self,\n    block_type_slug: str,\n    offset: Optional[int] = None,\n    limit: Optional[int] = None,\n    include_secrets: bool = True,\n) -> List[BlockDocument]:\n    \"\"\"Retrieve block documents by block type slug.\n\n    Args:\n        block_type_slug: The block type slug.\n        offset: an offset\n        limit: the number of blocks to return\n        include_secrets: whether to include secret values\n\n    Returns:\n        A list of block documents\n    \"\"\"\n    response = await self._client.get(\n        f\"/block_types/slug/{block_type_slug}/block_documents\",\n        params=dict(\n            offset=offset,\n            limit=limit,\n            include_secrets=include_secrets,\n        ),\n    )\n\n    return pydantic.parse_obj_as(List[BlockDocument], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_deployment","title":"create_deployment async","text":"

    Create a deployment.

    Parameters:

    Name Type Description Default flow_id UUID

    the flow ID to create a deployment for

    required name str

    the name of the deployment

    required version str

    an optional version string for the deployment

    None schedule SCHEDULE_TYPES

    an optional schedule to apply to the deployment

    None tags List[str]

    an optional list of tags to apply to the deployment

    None storage_document_id UUID

    an reference to the storage block document used for the deployed flow

    None infrastructure_document_id UUID

    an reference to the infrastructure block document to use for this deployment

    None

    Raises:

    Type Description RequestError

    if the deployment was not created for any reason

    Returns:

    Type Description UUID

    the ID of the deployment in the backend

    Source code in prefect/client/orchestration.py
    async def create_deployment(\n    self,\n    flow_id: UUID,\n    name: str,\n    version: str = None,\n    schedule: SCHEDULE_TYPES = None,\n    parameters: Dict[str, Any] = None,\n    description: str = None,\n    work_queue_name: str = None,\n    work_pool_name: str = None,\n    tags: List[str] = None,\n    storage_document_id: UUID = None,\n    manifest_path: str = None,\n    path: str = None,\n    entrypoint: str = None,\n    infrastructure_document_id: UUID = None,\n    infra_overrides: Dict[str, Any] = None,\n    parameter_openapi_schema: dict = None,\n    is_schedule_active: Optional[bool] = None,\n    pull_steps: Optional[List[dict]] = None,\n    enforce_parameter_schema: Optional[bool] = None,\n) -> UUID:\n    \"\"\"\n    Create a deployment.\n\n    Args:\n        flow_id: the flow ID to create a deployment for\n        name: the name of the deployment\n        version: an optional version string for the deployment\n        schedule: an optional schedule to apply to the deployment\n        tags: an optional list of tags to apply to the deployment\n        storage_document_id: an reference to the storage block document\n            used for the deployed flow\n        infrastructure_document_id: an reference to the infrastructure block document\n            to use for this deployment\n\n    Raises:\n        httpx.RequestError: if the deployment was not created for any reason\n\n    Returns:\n        the ID of the deployment in the backend\n    \"\"\"\n    deployment_create = DeploymentCreate(\n        flow_id=flow_id,\n        name=name,\n        version=version,\n        schedule=schedule,\n        parameters=dict(parameters or {}),\n        tags=list(tags or []),\n        work_queue_name=work_queue_name,\n        description=description,\n        storage_document_id=storage_document_id,\n        path=path,\n        entrypoint=entrypoint,\n        manifest_path=manifest_path,  # for backwards compat\n        infrastructure_document_id=infrastructure_document_id,\n        infra_overrides=infra_overrides or {},\n        parameter_openapi_schema=parameter_openapi_schema,\n        is_schedule_active=is_schedule_active,\n        pull_steps=pull_steps,\n        enforce_parameter_schema=enforce_parameter_schema,\n    )\n\n    if work_pool_name is not None:\n        deployment_create.work_pool_name = work_pool_name\n\n    # Exclude newer fields that are not set to avoid compatibility issues\n    exclude = {\n        field\n        for field in [\"work_pool_name\", \"work_queue_name\"]\n        if field not in deployment_create.__fields_set__\n    }\n\n    if deployment_create.is_schedule_active is None:\n        exclude.add(\"is_schedule_active\")\n\n    if deployment_create.pull_steps is None:\n        exclude.add(\"pull_steps\")\n\n    if deployment_create.enforce_parameter_schema is None:\n        exclude.add(\"enforce_parameter_schema\")\n\n    json = deployment_create.dict(json_compatible=True, exclude=exclude)\n    response = await self._client.post(\n        \"/deployments/\",\n        json=json,\n    )\n    deployment_id = response.json().get(\"id\")\n    if not deployment_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    return UUID(deployment_id)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_deployment","title":"read_deployment async","text":"

    Query the Prefect API for a deployment by id.

    Parameters:

    Name Type Description Default deployment_id UUID

    the deployment ID of interest

    required

    Returns:

    Type Description DeploymentResponse

    a Deployment model representation of the deployment

    Source code in prefect/client/orchestration.py
    async def read_deployment(\n    self,\n    deployment_id: UUID,\n) -> DeploymentResponse:\n    \"\"\"\n    Query the Prefect API for a deployment by id.\n\n    Args:\n        deployment_id: the deployment ID of interest\n\n    Returns:\n        a [Deployment model][prefect.client.schemas.objects.Deployment] representation of the deployment\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/deployments/{deployment_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return DeploymentResponse.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_deployment_by_name","title":"read_deployment_by_name async","text":"

    Query the Prefect API for a deployment by name.

    Parameters:

    Name Type Description Default name str

    A deployed flow's name: / required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Returns:

    Type Description DeploymentResponse

    a Deployment model representation of the deployment

    Source code in prefect/client/orchestration.py
    async def read_deployment_by_name(\n    self,\n    name: str,\n) -> DeploymentResponse:\n    \"\"\"\n    Query the Prefect API for a deployment by name.\n\n    Args:\n        name: A deployed flow's name: <FLOW_NAME>/<DEPLOYMENT_NAME>\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    Returns:\n        a Deployment model representation of the deployment\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/deployments/name/{name}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n\n    return DeploymentResponse.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_deployments","title":"read_deployments async","text":"

    Query the Prefect API for deployments. Only deployments matching all the provided criteria will be returned.

    Parameters:

    Name Type Description Default flow_filter FlowFilter

    filter criteria for flows

    None flow_run_filter FlowRunFilter

    filter criteria for flow runs

    None task_run_filter TaskRunFilter

    filter criteria for task runs

    None deployment_filter DeploymentFilter

    filter criteria for deployments

    None work_pool_filter WorkPoolFilter

    filter criteria for work pools

    None work_queue_filter WorkQueueFilter

    filter criteria for work pool queues

    None limit int

    a limit for the deployment query

    None offset int

    an offset for the deployment query

    0

    Returns:

    Type Description List[DeploymentResponse]

    a list of Deployment model representations of the deployments

    Source code in prefect/client/orchestration.py
    async def read_deployments(\n    self,\n    *,\n    flow_filter: FlowFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    deployment_filter: DeploymentFilter = None,\n    work_pool_filter: WorkPoolFilter = None,\n    work_queue_filter: WorkQueueFilter = None,\n    limit: int = None,\n    sort: DeploymentSort = None,\n    offset: int = 0,\n) -> List[DeploymentResponse]:\n    \"\"\"\n    Query the Prefect API for deployments. Only deployments matching all\n    the provided criteria will be returned.\n\n    Args:\n        flow_filter: filter criteria for flows\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        deployment_filter: filter criteria for deployments\n        work_pool_filter: filter criteria for work pools\n        work_queue_filter: filter criteria for work pool queues\n        limit: a limit for the deployment query\n        offset: an offset for the deployment query\n\n    Returns:\n        a list of Deployment model representations\n            of the deployments\n    \"\"\"\n    body = {\n        \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n            if flow_run_filter\n            else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"deployments\": (\n            deployment_filter.dict(json_compatible=True)\n            if deployment_filter\n            else None\n        ),\n        \"work_pools\": (\n            work_pool_filter.dict(json_compatible=True)\n            if work_pool_filter\n            else None\n        ),\n        \"work_pool_queues\": (\n            work_queue_filter.dict(json_compatible=True)\n            if work_queue_filter\n            else None\n        ),\n        \"limit\": limit,\n        \"offset\": offset,\n        \"sort\": sort,\n    }\n\n    response = await self._client.post(\"/deployments/filter\", json=body)\n    return pydantic.parse_obj_as(List[DeploymentResponse], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_deployment","title":"delete_deployment async","text":"

    Delete deployment by id.

    Parameters:

    Name Type Description Default deployment_id UUID

    The deployment id of interest.

    required Source code in prefect/client/orchestration.py
    async def delete_deployment(\n    self,\n    deployment_id: UUID,\n):\n    \"\"\"\n    Delete deployment by id.\n\n    Args:\n        deployment_id: The deployment id of interest.\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If requests fails\n    \"\"\"\n    try:\n        await self._client.delete(f\"/deployments/{deployment_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_run","title":"read_flow_run async","text":"

    Query the Prefect API for a flow run by id.

    Parameters:

    Name Type Description Default flow_run_id UUID

    the flow run ID of interest

    required

    Returns:

    Type Description FlowRun

    a Flow Run model representation of the flow run

    Source code in prefect/client/orchestration.py
    async def read_flow_run(self, flow_run_id: UUID) -> FlowRun:\n    \"\"\"\n    Query the Prefect API for a flow run by id.\n\n    Args:\n        flow_run_id: the flow run ID of interest\n\n    Returns:\n        a Flow Run model representation of the flow run\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/flow_runs/{flow_run_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return FlowRun.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.resume_flow_run","title":"resume_flow_run async","text":"

    Resumes a paused flow run.

    Parameters:

    Name Type Description Default flow_run_id UUID

    the flow run ID of interest

    required run_input Optional[Dict]

    the input to resume the flow run with

    None

    Returns:

    Type Description OrchestrationResult

    an OrchestrationResult model representation of state orchestration output

    Source code in prefect/client/orchestration.py
    async def resume_flow_run(\n    self, flow_run_id: UUID, run_input: Optional[Dict] = None\n) -> OrchestrationResult:\n    \"\"\"\n    Resumes a paused flow run.\n\n    Args:\n        flow_run_id: the flow run ID of interest\n        run_input: the input to resume the flow run with\n\n    Returns:\n        an OrchestrationResult model representation of state orchestration output\n    \"\"\"\n    try:\n        response = await self._client.post(\n            f\"/flow_runs/{flow_run_id}/resume\", json={\"run_input\": run_input}\n        )\n    except httpx.HTTPStatusError:\n        raise\n\n    return OrchestrationResult.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_runs","title":"read_flow_runs async","text":"

    Query the Prefect API for flow runs. Only flow runs matching all criteria will be returned.

    Parameters:

    Name Type Description Default flow_filter FlowFilter

    filter criteria for flows

    None flow_run_filter FlowRunFilter

    filter criteria for flow runs

    None task_run_filter TaskRunFilter

    filter criteria for task runs

    None deployment_filter DeploymentFilter

    filter criteria for deployments

    None work_pool_filter WorkPoolFilter

    filter criteria for work pools

    None work_queue_filter WorkQueueFilter

    filter criteria for work pool queues

    None sort FlowRunSort

    sort criteria for the flow runs

    None limit int

    limit for the flow run query

    None offset int

    offset for the flow run query

    0

    Returns:

    Type Description List[FlowRun]

    a list of Flow Run model representations of the flow runs

    Source code in prefect/client/orchestration.py
    async def read_flow_runs(\n    self,\n    *,\n    flow_filter: FlowFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    deployment_filter: DeploymentFilter = None,\n    work_pool_filter: WorkPoolFilter = None,\n    work_queue_filter: WorkQueueFilter = None,\n    sort: FlowRunSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[FlowRun]:\n    \"\"\"\n    Query the Prefect API for flow runs. Only flow runs matching all criteria will\n    be returned.\n\n    Args:\n        flow_filter: filter criteria for flows\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        deployment_filter: filter criteria for deployments\n        work_pool_filter: filter criteria for work pools\n        work_queue_filter: filter criteria for work pool queues\n        sort: sort criteria for the flow runs\n        limit: limit for the flow run query\n        offset: offset for the flow run query\n\n    Returns:\n        a list of Flow Run model representations\n            of the flow runs\n    \"\"\"\n    body = {\n        \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n            if flow_run_filter\n            else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"deployments\": (\n            deployment_filter.dict(json_compatible=True)\n            if deployment_filter\n            else None\n        ),\n        \"work_pools\": (\n            work_pool_filter.dict(json_compatible=True)\n            if work_pool_filter\n            else None\n        ),\n        \"work_pool_queues\": (\n            work_queue_filter.dict(json_compatible=True)\n            if work_queue_filter\n            else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n\n    response = await self._client.post(\"/flow_runs/filter\", json=body)\n    return pydantic.parse_obj_as(List[FlowRun], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.set_flow_run_state","title":"set_flow_run_state async","text":"

    Set the state of a flow run.

    Parameters:

    Name Type Description Default flow_run_id UUID

    the id of the flow run

    required state State

    the state to set

    required force bool

    if True, disregard orchestration logic when setting the state, forcing the Prefect API to accept the state

    False

    Returns:

    Type Description OrchestrationResult

    an OrchestrationResult model representation of state orchestration output

    Source code in prefect/client/orchestration.py
    async def set_flow_run_state(\n    self,\n    flow_run_id: UUID,\n    state: \"prefect.states.State\",\n    force: bool = False,\n) -> OrchestrationResult:\n    \"\"\"\n    Set the state of a flow run.\n\n    Args:\n        flow_run_id: the id of the flow run\n        state: the state to set\n        force: if True, disregard orchestration logic when setting the state,\n            forcing the Prefect API to accept the state\n\n    Returns:\n        an OrchestrationResult model representation of state orchestration output\n    \"\"\"\n    state_create = state.to_state_create()\n    state_create.state_details.flow_run_id = flow_run_id\n    try:\n        response = await self._client.post(\n            f\"/flow_runs/{flow_run_id}/set_state\",\n            json=dict(state=state_create.dict(json_compatible=True), force=force),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n\n    return OrchestrationResult.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_run_states","title":"read_flow_run_states async","text":"

    Query for the states of a flow run

    Parameters:

    Name Type Description Default flow_run_id UUID

    the id of the flow run

    required

    Returns:

    Type Description List[State]

    a list of State model representations of the flow run states

    Source code in prefect/client/orchestration.py
    async def read_flow_run_states(\n    self, flow_run_id: UUID\n) -> List[prefect.states.State]:\n    \"\"\"\n    Query for the states of a flow run\n\n    Args:\n        flow_run_id: the id of the flow run\n\n    Returns:\n        a list of State model representations\n            of the flow run states\n    \"\"\"\n    response = await self._client.get(\n        \"/flow_run_states/\", params=dict(flow_run_id=str(flow_run_id))\n    )\n    return pydantic.parse_obj_as(List[prefect.states.State], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_task_run","title":"create_task_run async","text":"

    Create a task run

    Parameters:

    Name Type Description Default task Task

    The Task to run

    required flow_run_id UUID

    The flow run id with which to associate the task run

    required dynamic_key str

    A key unique to this particular run of a Task within the flow

    required name str

    An optional name for the task run

    None extra_tags Iterable[str]

    an optional list of extra tags to apply to the task run in addition to task.tags

    None state State

    The initial state for the run. If not provided, defaults to Pending for now. Should always be a Scheduled type.

    None task_inputs Dict[str, List[Union[TaskRunResult, Parameter, Constant]]]

    the set of inputs passed to the task

    None

    Returns:

    Type Description TaskRun

    The created task run.

    Source code in prefect/client/orchestration.py
    async def create_task_run(\n    self,\n    task: \"TaskObject\",\n    flow_run_id: UUID,\n    dynamic_key: str,\n    name: str = None,\n    extra_tags: Iterable[str] = None,\n    state: prefect.states.State = None,\n    task_inputs: Dict[\n        str,\n        List[\n            Union[\n                TaskRunResult,\n                Parameter,\n                Constant,\n            ]\n        ],\n    ] = None,\n) -> TaskRun:\n    \"\"\"\n    Create a task run\n\n    Args:\n        task: The Task to run\n        flow_run_id: The flow run id with which to associate the task run\n        dynamic_key: A key unique to this particular run of a Task within the flow\n        name: An optional name for the task run\n        extra_tags: an optional list of extra tags to apply to the task run in\n            addition to `task.tags`\n        state: The initial state for the run. If not provided, defaults to\n            `Pending` for now. Should always be a `Scheduled` type.\n        task_inputs: the set of inputs passed to the task\n\n    Returns:\n        The created task run.\n    \"\"\"\n    tags = set(task.tags).union(extra_tags or [])\n\n    if state is None:\n        state = prefect.states.Pending()\n\n    task_run_data = TaskRunCreate(\n        name=name,\n        flow_run_id=flow_run_id,\n        task_key=task.task_key,\n        dynamic_key=dynamic_key,\n        tags=list(tags),\n        task_version=task.version,\n        empirical_policy=TaskRunPolicy(\n            retries=task.retries,\n            retry_delay=task.retry_delay_seconds,\n            retry_jitter_factor=task.retry_jitter_factor,\n        ),\n        state=state.to_state_create(),\n        task_inputs=task_inputs or {},\n    )\n\n    response = await self._client.post(\n        \"/task_runs/\", json=task_run_data.dict(json_compatible=True)\n    )\n    return TaskRun.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_task_run","title":"read_task_run async","text":"

    Query the Prefect API for a task run by id.

    Parameters:

    Name Type Description Default task_run_id UUID

    the task run ID of interest

    required

    Returns:

    Type Description TaskRun

    a Task Run model representation of the task run

    Source code in prefect/client/orchestration.py
    async def read_task_run(self, task_run_id: UUID) -> TaskRun:\n    \"\"\"\n    Query the Prefect API for a task run by id.\n\n    Args:\n        task_run_id: the task run ID of interest\n\n    Returns:\n        a Task Run model representation of the task run\n    \"\"\"\n    response = await self._client.get(f\"/task_runs/{task_run_id}\")\n    return TaskRun.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_task_runs","title":"read_task_runs async","text":"

    Query the Prefect API for task runs. Only task runs matching all criteria will be returned.

    Parameters:

    Name Type Description Default flow_filter FlowFilter

    filter criteria for flows

    None flow_run_filter FlowRunFilter

    filter criteria for flow runs

    None task_run_filter TaskRunFilter

    filter criteria for task runs

    None deployment_filter DeploymentFilter

    filter criteria for deployments

    None sort TaskRunSort

    sort criteria for the task runs

    None limit int

    a limit for the task run query

    None offset int

    an offset for the task run query

    0

    Returns:

    Type Description List[TaskRun]

    a list of Task Run model representations of the task runs

    Source code in prefect/client/orchestration.py
    async def read_task_runs(\n    self,\n    *,\n    flow_filter: FlowFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    deployment_filter: DeploymentFilter = None,\n    sort: TaskRunSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[TaskRun]:\n    \"\"\"\n    Query the Prefect API for task runs. Only task runs matching all criteria will\n    be returned.\n\n    Args:\n        flow_filter: filter criteria for flows\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        deployment_filter: filter criteria for deployments\n        sort: sort criteria for the task runs\n        limit: a limit for the task run query\n        offset: an offset for the task run query\n\n    Returns:\n        a list of Task Run model representations\n            of the task runs\n    \"\"\"\n    body = {\n        \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n            if flow_run_filter\n            else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"deployments\": (\n            deployment_filter.dict(json_compatible=True)\n            if deployment_filter\n            else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n    response = await self._client.post(\"/task_runs/filter\", json=body)\n    return pydantic.parse_obj_as(List[TaskRun], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.set_task_run_state","title":"set_task_run_state async","text":"

    Set the state of a task run.

    Parameters:

    Name Type Description Default task_run_id UUID

    the id of the task run

    required state State

    the state to set

    required force bool

    if True, disregard orchestration logic when setting the state, forcing the Prefect API to accept the state

    False

    Returns:

    Type Description OrchestrationResult

    an OrchestrationResult model representation of state orchestration output

    Source code in prefect/client/orchestration.py
    async def set_task_run_state(\n    self,\n    task_run_id: UUID,\n    state: prefect.states.State,\n    force: bool = False,\n) -> OrchestrationResult:\n    \"\"\"\n    Set the state of a task run.\n\n    Args:\n        task_run_id: the id of the task run\n        state: the state to set\n        force: if True, disregard orchestration logic when setting the state,\n            forcing the Prefect API to accept the state\n\n    Returns:\n        an OrchestrationResult model representation of state orchestration output\n    \"\"\"\n    state_create = state.to_state_create()\n    state_create.state_details.task_run_id = task_run_id\n    response = await self._client.post(\n        f\"/task_runs/{task_run_id}/set_state\",\n        json=dict(state=state_create.dict(json_compatible=True), force=force),\n    )\n    return OrchestrationResult.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_task_run_states","title":"read_task_run_states async","text":"

    Query for the states of a task run

    Parameters:

    Name Type Description Default task_run_id UUID

    the id of the task run

    required

    Returns:

    Type Description List[State]

    a list of State model representations of the task run states

    Source code in prefect/client/orchestration.py
    async def read_task_run_states(\n    self, task_run_id: UUID\n) -> List[prefect.states.State]:\n    \"\"\"\n    Query for the states of a task run\n\n    Args:\n        task_run_id: the id of the task run\n\n    Returns:\n        a list of State model representations of the task run states\n    \"\"\"\n    response = await self._client.get(\n        \"/task_run_states/\", params=dict(task_run_id=str(task_run_id))\n    )\n    return pydantic.parse_obj_as(List[prefect.states.State], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_logs","title":"create_logs async","text":"

    Create logs for a flow or task run

    Parameters:

    Name Type Description Default logs Iterable[Union[LogCreate, dict]]

    An iterable of LogCreate objects or already json-compatible dicts

    required Source code in prefect/client/orchestration.py
    async def create_logs(self, logs: Iterable[Union[LogCreate, dict]]) -> None:\n    \"\"\"\n    Create logs for a flow or task run\n\n    Args:\n        logs: An iterable of `LogCreate` objects or already json-compatible dicts\n    \"\"\"\n    serialized_logs = [\n        log.dict(json_compatible=True) if isinstance(log, LogCreate) else log\n        for log in logs\n    ]\n    await self._client.post(\"/logs/\", json=serialized_logs)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_run_notification_policy","title":"create_flow_run_notification_policy async","text":"

    Create a notification policy for flow runs

    Parameters:

    Name Type Description Default block_document_id UUID

    The block document UUID

    required is_active bool

    Whether the notification policy is active

    True tags List[str]

    List of flow tags

    None state_names List[str]

    List of state names

    None message_template Optional[str]

    Notification message template

    None Source code in prefect/client/orchestration.py
    async def create_flow_run_notification_policy(\n    self,\n    block_document_id: UUID,\n    is_active: bool = True,\n    tags: List[str] = None,\n    state_names: List[str] = None,\n    message_template: Optional[str] = None,\n) -> UUID:\n    \"\"\"\n    Create a notification policy for flow runs\n\n    Args:\n        block_document_id: The block document UUID\n        is_active: Whether the notification policy is active\n        tags: List of flow tags\n        state_names: List of state names\n        message_template: Notification message template\n    \"\"\"\n    if tags is None:\n        tags = []\n    if state_names is None:\n        state_names = []\n\n    policy = FlowRunNotificationPolicyCreate(\n        block_document_id=block_document_id,\n        is_active=is_active,\n        tags=tags,\n        state_names=state_names,\n        message_template=message_template,\n    )\n    response = await self._client.post(\n        \"/flow_run_notification_policies/\",\n        json=policy.dict(json_compatible=True),\n    )\n\n    policy_id = response.json().get(\"id\")\n    if not policy_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    return UUID(policy_id)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_run_notification_policies","title":"read_flow_run_notification_policies async","text":"

    Query the Prefect API for flow run notification policies. Only policies matching all criteria will be returned.

    Parameters:

    Name Type Description Default flow_run_notification_policy_filter FlowRunNotificationPolicyFilter

    filter criteria for notification policies

    required limit Optional[int]

    a limit for the notification policies query

    None offset int

    an offset for the notification policies query

    0

    Returns:

    Type Description List[FlowRunNotificationPolicy]

    a list of FlowRunNotificationPolicy model representations of the notification policies

    Source code in prefect/client/orchestration.py
    async def read_flow_run_notification_policies(\n    self,\n    flow_run_notification_policy_filter: FlowRunNotificationPolicyFilter,\n    limit: Optional[int] = None,\n    offset: int = 0,\n) -> List[FlowRunNotificationPolicy]:\n    \"\"\"\n    Query the Prefect API for flow run notification policies. Only policies matching all criteria will\n    be returned.\n\n    Args:\n        flow_run_notification_policy_filter: filter criteria for notification policies\n        limit: a limit for the notification policies query\n        offset: an offset for the notification policies query\n\n    Returns:\n        a list of FlowRunNotificationPolicy model representations\n            of the notification policies\n    \"\"\"\n    body = {\n        \"flow_run_notification_policy_filter\": (\n            flow_run_notification_policy_filter.dict(json_compatible=True)\n            if flow_run_notification_policy_filter\n            else None\n        ),\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n    response = await self._client.post(\n        \"/flow_run_notification_policies/filter\", json=body\n    )\n    return pydantic.parse_obj_as(List[FlowRunNotificationPolicy], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_logs","title":"read_logs async","text":"

    Read flow and task run logs.

    Source code in prefect/client/orchestration.py
    async def read_logs(\n    self,\n    log_filter: LogFilter = None,\n    limit: int = None,\n    offset: int = None,\n    sort: LogSort = LogSort.TIMESTAMP_ASC,\n) -> List[Log]:\n    \"\"\"\n    Read flow and task run logs.\n    \"\"\"\n    body = {\n        \"logs\": log_filter.dict(json_compatible=True) if log_filter else None,\n        \"limit\": limit,\n        \"offset\": offset,\n        \"sort\": sort,\n    }\n\n    response = await self._client.post(\"/logs/filter\", json=body)\n    return pydantic.parse_obj_as(List[Log], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.resolve_datadoc","title":"resolve_datadoc async","text":"

    Recursively decode possibly nested data documents.

    \"server\" encoded documents will be retrieved from the server.

    Parameters:

    Name Type Description Default datadoc DataDocument

    The data document to resolve

    required

    Returns:

    Type Description Any

    a decoded object, the innermost data

    Source code in prefect/client/orchestration.py
    async def resolve_datadoc(self, datadoc: DataDocument) -> Any:\n    \"\"\"\n    Recursively decode possibly nested data documents.\n\n    \"server\" encoded documents will be retrieved from the server.\n\n    Args:\n        datadoc: The data document to resolve\n\n    Returns:\n        a decoded object, the innermost data\n    \"\"\"\n    if not isinstance(datadoc, DataDocument):\n        raise TypeError(\n            f\"`resolve_datadoc` received invalid type {type(datadoc).__name__}\"\n        )\n\n    async def resolve_inner(data):\n        if isinstance(data, bytes):\n            try:\n                data = DataDocument.parse_raw(data)\n            except pydantic.ValidationError:\n                return data\n\n        if isinstance(data, DataDocument):\n            return await resolve_inner(data.decode())\n\n        return data\n\n    return await resolve_inner(datadoc)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.send_worker_heartbeat","title":"send_worker_heartbeat async","text":"

    Sends a worker heartbeat for a given work pool.

    Parameters:

    Name Type Description Default work_pool_name str

    The name of the work pool to heartbeat against.

    required worker_name str

    The name of the worker sending the heartbeat.

    required Source code in prefect/client/orchestration.py
    async def send_worker_heartbeat(\n    self,\n    work_pool_name: str,\n    worker_name: str,\n    heartbeat_interval_seconds: Optional[float] = None,\n):\n    \"\"\"\n    Sends a worker heartbeat for a given work pool.\n\n    Args:\n        work_pool_name: The name of the work pool to heartbeat against.\n        worker_name: The name of the worker sending the heartbeat.\n    \"\"\"\n    await self._client.post(\n        f\"/work_pools/{work_pool_name}/workers/heartbeat\",\n        json={\n            \"name\": worker_name,\n            \"heartbeat_interval_seconds\": heartbeat_interval_seconds,\n        },\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_workers_for_work_pool","title":"read_workers_for_work_pool async","text":"

    Reads workers for a given work pool.

    Parameters:

    Name Type Description Default work_pool_name str

    The name of the work pool for which to get member workers.

    required worker_filter Optional[WorkerFilter]

    Criteria by which to filter workers.

    None limit Optional[int]

    Limit for the worker query.

    None offset Optional[int]

    Limit for the worker query.

    None Source code in prefect/client/orchestration.py
    async def read_workers_for_work_pool(\n    self,\n    work_pool_name: str,\n    worker_filter: Optional[WorkerFilter] = None,\n    offset: Optional[int] = None,\n    limit: Optional[int] = None,\n) -> List[Worker]:\n    \"\"\"\n    Reads workers for a given work pool.\n\n    Args:\n        work_pool_name: The name of the work pool for which to get\n            member workers.\n        worker_filter: Criteria by which to filter workers.\n        limit: Limit for the worker query.\n        offset: Limit for the worker query.\n    \"\"\"\n    response = await self._client.post(\n        f\"/work_pools/{work_pool_name}/workers/filter\",\n        json={\n            \"worker_filter\": (\n                worker_filter.dict(json_compatible=True, exclude_unset=True)\n                if worker_filter\n                else None\n            ),\n            \"offset\": offset,\n            \"limit\": limit,\n        },\n    )\n\n    return pydantic.parse_obj_as(List[Worker], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_pool","title":"read_work_pool async","text":"

    Reads information for a given work pool

    Parameters:

    Name Type Description Default work_pool_name str

    The name of the work pool to for which to get information.

    required

    Returns:

    Type Description WorkPool

    Information about the requested work pool.

    Source code in prefect/client/orchestration.py
    async def read_work_pool(self, work_pool_name: str) -> WorkPool:\n    \"\"\"\n    Reads information for a given work pool\n\n    Args:\n        work_pool_name: The name of the work pool to for which to get\n            information.\n\n    Returns:\n        Information about the requested work pool.\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/work_pools/{work_pool_name}\")\n        return pydantic.parse_obj_as(WorkPool, response.json())\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_pools","title":"read_work_pools async","text":"

    Reads work pools.

    Parameters:

    Name Type Description Default limit Optional[int]

    Limit for the work pool query.

    None offset int

    Offset for the work pool query.

    0 work_pool_filter Optional[WorkPoolFilter]

    Criteria by which to filter work pools.

    None

    Returns:

    Type Description List[WorkPool]

    A list of work pools.

    Source code in prefect/client/orchestration.py
    async def read_work_pools(\n    self,\n    limit: Optional[int] = None,\n    offset: int = 0,\n    work_pool_filter: Optional[WorkPoolFilter] = None,\n) -> List[WorkPool]:\n    \"\"\"\n    Reads work pools.\n\n    Args:\n        limit: Limit for the work pool query.\n        offset: Offset for the work pool query.\n        work_pool_filter: Criteria by which to filter work pools.\n\n    Returns:\n        A list of work pools.\n    \"\"\"\n\n    body = {\n        \"limit\": limit,\n        \"offset\": offset,\n        \"work_pools\": (\n            work_pool_filter.dict(json_compatible=True)\n            if work_pool_filter\n            else None\n        ),\n    }\n    response = await self._client.post(\"/work_pools/filter\", json=body)\n    return pydantic.parse_obj_as(List[WorkPool], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_work_pool","title":"create_work_pool async","text":"

    Creates a work pool with the provided configuration.

    Parameters:

    Name Type Description Default work_pool WorkPoolCreate

    Desired configuration for the new work pool.

    required

    Returns:

    Type Description WorkPool

    Information about the newly created work pool.

    Source code in prefect/client/orchestration.py
    async def create_work_pool(\n    self,\n    work_pool: WorkPoolCreate,\n) -> WorkPool:\n    \"\"\"\n    Creates a work pool with the provided configuration.\n\n    Args:\n        work_pool: Desired configuration for the new work pool.\n\n    Returns:\n        Information about the newly created work pool.\n    \"\"\"\n    try:\n        response = await self._client.post(\n            \"/work_pools/\",\n            json=work_pool.dict(json_compatible=True, exclude_unset=True),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        else:\n            raise\n\n    return pydantic.parse_obj_as(WorkPool, response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_work_pool","title":"update_work_pool async","text":"

    Updates a work pool.

    Parameters:

    Name Type Description Default work_pool_name str

    Name of the work pool to update.

    required work_pool WorkPoolUpdate

    Fields to update in the work pool.

    required Source code in prefect/client/orchestration.py
    async def update_work_pool(\n    self,\n    work_pool_name: str,\n    work_pool: WorkPoolUpdate,\n):\n    \"\"\"\n    Updates a work pool.\n\n    Args:\n        work_pool_name: Name of the work pool to update.\n        work_pool: Fields to update in the work pool.\n    \"\"\"\n    try:\n        await self._client.patch(\n            f\"/work_pools/{work_pool_name}\",\n            json=work_pool.dict(json_compatible=True, exclude_unset=True),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_work_pool","title":"delete_work_pool async","text":"

    Deletes a work pool.

    Parameters:

    Name Type Description Default work_pool_name str

    Name of the work pool to delete.

    required Source code in prefect/client/orchestration.py
    async def delete_work_pool(\n    self,\n    work_pool_name: str,\n):\n    \"\"\"\n    Deletes a work pool.\n\n    Args:\n        work_pool_name: Name of the work pool to delete.\n    \"\"\"\n    try:\n        await self._client.delete(f\"/work_pools/{work_pool_name}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_queues","title":"read_work_queues async","text":"

    Retrieves queues for a work pool.

    Parameters:

    Name Type Description Default work_pool_name Optional[str]

    Name of the work pool for which to get queues.

    None work_queue_filter Optional[WorkQueueFilter]

    Criteria by which to filter queues.

    None limit Optional[int]

    Limit for the queue query.

    None offset Optional[int]

    Limit for the queue query.

    None

    Returns:

    Type Description List[WorkQueue]

    List of queues for the specified work pool.

    Source code in prefect/client/orchestration.py
    async def read_work_queues(\n    self,\n    work_pool_name: Optional[str] = None,\n    work_queue_filter: Optional[WorkQueueFilter] = None,\n    limit: Optional[int] = None,\n    offset: Optional[int] = None,\n) -> List[WorkQueue]:\n    \"\"\"\n    Retrieves queues for a work pool.\n\n    Args:\n        work_pool_name: Name of the work pool for which to get queues.\n        work_queue_filter: Criteria by which to filter queues.\n        limit: Limit for the queue query.\n        offset: Limit for the queue query.\n\n    Returns:\n        List of queues for the specified work pool.\n    \"\"\"\n    json = {\n        \"work_queues\": (\n            work_queue_filter.dict(json_compatible=True, exclude_unset=True)\n            if work_queue_filter\n            else None\n        ),\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n\n    if work_pool_name:\n        try:\n            response = await self._client.post(\n                f\"/work_pools/{work_pool_name}/queues/filter\",\n                json=json,\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n    else:\n        response = await self._client.post(\"/work_queues/filter\", json=json)\n\n    return pydantic.parse_obj_as(List[WorkQueue], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.get_scheduled_flow_runs_for_work_pool","title":"get_scheduled_flow_runs_for_work_pool async","text":"

    Retrieves scheduled flow runs for the provided set of work pool queues.

    Parameters:

    Name Type Description Default work_pool_name str

    The name of the work pool that the work pool queues are associated with.

    required work_queue_names Optional[List[str]]

    The names of the work pool queues from which to get scheduled flow runs.

    None scheduled_before Optional[datetime]

    Datetime used to filter returned flow runs. Flow runs scheduled for after the given datetime string will not be returned.

    None

    Returns:

    Type Description List[WorkerFlowRunResponse]

    A list of worker flow run responses containing information about the

    List[WorkerFlowRunResponse]

    retrieved flow runs.

    Source code in prefect/client/orchestration.py
    async def get_scheduled_flow_runs_for_work_pool(\n    self,\n    work_pool_name: str,\n    work_queue_names: Optional[List[str]] = None,\n    scheduled_before: Optional[datetime.datetime] = None,\n) -> List[WorkerFlowRunResponse]:\n    \"\"\"\n    Retrieves scheduled flow runs for the provided set of work pool queues.\n\n    Args:\n        work_pool_name: The name of the work pool that the work pool\n            queues are associated with.\n        work_queue_names: The names of the work pool queues from which\n            to get scheduled flow runs.\n        scheduled_before: Datetime used to filter returned flow runs. Flow runs\n            scheduled for after the given datetime string will not be returned.\n\n    Returns:\n        A list of worker flow run responses containing information about the\n        retrieved flow runs.\n    \"\"\"\n    body: Dict[str, Any] = {}\n    if work_queue_names is not None:\n        body[\"work_queue_names\"] = list(work_queue_names)\n    if scheduled_before:\n        body[\"scheduled_before\"] = str(scheduled_before)\n\n    response = await self._client.post(\n        f\"/work_pools/{work_pool_name}/get_scheduled_flow_runs\",\n        json=body,\n    )\n    return pydantic.parse_obj_as(List[WorkerFlowRunResponse], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_artifact","title":"create_artifact async","text":"

    Creates an artifact with the provided configuration.

    Parameters:

    Name Type Description Default artifact ArtifactCreate

    Desired configuration for the new artifact.

    required Source code in prefect/client/orchestration.py
    async def create_artifact(\n    self,\n    artifact: ArtifactCreate,\n) -> Artifact:\n    \"\"\"\n    Creates an artifact with the provided configuration.\n\n    Args:\n        artifact: Desired configuration for the new artifact.\n    Returns:\n        Information about the newly created artifact.\n    \"\"\"\n\n    response = await self._client.post(\n        \"/artifacts/\",\n        json=artifact.dict(json_compatible=True, exclude_unset=True),\n    )\n\n    return pydantic.parse_obj_as(Artifact, response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_artifacts","title":"read_artifacts async","text":"

    Query the Prefect API for artifacts. Only artifacts matching all criteria will be returned. Args: artifact_filter: filter criteria for artifacts flow_run_filter: filter criteria for flow runs task_run_filter: filter criteria for task runs sort: sort criteria for the artifacts limit: limit for the artifact query offset: offset for the artifact query Returns: a list of Artifact model representations of the artifacts

    Source code in prefect/client/orchestration.py
    async def read_artifacts(\n    self,\n    *,\n    artifact_filter: ArtifactFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    sort: ArtifactSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[Artifact]:\n    \"\"\"\n    Query the Prefect API for artifacts. Only artifacts matching all criteria will\n    be returned.\n    Args:\n        artifact_filter: filter criteria for artifacts\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        sort: sort criteria for the artifacts\n        limit: limit for the artifact query\n        offset: offset for the artifact query\n    Returns:\n        a list of Artifact model representations of the artifacts\n    \"\"\"\n    body = {\n        \"artifacts\": (\n            artifact_filter.dict(json_compatible=True) if artifact_filter else None\n        ),\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True) if flow_run_filter else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n    response = await self._client.post(\"/artifacts/filter\", json=body)\n    return pydantic.parse_obj_as(List[Artifact], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_latest_artifacts","title":"read_latest_artifacts async","text":"

    Query the Prefect API for artifacts. Only artifacts matching all criteria will be returned. Args: artifact_filter: filter criteria for artifacts flow_run_filter: filter criteria for flow runs task_run_filter: filter criteria for task runs sort: sort criteria for the artifacts limit: limit for the artifact query offset: offset for the artifact query Returns: a list of Artifact model representations of the artifacts

    Source code in prefect/client/orchestration.py
    async def read_latest_artifacts(\n    self,\n    *,\n    artifact_filter: ArtifactCollectionFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    sort: ArtifactCollectionSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[ArtifactCollection]:\n    \"\"\"\n    Query the Prefect API for artifacts. Only artifacts matching all criteria will\n    be returned.\n    Args:\n        artifact_filter: filter criteria for artifacts\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        sort: sort criteria for the artifacts\n        limit: limit for the artifact query\n        offset: offset for the artifact query\n    Returns:\n        a list of Artifact model representations of the artifacts\n    \"\"\"\n    body = {\n        \"artifacts\": (\n            artifact_filter.dict(json_compatible=True) if artifact_filter else None\n        ),\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True) if flow_run_filter else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n    response = await self._client.post(\"/artifacts/latest/filter\", json=body)\n    return pydantic.parse_obj_as(List[ArtifactCollection], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_artifact","title":"delete_artifact async","text":"

    Deletes an artifact with the provided id.

    Parameters:

    Name Type Description Default artifact_id UUID

    The id of the artifact to delete.

    required Source code in prefect/client/orchestration.py
    async def delete_artifact(self, artifact_id: UUID) -> None:\n    \"\"\"\n    Deletes an artifact with the provided id.\n\n    Args:\n        artifact_id: The id of the artifact to delete.\n    \"\"\"\n    try:\n        await self._client.delete(f\"/artifacts/{artifact_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_variable_by_name","title":"read_variable_by_name async","text":"

    Reads a variable by name. Returns None if no variable is found.

    Source code in prefect/client/orchestration.py
    async def read_variable_by_name(self, name: str) -> Optional[Variable]:\n    \"\"\"Reads a variable by name. Returns None if no variable is found.\"\"\"\n    try:\n        response = await self._client.get(f\"/variables/name/{name}\")\n        return pydantic.parse_obj_as(Variable, response.json())\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            return None\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_variable_by_name","title":"delete_variable_by_name async","text":"

    Deletes a variable by name.

    Source code in prefect/client/orchestration.py
    async def delete_variable_by_name(self, name: str):\n    \"\"\"Deletes a variable by name.\"\"\"\n    try:\n        await self._client.delete(f\"/variables/name/{name}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_variables","title":"read_variables async","text":"

    Reads all variables.

    Source code in prefect/client/orchestration.py
    async def read_variables(self, limit: int = None) -> List[Variable]:\n    \"\"\"Reads all variables.\"\"\"\n    response = await self._client.post(\"/variables/filter\", json={\"limit\": limit})\n    return pydantic.parse_obj_as(List[Variable], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_worker_metadata","title":"read_worker_metadata async","text":"

    Reads worker metadata stored in Prefect collection registry.

    Source code in prefect/client/orchestration.py
    async def read_worker_metadata(self) -> Dict[str, Any]:\n    \"\"\"Reads worker metadata stored in Prefect collection registry.\"\"\"\n    response = await self._client.get(\"collections/views/aggregate-worker-metadata\")\n    response.raise_for_status()\n    return response.json()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_automation","title":"create_automation async","text":"

    Creates an automation in Prefect Cloud.

    Source code in prefect/client/orchestration.py
    async def create_automation(self, automation: Automation) -> UUID:\n    \"\"\"Creates an automation in Prefect Cloud.\"\"\"\n    if self.server_type != ServerType.CLOUD:\n        raise RuntimeError(\"Automations are only supported for Prefect Cloud.\")\n\n    response = await self._client.post(\n        \"/automations/\",\n        json=automation.dict(json_compatible=True),\n    )\n\n    return UUID(response.json()[\"id\"])\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_run_input","title":"create_flow_run_input async","text":"

    Creates a flow run input.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The flow run id.

    required key str

    The input key.

    required value str

    The input value.

    required sender Optional[str]

    The sender of the input.

    None Source code in prefect/client/orchestration.py
    async def create_flow_run_input(\n    self, flow_run_id: UUID, key: str, value: str, sender: Optional[str] = None\n):\n    \"\"\"\n    Creates a flow run input.\n\n    Args:\n        flow_run_id: The flow run id.\n        key: The input key.\n        value: The input value.\n        sender: The sender of the input.\n    \"\"\"\n\n    # Initialize the input to ensure that the key is valid.\n    FlowRunInput(flow_run_id=flow_run_id, key=key, value=value)\n\n    response = await self._client.post(\n        f\"/flow_runs/{flow_run_id}/input\",\n        json={\"key\": key, \"value\": value, \"sender\": sender},\n    )\n    response.raise_for_status()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_run_input","title":"read_flow_run_input async","text":"

    Reads a flow run input.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The flow run id.

    required key str

    The input key.

    required Source code in prefect/client/orchestration.py
    async def read_flow_run_input(self, flow_run_id: UUID, key: str) -> str:\n    \"\"\"\n    Reads a flow run input.\n\n    Args:\n        flow_run_id: The flow run id.\n        key: The input key.\n    \"\"\"\n    response = await self._client.get(f\"/flow_runs/{flow_run_id}/input/{key}\")\n    response.raise_for_status()\n    return response.content.decode()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_flow_run_input","title":"delete_flow_run_input async","text":"

    Deletes a flow run input.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The flow run id.

    required key str

    The input key.

    required Source code in prefect/client/orchestration.py
    async def delete_flow_run_input(self, flow_run_id: UUID, key: str):\n    \"\"\"\n    Deletes a flow run input.\n\n    Args:\n        flow_run_id: The flow run id.\n        key: The input key.\n    \"\"\"\n    response = await self._client.delete(f\"/flow_runs/{flow_run_id}/input/{key}\")\n    response.raise_for_status()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.get_client","title":"get_client","text":"

    Retrieve a HTTP client for communicating with the Prefect REST API.

    The client must be context managed; for example:

    async with get_client() as client:\n    await client.hello()\n
    Source code in prefect/client/orchestration.py
    def get_client(httpx_settings: Optional[dict] = None) -> \"PrefectClient\":\n    \"\"\"\n    Retrieve a HTTP client for communicating with the Prefect REST API.\n\n    The client must be context managed; for example:\n\n    ```python\n    async with get_client() as client:\n        await client.hello()\n    ```\n    \"\"\"\n    ctx = prefect.context.get_settings_context()\n    api = PREFECT_API_URL.value()\n\n    if not api:\n        # create an ephemeral API if none was provided\n        from prefect.server.api.server import create_app\n\n        api = create_app(ctx.settings, ephemeral=True)\n\n    return PrefectClient(\n        api,\n        api_key=PREFECT_API_KEY.value(),\n        httpx_settings=httpx_settings,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_1","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions","title":"prefect.client.schemas.actions","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.StateCreate","title":"StateCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a new state.

    Source code in prefect/client/schemas/actions.py
    class StateCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a new state.\"\"\"\n\n    type: StateType\n    name: Optional[str] = Field(default=None)\n    message: Optional[str] = Field(default=None, example=\"Run started\")\n    state_details: StateDetails = Field(default_factory=StateDetails)\n    data: Union[\"BaseResult[R]\", \"DataDocument[R]\", Any] = Field(\n        default=None,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowCreate","title":"FlowCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow.\"\"\"\n\n    name: str = FieldFrom(objects.Flow)\n    tags: List[str] = FieldFrom(objects.Flow)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowUpdate","title":"FlowUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow.\"\"\"\n\n    tags: List[str] = FieldFrom(objects.Flow)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentCreate","title":"DeploymentCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a deployment.

    Source code in prefect/client/schemas/actions.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None,\n)\n@copy_model_fields\nclass DeploymentCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a deployment.\"\"\"\n\n    @root_validator(pre=True)\n    def remove_old_fields(cls, values):\n        # 2.7.7 removed worker_pool_queue_id in lieu of worker_pool_name and\n        # worker_pool_queue_name. Those fields were later renamed to work_pool_name\n        # and work_queue_name. This validator removes old fields provided\n        # by older clients to avoid 422 errors.\n        values_copy = copy(values)\n        worker_pool_queue_id = values_copy.pop(\"worker_pool_queue_id\", None)\n        worker_pool_name = values_copy.pop(\"worker_pool_name\", None)\n        worker_pool_queue_name = values_copy.pop(\"worker_pool_queue_name\", None)\n        work_pool_queue_name = values_copy.pop(\"work_pool_queue_name\", None)\n        if worker_pool_queue_id:\n            warnings.warn(\n                (\n                    \"`worker_pool_queue_id` is no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        if worker_pool_name or worker_pool_queue_name or work_pool_queue_name:\n            warnings.warn(\n                (\n                    \"`worker_pool_name`, `worker_pool_queue_name`, and \"\n                    \"`work_pool_name` are\"\n                    \"no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        return values_copy\n\n    name: str = FieldFrom(objects.Deployment)\n    flow_id: UUID = FieldFrom(objects.Deployment)\n    is_schedule_active: Optional[bool] = FieldFrom(objects.Deployment)\n    enforce_parameter_schema: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n    parameter_openapi_schema: Optional[Dict[str, Any]] = FieldFrom(objects.Deployment)\n    parameters: Dict[str, Any] = FieldFrom(objects.Deployment)\n    tags: List[str] = FieldFrom(objects.Deployment)\n    pull_steps: Optional[List[dict]] = FieldFrom(objects.Deployment)\n\n    manifest_path: Optional[str] = FieldFrom(objects.Deployment)\n    work_queue_name: Optional[str] = FieldFrom(objects.Deployment)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the deployment's work pool.\",\n        example=\"my-work-pool\",\n    )\n    storage_document_id: Optional[UUID] = FieldFrom(objects.Deployment)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.Deployment)\n    schedule: Optional[SCHEDULE_TYPES] = FieldFrom(objects.Deployment)\n    description: Optional[str] = FieldFrom(objects.Deployment)\n    path: Optional[str] = FieldFrom(objects.Deployment)\n    version: Optional[str] = FieldFrom(objects.Deployment)\n    entrypoint: Optional[str] = FieldFrom(objects.Deployment)\n    infra_overrides: Optional[Dict[str, Any]] = FieldFrom(objects.Deployment)\n\n    def check_valid_configuration(self, base_job_template: dict):\n        \"\"\"Check that the combination of base_job_template defaults\n        and infra_overrides conforms to the specified schema.\n        \"\"\"\n        variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n        if variables_schema is not None:\n            # jsonschema considers required fields, even if that field has a default,\n            # to still be required. To get around this we remove the fields from\n            # required if there is a default present.\n            required = variables_schema.get(\"required\")\n            properties = variables_schema.get(\"properties\")\n            if required is not None and properties is not None:\n                for k, v in properties.items():\n                    if \"default\" in v and k in required:\n                        required.remove(k)\n\n            jsonschema.validate(self.infra_overrides, variables_schema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentCreate.check_valid_configuration","title":"check_valid_configuration","text":"

    Check that the combination of base_job_template defaults and infra_overrides conforms to the specified schema.

    Source code in prefect/client/schemas/actions.py
    def check_valid_configuration(self, base_job_template: dict):\n    \"\"\"Check that the combination of base_job_template defaults\n    and infra_overrides conforms to the specified schema.\n    \"\"\"\n    variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n    if variables_schema is not None:\n        # jsonschema considers required fields, even if that field has a default,\n        # to still be required. To get around this we remove the fields from\n        # required if there is a default present.\n        required = variables_schema.get(\"required\")\n        properties = variables_schema.get(\"properties\")\n        if required is not None and properties is not None:\n            for k, v in properties.items():\n                if \"default\" in v and k in required:\n                    required.remove(k)\n\n        jsonschema.validate(self.infra_overrides, variables_schema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentUpdate","title":"DeploymentUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a deployment.

    Source code in prefect/client/schemas/actions.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None,\n)\n@copy_model_fields\nclass DeploymentUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a deployment.\"\"\"\n\n    @root_validator(pre=True)\n    def remove_old_fields(cls, values):\n        # 2.7.7 removed worker_pool_queue_id in lieu of worker_pool_name and\n        # worker_pool_queue_name. Those fields were later renamed to work_pool_name\n        # and work_queue_name. This validator removes old fields provided\n        # by older clients to avoid 422 errors.\n        values_copy = copy(values)\n        worker_pool_queue_id = values_copy.pop(\"worker_pool_queue_id\", None)\n        worker_pool_name = values_copy.pop(\"worker_pool_name\", None)\n        worker_pool_queue_name = values_copy.pop(\"worker_pool_queue_name\", None)\n        work_pool_queue_name = values_copy.pop(\"work_pool_queue_name\", None)\n        if worker_pool_queue_id:\n            warnings.warn(\n                (\n                    \"`worker_pool_queue_id` is no longer supported for updating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        if worker_pool_name or worker_pool_queue_name or work_pool_queue_name:\n            warnings.warn(\n                (\n                    \"`worker_pool_name`, `worker_pool_queue_name`, and \"\n                    \"`work_pool_name` are\"\n                    \"no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        return values_copy\n\n    @validator(\"schedule\")\n    def return_none_schedule(cls, v):\n        if isinstance(v, NoSchedule):\n            return None\n        return v\n\n    version: Optional[str] = FieldFrom(objects.Deployment)\n    schedule: Optional[SCHEDULE_TYPES] = FieldFrom(objects.Deployment)\n    description: Optional[str] = FieldFrom(objects.Deployment)\n    is_schedule_active: bool = FieldFrom(objects.Deployment)\n    parameters: Optional[Dict[str, Any]] = Field(\n        default=None,\n        description=\"Parameters for flow runs scheduled by the deployment.\",\n    )\n    tags: List[str] = FieldFrom(objects.Deployment)\n    work_queue_name: Optional[str] = FieldFrom(objects.Deployment)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the deployment's work pool.\",\n        example=\"my-work-pool\",\n    )\n    path: Optional[str] = FieldFrom(objects.Deployment)\n    infra_overrides: Optional[Dict[str, Any]] = FieldFrom(objects.Deployment)\n    entrypoint: Optional[str] = FieldFrom(objects.Deployment)\n    manifest_path: Optional[str] = FieldFrom(objects.Deployment)\n    storage_document_id: Optional[UUID] = FieldFrom(objects.Deployment)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.Deployment)\n    enforce_parameter_schema: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n\n    def check_valid_configuration(self, base_job_template: dict):\n        \"\"\"Check that the combination of base_job_template defaults\n        and infra_overrides conforms to the specified schema.\n        \"\"\"\n        variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n        if variables_schema is not None:\n            # jsonschema considers required fields, even if that field has a default,\n            # to still be required. To get around this we remove the fields from\n            # required if there is a default present.\n            required = variables_schema.get(\"required\")\n            properties = variables_schema.get(\"properties\")\n            if required is not None and properties is not None:\n                for k, v in properties.items():\n                    if \"default\" in v and k in required:\n                        required.remove(k)\n\n        if variables_schema is not None:\n            jsonschema.validate(self.infra_overrides, variables_schema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentUpdate.check_valid_configuration","title":"check_valid_configuration","text":"

    Check that the combination of base_job_template defaults and infra_overrides conforms to the specified schema.

    Source code in prefect/client/schemas/actions.py
    def check_valid_configuration(self, base_job_template: dict):\n    \"\"\"Check that the combination of base_job_template defaults\n    and infra_overrides conforms to the specified schema.\n    \"\"\"\n    variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n    if variables_schema is not None:\n        # jsonschema considers required fields, even if that field has a default,\n        # to still be required. To get around this we remove the fields from\n        # required if there is a default present.\n        required = variables_schema.get(\"required\")\n        properties = variables_schema.get(\"properties\")\n        if required is not None and properties is not None:\n            for k, v in properties.items():\n                if \"default\" in v and k in required:\n                    required.remove(k)\n\n    if variables_schema is not None:\n        jsonschema.validate(self.infra_overrides, variables_schema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowRunUpdate","title":"FlowRunUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow run.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowRunUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow run.\"\"\"\n\n    name: Optional[str] = FieldFrom(objects.FlowRun)\n    flow_version: Optional[str] = FieldFrom(objects.FlowRun)\n    parameters: dict = FieldFrom(objects.FlowRun)\n    empirical_policy: objects.FlowRunPolicy = FieldFrom(objects.FlowRun)\n    tags: List[str] = FieldFrom(objects.FlowRun)\n    infrastructure_pid: Optional[str] = FieldFrom(objects.FlowRun)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.TaskRunCreate","title":"TaskRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a task run

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass TaskRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a task run\"\"\"\n\n    # TaskRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the task run to create\"\n    )\n\n    name: str = FieldFrom(objects.TaskRun)\n    flow_run_id: Optional[UUID] = FieldFrom(objects.TaskRun)\n    task_key: str = FieldFrom(objects.TaskRun)\n    dynamic_key: str = FieldFrom(objects.TaskRun)\n    cache_key: Optional[str] = FieldFrom(objects.TaskRun)\n    cache_expiration: Optional[objects.DateTimeTZ] = FieldFrom(objects.TaskRun)\n    task_version: Optional[str] = FieldFrom(objects.TaskRun)\n    empirical_policy: objects.TaskRunPolicy = FieldFrom(objects.TaskRun)\n    tags: List[str] = FieldFrom(objects.TaskRun)\n    task_inputs: Dict[\n        str,\n        List[\n            Union[\n                objects.TaskRunResult,\n                objects.Parameter,\n                objects.Constant,\n            ]\n        ],\n    ] = FieldFrom(objects.TaskRun)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.TaskRunUpdate","title":"TaskRunUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a task run

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass TaskRunUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a task run\"\"\"\n\n    name: str = FieldFrom(objects.TaskRun)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowRunCreate","title":"FlowRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run.\"\"\"\n\n    # FlowRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the flow run to create\"\n    )\n\n    name: str = FieldFrom(objects.FlowRun)\n    flow_id: UUID = FieldFrom(objects.FlowRun)\n    deployment_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    flow_version: Optional[str] = FieldFrom(objects.FlowRun)\n    parameters: dict = FieldFrom(objects.FlowRun)\n    context: dict = FieldFrom(objects.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    empirical_policy: objects.FlowRunPolicy = FieldFrom(objects.FlowRun)\n    tags: List[str] = FieldFrom(objects.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(objects.FlowRun)\n\n    class Config(ActionBaseModel.Config):\n        json_dumps = orjson_dumps_extra_compatible\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentFlowRunCreate","title":"DeploymentFlowRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run from a deployment.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass DeploymentFlowRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run from a deployment.\"\"\"\n\n    # FlowRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the flow run to create\"\n    )\n\n    name: Optional[str] = FieldFrom(objects.FlowRun)\n    parameters: dict = FieldFrom(objects.FlowRun)\n    context: dict = FieldFrom(objects.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    empirical_policy: objects.FlowRunPolicy = FieldFrom(objects.FlowRun)\n    tags: List[str] = FieldFrom(objects.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(objects.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    work_queue_name: Optional[str] = FieldFrom(objects.FlowRun)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.SavedSearchCreate","title":"SavedSearchCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a saved search.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass SavedSearchCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a saved search.\"\"\"\n\n    name: str = FieldFrom(objects.SavedSearch)\n    filters: List[objects.SavedSearchFilter] = FieldFrom(objects.SavedSearch)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.ConcurrencyLimitCreate","title":"ConcurrencyLimitCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a concurrency limit.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass ConcurrencyLimitCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a concurrency limit.\"\"\"\n\n    tag: str = FieldFrom(objects.ConcurrencyLimit)\n    concurrency_limit: int = FieldFrom(objects.ConcurrencyLimit)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockTypeCreate","title":"BlockTypeCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block type.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockTypeCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block type.\"\"\"\n\n    name: str = FieldFrom(objects.BlockType)\n    slug: str = FieldFrom(objects.BlockType)\n    logo_url: Optional[objects.HttpUrl] = FieldFrom(objects.BlockType)\n    documentation_url: Optional[objects.HttpUrl] = FieldFrom(objects.BlockType)\n    description: Optional[str] = FieldFrom(objects.BlockType)\n    code_example: Optional[str] = FieldFrom(objects.BlockType)\n\n    # validators\n    _validate_slug_format = validator(\"slug\", allow_reuse=True)(\n        validate_block_type_slug\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockTypeUpdate","title":"BlockTypeUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a block type.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockTypeUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a block type.\"\"\"\n\n    logo_url: Optional[objects.HttpUrl] = FieldFrom(objects.BlockType)\n    documentation_url: Optional[objects.HttpUrl] = FieldFrom(objects.BlockType)\n    description: Optional[str] = FieldFrom(objects.BlockType)\n    code_example: Optional[str] = FieldFrom(objects.BlockType)\n\n    @classmethod\n    def updatable_fields(cls) -> set:\n        return get_class_fields_only(cls)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockSchemaCreate","title":"BlockSchemaCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block schema.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockSchemaCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block schema.\"\"\"\n\n    fields: dict = FieldFrom(objects.BlockSchema)\n    block_type_id: Optional[UUID] = FieldFrom(objects.BlockSchema)\n    capabilities: List[str] = FieldFrom(objects.BlockSchema)\n    version: str = FieldFrom(objects.BlockSchema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockDocumentCreate","title":"BlockDocumentCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block document.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block document.\"\"\"\n\n    name: Optional[str] = FieldFrom(objects.BlockDocument)\n    data: dict = FieldFrom(objects.BlockDocument)\n    block_schema_id: UUID = FieldFrom(objects.BlockDocument)\n    block_type_id: UUID = FieldFrom(objects.BlockDocument)\n    is_anonymous: bool = FieldFrom(objects.BlockDocument)\n\n    _validate_name_format = validator(\"name\", allow_reuse=True)(\n        validate_block_document_name\n    )\n\n    @root_validator\n    def validate_name_is_present_if_not_anonymous(cls, values):\n        # TODO: We should find an elegant way to reuse this logic from the origin model\n        if not values.get(\"is_anonymous\") and not values.get(\"name\"):\n            raise ValueError(\"Names must be provided for block documents.\")\n        return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockDocumentUpdate","title":"BlockDocumentUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a block document.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a block document.\"\"\"\n\n    block_schema_id: Optional[UUID] = Field(\n        default=None, description=\"A block schema ID\"\n    )\n    data: dict = FieldFrom(objects.BlockDocument)\n    merge_existing_data: bool = True\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockDocumentReferenceCreate","title":"BlockDocumentReferenceCreate","text":"

    Bases: ActionBaseModel

    Data used to create block document reference.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentReferenceCreate(ActionBaseModel):\n    \"\"\"Data used to create block document reference.\"\"\"\n\n    id: UUID = FieldFrom(objects.BlockDocumentReference)\n    parent_block_document_id: UUID = FieldFrom(objects.BlockDocumentReference)\n    reference_block_document_id: UUID = FieldFrom(objects.BlockDocumentReference)\n    name: str = FieldFrom(objects.BlockDocumentReference)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.LogCreate","title":"LogCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a log.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass LogCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a log.\"\"\"\n\n    name: str = FieldFrom(objects.Log)\n    level: int = FieldFrom(objects.Log)\n    message: str = FieldFrom(objects.Log)\n    timestamp: objects.DateTimeTZ = FieldFrom(objects.Log)\n    flow_run_id: Optional[UUID] = FieldFrom(objects.Log)\n    task_run_id: Optional[UUID] = FieldFrom(objects.Log)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.WorkPoolCreate","title":"WorkPoolCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a work pool.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass WorkPoolCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a work pool.\"\"\"\n\n    name: str = FieldFrom(objects.WorkPool)\n    description: Optional[str] = FieldFrom(objects.WorkPool)\n    type: str = Field(description=\"The work pool type.\", default=\"prefect-agent\")\n    base_job_template: Dict[str, Any] = FieldFrom(objects.WorkPool)\n    is_paused: bool = FieldFrom(objects.WorkPool)\n    concurrency_limit: Optional[int] = FieldFrom(objects.WorkPool)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.WorkPoolUpdate","title":"WorkPoolUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a work pool.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass WorkPoolUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a work pool.\"\"\"\n\n    description: Optional[str] = FieldFrom(objects.WorkPool)\n    is_paused: Optional[bool] = FieldFrom(objects.WorkPool)\n    base_job_template: Optional[Dict[str, Any]] = FieldFrom(objects.WorkPool)\n    concurrency_limit: Optional[int] = FieldFrom(objects.WorkPool)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.WorkQueueCreate","title":"WorkQueueCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a work queue.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass WorkQueueCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a work queue.\"\"\"\n\n    name: str = FieldFrom(objects.WorkQueue)\n    description: Optional[str] = FieldFrom(objects.WorkQueue)\n    is_paused: bool = FieldFrom(objects.WorkQueue)\n    concurrency_limit: Optional[int] = FieldFrom(objects.WorkQueue)\n    priority: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The queue's priority. Lower values are higher priority (1 is the highest).\"\n        ),\n    )\n\n    # DEPRECATED\n\n    filter: Optional[objects.QueueFilter] = Field(\n        None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.WorkQueueUpdate","title":"WorkQueueUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a work queue.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass WorkQueueUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a work queue.\"\"\"\n\n    name: str = FieldFrom(objects.WorkQueue)\n    description: Optional[str] = FieldFrom(objects.WorkQueue)\n    is_paused: bool = FieldFrom(objects.WorkQueue)\n    concurrency_limit: Optional[int] = FieldFrom(objects.WorkQueue)\n    priority: Optional[int] = FieldFrom(objects.WorkQueue)\n    last_polled: Optional[DateTimeTZ] = FieldFrom(objects.WorkQueue)\n\n    # DEPRECATED\n\n    filter: Optional[objects.QueueFilter] = Field(\n        None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowRunNotificationPolicyCreate","title":"FlowRunNotificationPolicyCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run notification policy.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowRunNotificationPolicyCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run notification policy.\"\"\"\n\n    is_active: bool = FieldFrom(objects.FlowRunNotificationPolicy)\n    state_names: List[str] = FieldFrom(objects.FlowRunNotificationPolicy)\n    tags: List[str] = FieldFrom(objects.FlowRunNotificationPolicy)\n    block_document_id: UUID = FieldFrom(objects.FlowRunNotificationPolicy)\n    message_template: Optional[str] = FieldFrom(objects.FlowRunNotificationPolicy)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowRunNotificationPolicyUpdate","title":"FlowRunNotificationPolicyUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow run notification policy.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowRunNotificationPolicyUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow run notification policy.\"\"\"\n\n    is_active: Optional[bool] = FieldFrom(objects.FlowRunNotificationPolicy)\n    state_names: Optional[List[str]] = FieldFrom(objects.FlowRunNotificationPolicy)\n    tags: Optional[List[str]] = FieldFrom(objects.FlowRunNotificationPolicy)\n    block_document_id: Optional[UUID] = FieldFrom(objects.FlowRunNotificationPolicy)\n    message_template: Optional[str] = FieldFrom(objects.FlowRunNotificationPolicy)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.ArtifactCreate","title":"ArtifactCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create an artifact.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass ArtifactCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create an artifact.\"\"\"\n\n    key: Optional[str] = FieldFrom(objects.Artifact)\n    type: Optional[str] = FieldFrom(objects.Artifact)\n    description: Optional[str] = FieldFrom(objects.Artifact)\n    data: Optional[Union[Dict[str, Any], Any]] = FieldFrom(objects.Artifact)\n    metadata_: Optional[Dict[str, str]] = FieldFrom(objects.Artifact)\n    flow_run_id: Optional[UUID] = FieldFrom(objects.Artifact)\n    task_run_id: Optional[UUID] = FieldFrom(objects.Artifact)\n\n    _validate_artifact_format = validator(\"key\", allow_reuse=True)(\n        validate_artifact_key\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.ArtifactUpdate","title":"ArtifactUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update an artifact.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass ArtifactUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update an artifact.\"\"\"\n\n    data: Optional[Union[Dict[str, Any], Any]] = FieldFrom(objects.Artifact)\n    description: Optional[str] = FieldFrom(objects.Artifact)\n    metadata_: Optional[Dict[str, str]] = FieldFrom(objects.Artifact)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.VariableCreate","title":"VariableCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a Variable.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass VariableCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a Variable.\"\"\"\n\n    name: str = FieldFrom(objects.Variable)\n    value: str = FieldFrom(objects.Variable)\n    tags: Optional[List[str]] = FieldFrom(objects.Variable)\n\n    # validators\n    _validate_name_format = validator(\"name\", allow_reuse=True)(validate_variable_name)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.VariableUpdate","title":"VariableUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a Variable.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass VariableUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a Variable.\"\"\"\n\n    name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the variable\",\n        example=\"my_variable\",\n        max_length=objects.MAX_VARIABLE_NAME_LENGTH,\n    )\n    value: Optional[str] = Field(\n        default=None,\n        description=\"The value of the variable\",\n        example=\"my-value\",\n        max_length=objects.MAX_VARIABLE_NAME_LENGTH,\n    )\n    tags: Optional[List[str]] = FieldFrom(objects.Variable)\n\n    # validators\n    _validate_name_format = validator(\"name\", allow_reuse=True)(validate_variable_name)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_2","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters","title":"prefect.client.schemas.filters","text":"

    Schemas that define Prefect REST API filtering operations.

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.Operator","title":"Operator","text":"

    Bases: AutoEnum

    Operators for combining filter criteria.

    Source code in prefect/client/schemas/filters.py
    class Operator(AutoEnum):\n    \"\"\"Operators for combining filter criteria.\"\"\"\n\n    and_ = AutoEnum.auto()\n    or_ = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.OperatorMixin","title":"OperatorMixin","text":"

    Base model for Prefect filters that combines criteria with a user-provided operator

    Source code in prefect/client/schemas/filters.py
    class OperatorMixin:\n    \"\"\"Base model for Prefect filters that combines criteria with a user-provided operator\"\"\"\n\n    operator: Operator = Field(\n        default=Operator.and_,\n        description=\"Operator for combining filter criteria. Defaults to 'and_'.\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowFilterId","title":"FlowFilterId","text":"

    Bases: PrefectBaseModel

    Filter by Flow.id.

    Source code in prefect/client/schemas/filters.py
    class FlowFilterId(PrefectBaseModel):\n    \"\"\"Filter by `Flow.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowFilterName","title":"FlowFilterName","text":"

    Bases: PrefectBaseModel

    Filter by Flow.name.

    Source code in prefect/client/schemas/filters.py
    class FlowFilterName(PrefectBaseModel):\n    \"\"\"Filter by `Flow.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of flow names to include\",\n        example=[\"my-flow-1\", \"my-flow-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowFilterTags","title":"FlowFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by Flow.tags.

    Source code in prefect/client/schemas/filters.py
    class FlowFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `Flow.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Flows will be returned only if their tags are a superset\"\n            \" of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include flows without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowFilter","title":"FlowFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter for flows. Only flows matching all criteria will be returned.

    Source code in prefect/client/schemas/filters.py
    class FlowFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter for flows. Only flows matching all criteria will be returned.\"\"\"\n\n    id: Optional[FlowFilterId] = Field(\n        default=None, description=\"Filter criteria for `Flow.id`\"\n    )\n    name: Optional[FlowFilterName] = Field(\n        default=None, description=\"Filter criteria for `Flow.name`\"\n    )\n    tags: Optional[FlowFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Flow.tags`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterId","title":"FlowRunFilterId","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.id.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterId(PrefectBaseModel):\n    \"\"\"Filter by FlowRun.id.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run ids to include\"\n    )\n    not_any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run ids to exclude\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterName","title":"FlowRunFilterName","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.name.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterName(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of flow run names to include\",\n        example=[\"my-flow-run-1\", \"my-flow-run-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterTags","title":"FlowRunFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by FlowRun.tags.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `FlowRun.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Flow runs will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include flow runs without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterDeploymentId","title":"FlowRunFilterDeploymentId","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by FlowRun.deployment_id.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterDeploymentId(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `FlowRun.deployment_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run deployment ids to include\"\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without deployment ids\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterWorkQueueName","title":"FlowRunFilterWorkQueueName","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by FlowRun.work_queue_name.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterWorkQueueName(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `FlowRun.work_queue_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"work_queue_1\", \"work_queue_2\"],\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without work queue names\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterStateType","title":"FlowRunFilterStateType","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.state_type.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterStateType(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.state_type`.\"\"\"\n\n    any_: Optional[List[StateType]] = Field(\n        default=None, description=\"A list of flow run state types to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterFlowVersion","title":"FlowRunFilterFlowVersion","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.flow_version.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterFlowVersion(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.flow_version`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run flow_versions to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterStartTime","title":"FlowRunFilterStartTime","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.start_time.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterStartTime(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs starting at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs starting at or after this time\",\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only return flow runs without a start time\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterExpectedStartTime","title":"FlowRunFilterExpectedStartTime","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.expected_start_time.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterExpectedStartTime(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.expected_start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs scheduled to start at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs scheduled to start at or after this time\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterNextScheduledStartTime","title":"FlowRunFilterNextScheduledStartTime","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.next_scheduled_start_time.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterNextScheduledStartTime(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.next_scheduled_start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include flow runs with a next_scheduled_start_time or before this\"\n            \" time\"\n        ),\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include flow runs with a next_scheduled_start_time at or after this\"\n            \" time\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterParentFlowRunId","title":"FlowRunFilterParentFlowRunId","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter for subflows of the given flow runs

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterParentFlowRunId(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter for subflows of the given flow runs\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run parents to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterParentTaskRunId","title":"FlowRunFilterParentTaskRunId","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by FlowRun.parent_task_run_id.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterParentTaskRunId(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `FlowRun.parent_task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run parent_task_run_ids to include\"\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without parent_task_run_id\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterIdempotencyKey","title":"FlowRunFilterIdempotencyKey","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.idempotency_key.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterIdempotencyKey(PrefectBaseModel):\n    \"\"\"Filter by FlowRun.idempotency_key.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run idempotency keys to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run idempotency keys to exclude\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilter","title":"FlowRunFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter flow runs. Only flow runs matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter flow runs. Only flow runs matching all criteria will be returned\"\"\"\n\n    id: Optional[FlowRunFilterId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.id`\"\n    )\n    name: Optional[FlowRunFilterName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.name`\"\n    )\n    tags: Optional[FlowRunFilterTags] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.tags`\"\n    )\n    deployment_id: Optional[FlowRunFilterDeploymentId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.deployment_id`\"\n    )\n    work_queue_name: Optional[FlowRunFilterWorkQueueName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.work_queue_name\"\n    )\n    state: Optional[FlowRunFilterState] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.state`\"\n    )\n    flow_version: Optional[FlowRunFilterFlowVersion] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.flow_version`\"\n    )\n    start_time: Optional[FlowRunFilterStartTime] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.start_time`\"\n    )\n    expected_start_time: Optional[FlowRunFilterExpectedStartTime] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.expected_start_time`\"\n    )\n    next_scheduled_start_time: Optional[FlowRunFilterNextScheduledStartTime] = Field(\n        default=None,\n        description=\"Filter criteria for `FlowRun.next_scheduled_start_time`\",\n    )\n    parent_flow_run_id: Optional[FlowRunFilterParentFlowRunId] = Field(\n        default=None, description=\"Filter criteria for subflows of the given flow runs\"\n    )\n    parent_task_run_id: Optional[FlowRunFilterParentTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.parent_task_run_id`\"\n    )\n    idempotency_key: Optional[FlowRunFilterIdempotencyKey] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.idempotency_key`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterId","title":"TaskRunFilterId","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.id.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterId(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterName","title":"TaskRunFilterName","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.name.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterName(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of task run names to include\",\n        example=[\"my-task-run-1\", \"my-task-run-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterTags","title":"TaskRunFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by TaskRun.tags.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `TaskRun.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Task runs will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include task runs without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterStateType","title":"TaskRunFilterStateType","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.state_type.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterStateType(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.state_type`.\"\"\"\n\n    any_: Optional[List[StateType]] = Field(\n        default=None, description=\"A list of task run state types to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterSubFlowRuns","title":"TaskRunFilterSubFlowRuns","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.subflow_run.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterSubFlowRuns(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.subflow_run`.\"\"\"\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If true, only include task runs that are subflow run parents; if false,\"\n            \" exclude parent task runs\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterStartTime","title":"TaskRunFilterStartTime","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.start_time.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterStartTime(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include task runs starting at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include task runs starting at or after this time\",\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only return task runs without a start time\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilter","title":"TaskRunFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter task runs. Only task runs matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter task runs. Only task runs matching all criteria will be returned\"\"\"\n\n    id: Optional[TaskRunFilterId] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.id`\"\n    )\n    name: Optional[TaskRunFilterName] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.name`\"\n    )\n    tags: Optional[TaskRunFilterTags] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.tags`\"\n    )\n    state: Optional[TaskRunFilterState] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.state`\"\n    )\n    start_time: Optional[TaskRunFilterStartTime] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.start_time`\"\n    )\n    subflow_runs: Optional[TaskRunFilterSubFlowRuns] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.subflow_run`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterId","title":"DeploymentFilterId","text":"

    Bases: PrefectBaseModel

    Filter by Deployment.id.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterId(PrefectBaseModel):\n    \"\"\"Filter by `Deployment.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of deployment ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterName","title":"DeploymentFilterName","text":"

    Bases: PrefectBaseModel

    Filter by Deployment.name.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterName(PrefectBaseModel):\n    \"\"\"Filter by `Deployment.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of deployment names to include\",\n        example=[\"my-deployment-1\", \"my-deployment-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterWorkQueueName","title":"DeploymentFilterWorkQueueName","text":"

    Bases: PrefectBaseModel

    Filter by Deployment.work_queue_name.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterWorkQueueName(PrefectBaseModel):\n    \"\"\"Filter by `Deployment.work_queue_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"work_queue_1\", \"work_queue_2\"],\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterIsScheduleActive","title":"DeploymentFilterIsScheduleActive","text":"

    Bases: PrefectBaseModel

    Filter by Deployment.is_schedule_active.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterIsScheduleActive(PrefectBaseModel):\n    \"\"\"Filter by `Deployment.is_schedule_active`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=\"Only returns where deployment schedule is/is not active\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterTags","title":"DeploymentFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by Deployment.tags.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `Deployment.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Deployments will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include deployments without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilter","title":"DeploymentFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter for deployments. Only deployments matching all criteria will be returned.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter for deployments. Only deployments matching all criteria will be returned.\"\"\"\n\n    id: Optional[DeploymentFilterId] = Field(\n        default=None, description=\"Filter criteria for `Deployment.id`\"\n    )\n    name: Optional[DeploymentFilterName] = Field(\n        default=None, description=\"Filter criteria for `Deployment.name`\"\n    )\n    is_schedule_active: Optional[DeploymentFilterIsScheduleActive] = Field(\n        default=None, description=\"Filter criteria for `Deployment.is_schedule_active`\"\n    )\n    tags: Optional[DeploymentFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Deployment.tags`\"\n    )\n    work_queue_name: Optional[DeploymentFilterWorkQueueName] = Field(\n        default=None, description=\"Filter criteria for `Deployment.work_queue_name`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterName","title":"LogFilterName","text":"

    Bases: PrefectBaseModel

    Filter by Log.name.

    Source code in prefect/client/schemas/filters.py
    class LogFilterName(PrefectBaseModel):\n    \"\"\"Filter by `Log.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of log names to include\",\n        example=[\"prefect.logger.flow_runs\", \"prefect.logger.task_runs\"],\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterLevel","title":"LogFilterLevel","text":"

    Bases: PrefectBaseModel

    Filter by Log.level.

    Source code in prefect/client/schemas/filters.py
    class LogFilterLevel(PrefectBaseModel):\n    \"\"\"Filter by `Log.level`.\"\"\"\n\n    ge_: Optional[int] = Field(\n        default=None,\n        description=\"Include logs with a level greater than or equal to this level\",\n        example=20,\n    )\n\n    le_: Optional[int] = Field(\n        default=None,\n        description=\"Include logs with a level less than or equal to this level\",\n        example=50,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterTimestamp","title":"LogFilterTimestamp","text":"

    Bases: PrefectBaseModel

    Filter by Log.timestamp.

    Source code in prefect/client/schemas/filters.py
    class LogFilterTimestamp(PrefectBaseModel):\n    \"\"\"Filter by `Log.timestamp`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include logs with a timestamp at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include logs with a timestamp at or after this time\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterFlowRunId","title":"LogFilterFlowRunId","text":"

    Bases: PrefectBaseModel

    Filter by Log.flow_run_id.

    Source code in prefect/client/schemas/filters.py
    class LogFilterFlowRunId(PrefectBaseModel):\n    \"\"\"Filter by `Log.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterTaskRunId","title":"LogFilterTaskRunId","text":"

    Bases: PrefectBaseModel

    Filter by Log.task_run_id.

    Source code in prefect/client/schemas/filters.py
    class LogFilterTaskRunId(PrefectBaseModel):\n    \"\"\"Filter by `Log.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilter","title":"LogFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter logs. Only logs matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class LogFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter logs. Only logs matching all criteria will be returned\"\"\"\n\n    level: Optional[LogFilterLevel] = Field(\n        default=None, description=\"Filter criteria for `Log.level`\"\n    )\n    timestamp: Optional[LogFilterTimestamp] = Field(\n        default=None, description=\"Filter criteria for `Log.timestamp`\"\n    )\n    flow_run_id: Optional[LogFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Log.flow_run_id`\"\n    )\n    task_run_id: Optional[LogFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Log.task_run_id`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FilterSet","title":"FilterSet","text":"

    Bases: PrefectBaseModel

    A collection of filters for common objects

    Source code in prefect/client/schemas/filters.py
    class FilterSet(PrefectBaseModel):\n    \"\"\"A collection of filters for common objects\"\"\"\n\n    flows: FlowFilter = Field(\n        default_factory=FlowFilter, description=\"Filters that apply to flows\"\n    )\n    flow_runs: FlowRunFilter = Field(\n        default_factory=FlowRunFilter, description=\"Filters that apply to flow runs\"\n    )\n    task_runs: TaskRunFilter = Field(\n        default_factory=TaskRunFilter, description=\"Filters that apply to task runs\"\n    )\n    deployments: DeploymentFilter = Field(\n        default_factory=DeploymentFilter,\n        description=\"Filters that apply to deployments\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockTypeFilterName","title":"BlockTypeFilterName","text":"

    Bases: PrefectBaseModel

    Filter by BlockType.name

    Source code in prefect/client/schemas/filters.py
    class BlockTypeFilterName(PrefectBaseModel):\n    \"\"\"Filter by `BlockType.name`\"\"\"\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockTypeFilterSlug","title":"BlockTypeFilterSlug","text":"

    Bases: PrefectBaseModel

    Filter by BlockType.slug

    Source code in prefect/client/schemas/filters.py
    class BlockTypeFilterSlug(PrefectBaseModel):\n    \"\"\"Filter by `BlockType.slug`\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of slugs to match\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockTypeFilter","title":"BlockTypeFilter","text":"

    Bases: PrefectBaseModel

    Filter BlockTypes

    Source code in prefect/client/schemas/filters.py
    class BlockTypeFilter(PrefectBaseModel):\n    \"\"\"Filter BlockTypes\"\"\"\n\n    name: Optional[BlockTypeFilterName] = Field(\n        default=None, description=\"Filter criteria for `BlockType.name`\"\n    )\n\n    slug: Optional[BlockTypeFilterSlug] = Field(\n        default=None, description=\"Filter criteria for `BlockType.slug`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilterBlockTypeId","title":"BlockSchemaFilterBlockTypeId","text":"

    Bases: PrefectBaseModel

    Filter by BlockSchema.block_type_id.

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilterBlockTypeId(PrefectBaseModel):\n    \"\"\"Filter by `BlockSchema.block_type_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block type ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilterId","title":"BlockSchemaFilterId","text":"

    Bases: PrefectBaseModel

    Filter by BlockSchema.id

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilterId(PrefectBaseModel):\n    \"\"\"Filter by BlockSchema.id\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilterCapabilities","title":"BlockSchemaFilterCapabilities","text":"

    Bases: PrefectBaseModel

    Filter by BlockSchema.capabilities

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilterCapabilities(PrefectBaseModel):\n    \"\"\"Filter by `BlockSchema.capabilities`\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"write-storage\", \"read-storage\"],\n        description=(\n            \"A list of block capabilities. Block entities will be returned only if an\"\n            \" associated block schema has a superset of the defined capabilities.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilterVersion","title":"BlockSchemaFilterVersion","text":"

    Bases: PrefectBaseModel

    Filter by BlockSchema.capabilities

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilterVersion(PrefectBaseModel):\n    \"\"\"Filter by `BlockSchema.capabilities`\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"2.0.0\", \"2.1.0\"],\n        description=\"A list of block schema versions.\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilter","title":"BlockSchemaFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter BlockSchemas

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter BlockSchemas\"\"\"\n\n    block_type_id: Optional[BlockSchemaFilterBlockTypeId] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.block_type_id`\"\n    )\n    block_capabilities: Optional[BlockSchemaFilterCapabilities] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.capabilities`\"\n    )\n    id: Optional[BlockSchemaFilterId] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.id`\"\n    )\n    version: Optional[BlockSchemaFilterVersion] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.version`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilterIsAnonymous","title":"BlockDocumentFilterIsAnonymous","text":"

    Bases: PrefectBaseModel

    Filter by BlockDocument.is_anonymous.

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilterIsAnonymous(PrefectBaseModel):\n    \"\"\"Filter by `BlockDocument.is_anonymous`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Filter block documents for only those that are or are not anonymous.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilterBlockTypeId","title":"BlockDocumentFilterBlockTypeId","text":"

    Bases: PrefectBaseModel

    Filter by BlockDocument.block_type_id.

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilterBlockTypeId(PrefectBaseModel):\n    \"\"\"Filter by `BlockDocument.block_type_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block type ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilterId","title":"BlockDocumentFilterId","text":"

    Bases: PrefectBaseModel

    Filter by BlockDocument.id.

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilterId(PrefectBaseModel):\n    \"\"\"Filter by `BlockDocument.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilterName","title":"BlockDocumentFilterName","text":"

    Bases: PrefectBaseModel

    Filter by BlockDocument.name.

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilterName(PrefectBaseModel):\n    \"\"\"Filter by `BlockDocument.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of block names to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match block names against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-block%\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilter","title":"BlockDocumentFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter BlockDocuments. Only BlockDocuments matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter BlockDocuments. Only BlockDocuments matching all criteria will be returned\"\"\"\n\n    id: Optional[BlockDocumentFilterId] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.id`\"\n    )\n    is_anonymous: Optional[BlockDocumentFilterIsAnonymous] = Field(\n        # default is to exclude anonymous blocks\n        BlockDocumentFilterIsAnonymous(eq_=False),\n        description=(\n            \"Filter criteria for `BlockDocument.is_anonymous`. \"\n            \"Defaults to excluding anonymous blocks.\"\n        ),\n    )\n    block_type_id: Optional[BlockDocumentFilterBlockTypeId] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.block_type_id`\"\n    )\n    name: Optional[BlockDocumentFilterName] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.name`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunNotificationPolicyFilterIsActive","title":"FlowRunNotificationPolicyFilterIsActive","text":"

    Bases: PrefectBaseModel

    Filter by FlowRunNotificationPolicy.is_active.

    Source code in prefect/client/schemas/filters.py
    class FlowRunNotificationPolicyFilterIsActive(PrefectBaseModel):\n    \"\"\"Filter by `FlowRunNotificationPolicy.is_active`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Filter notification policies for only those that are or are not active.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunNotificationPolicyFilter","title":"FlowRunNotificationPolicyFilter","text":"

    Bases: PrefectBaseModel

    Filter FlowRunNotificationPolicies.

    Source code in prefect/client/schemas/filters.py
    class FlowRunNotificationPolicyFilter(PrefectBaseModel):\n    \"\"\"Filter FlowRunNotificationPolicies.\"\"\"\n\n    is_active: Optional[FlowRunNotificationPolicyFilterIsActive] = Field(\n        default=FlowRunNotificationPolicyFilterIsActive(eq_=False),\n        description=\"Filter criteria for `FlowRunNotificationPolicy.is_active`. \",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkQueueFilterId","title":"WorkQueueFilterId","text":"

    Bases: PrefectBaseModel

    Filter by WorkQueue.id.

    Source code in prefect/client/schemas/filters.py
    class WorkQueueFilterId(PrefectBaseModel):\n    \"\"\"Filter by `WorkQueue.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None,\n        description=\"A list of work queue ids to include\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkQueueFilterName","title":"WorkQueueFilterName","text":"

    Bases: PrefectBaseModel

    Filter by WorkQueue.name.

    Source code in prefect/client/schemas/filters.py
    class WorkQueueFilterName(PrefectBaseModel):\n    \"\"\"Filter by `WorkQueue.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"wq-1\", \"wq-2\"],\n    )\n\n    startswith_: Optional[List[str]] = Field(\n        default=None,\n        description=(\n            \"A list of case-insensitive starts-with matches. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', and 'Marvin-robot', but not 'sad-marvin'.\"\n        ),\n        example=[\"marvin\", \"Marvin-robot\"],\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkQueueFilter","title":"WorkQueueFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter work queues. Only work queues matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class WorkQueueFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter work queues. Only work queues matching all criteria will be\n    returned\"\"\"\n\n    id: Optional[WorkQueueFilterId] = Field(\n        default=None, description=\"Filter criteria for `WorkQueue.id`\"\n    )\n\n    name: Optional[WorkQueueFilterName] = Field(\n        default=None, description=\"Filter criteria for `WorkQueue.name`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkPoolFilterId","title":"WorkPoolFilterId","text":"

    Bases: PrefectBaseModel

    Filter by WorkPool.id.

    Source code in prefect/client/schemas/filters.py
    class WorkPoolFilterId(PrefectBaseModel):\n    \"\"\"Filter by `WorkPool.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of work pool ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkPoolFilterName","title":"WorkPoolFilterName","text":"

    Bases: PrefectBaseModel

    Filter by WorkPool.name.

    Source code in prefect/client/schemas/filters.py
    class WorkPoolFilterName(PrefectBaseModel):\n    \"\"\"Filter by `WorkPool.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of work pool names to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkPoolFilterType","title":"WorkPoolFilterType","text":"

    Bases: PrefectBaseModel

    Filter by WorkPool.type.

    Source code in prefect/client/schemas/filters.py
    class WorkPoolFilterType(PrefectBaseModel):\n    \"\"\"Filter by `WorkPool.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of work pool types to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkerFilterWorkPoolId","title":"WorkerFilterWorkPoolId","text":"

    Bases: PrefectBaseModel

    Filter by Worker.worker_config_id.

    Source code in prefect/client/schemas/filters.py
    class WorkerFilterWorkPoolId(PrefectBaseModel):\n    \"\"\"Filter by `Worker.worker_config_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of work pool ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkerFilterLastHeartbeatTime","title":"WorkerFilterLastHeartbeatTime","text":"

    Bases: PrefectBaseModel

    Filter by Worker.last_heartbeat_time.

    Source code in prefect/client/schemas/filters.py
    class WorkerFilterLastHeartbeatTime(PrefectBaseModel):\n    \"\"\"Filter by `Worker.last_heartbeat_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include processes whose last heartbeat was at or before this time\"\n        ),\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include processes whose last heartbeat was at or after this time\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterId","title":"ArtifactFilterId","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterId(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of artifact ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterKey","title":"ArtifactFilterKey","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.key.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterKey(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.key`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact keys to include\"\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match artifact keys against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-artifact-%\",\n    )\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If `true`, only include artifacts with a non-null key. If `false`, \"\n            \"only include artifacts with a null key.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterFlowRunId","title":"ArtifactFilterFlowRunId","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.flow_run_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterFlowRunId(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterTaskRunId","title":"ArtifactFilterTaskRunId","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.task_run_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterTaskRunId(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterType","title":"ArtifactFilterType","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.type.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterType(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to exclude\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilter","title":"ArtifactFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter artifacts. Only artifacts matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter artifacts. Only artifacts matching all criteria will be returned\"\"\"\n\n    id: Optional[ArtifactFilterId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.id`\"\n    )\n    key: Optional[ArtifactFilterKey] = Field(\n        default=None, description=\"Filter criteria for `Artifact.key`\"\n    )\n    flow_run_id: Optional[ArtifactFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.flow_run_id`\"\n    )\n    task_run_id: Optional[ArtifactFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.task_run_id`\"\n    )\n    type: Optional[ArtifactFilterType] = Field(\n        default=None, description=\"Filter criteria for `Artifact.type`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterLatestId","title":"ArtifactCollectionFilterLatestId","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.latest_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterLatestId(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.latest_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of artifact ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterKey","title":"ArtifactCollectionFilterKey","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.key.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterKey(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.key`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact keys to include\"\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match artifact keys against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-artifact-%\",\n    )\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If `true`, only include artifacts with a non-null key. If `false`, \"\n            \"only include artifacts with a null key. Should return all rows in \"\n            \"the ArtifactCollection table if specified.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterFlowRunId","title":"ArtifactCollectionFilterFlowRunId","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.flow_run_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterFlowRunId(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterTaskRunId","title":"ArtifactCollectionFilterTaskRunId","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.task_run_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterTaskRunId(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterType","title":"ArtifactCollectionFilterType","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.type.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterType(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to exclude\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilter","title":"ArtifactCollectionFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter artifact collections. Only artifact collections matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter artifact collections. Only artifact collections matching all criteria will be returned\"\"\"\n\n    latest_id: Optional[ArtifactCollectionFilterLatestId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.id`\"\n    )\n    key: Optional[ArtifactCollectionFilterKey] = Field(\n        default=None, description=\"Filter criteria for `Artifact.key`\"\n    )\n    flow_run_id: Optional[ArtifactCollectionFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.flow_run_id`\"\n    )\n    task_run_id: Optional[ArtifactCollectionFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.task_run_id`\"\n    )\n    type: Optional[ArtifactCollectionFilterType] = Field(\n        default=None, description=\"Filter criteria for `Artifact.type`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilterId","title":"VariableFilterId","text":"

    Bases: PrefectBaseModel

    Filter by Variable.id.

    Source code in prefect/client/schemas/filters.py
    class VariableFilterId(PrefectBaseModel):\n    \"\"\"Filter by `Variable.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of variable ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilterName","title":"VariableFilterName","text":"

    Bases: PrefectBaseModel

    Filter by Variable.name.

    Source code in prefect/client/schemas/filters.py
    class VariableFilterName(PrefectBaseModel):\n    \"\"\"Filter by `Variable.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of variables names to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match variable names against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my_variable_%\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilterValue","title":"VariableFilterValue","text":"

    Bases: PrefectBaseModel

    Filter by Variable.value.

    Source code in prefect/client/schemas/filters.py
    class VariableFilterValue(PrefectBaseModel):\n    \"\"\"Filter by `Variable.value`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of variables value to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match variable value against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-value-%\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilterTags","title":"VariableFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by Variable.tags.

    Source code in prefect/client/schemas/filters.py
    class VariableFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `Variable.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Variables will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include Variables without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilter","title":"VariableFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter variables. Only variables matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class VariableFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter variables. Only variables matching all criteria will be returned\"\"\"\n\n    id: Optional[VariableFilterId] = Field(\n        default=None, description=\"Filter criteria for `Variable.id`\"\n    )\n    name: Optional[VariableFilterName] = Field(\n        default=None, description=\"Filter criteria for `Variable.name`\"\n    )\n    value: Optional[VariableFilterValue] = Field(\n        default=None, description=\"Filter criteria for `Variable.value`\"\n    )\n    tags: Optional[VariableFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Variable.tags`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_3","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects","title":"prefect.client.schemas.objects","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.StateType","title":"StateType","text":"

    Bases: AutoEnum

    Enumeration of state types.

    Source code in prefect/client/schemas/objects.py
    class StateType(AutoEnum):\n    \"\"\"Enumeration of state types.\"\"\"\n\n    SCHEDULED = AutoEnum.auto()\n    PENDING = AutoEnum.auto()\n    RUNNING = AutoEnum.auto()\n    COMPLETED = AutoEnum.auto()\n    FAILED = AutoEnum.auto()\n    CANCELLED = AutoEnum.auto()\n    CRASHED = AutoEnum.auto()\n    PAUSED = AutoEnum.auto()\n    CANCELLING = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkPoolStatus","title":"WorkPoolStatus","text":"

    Bases: AutoEnum

    Enumeration of work pool statuses.

    Source code in prefect/client/schemas/objects.py
    class WorkPoolStatus(AutoEnum):\n    \"\"\"Enumeration of work pool statuses.\"\"\"\n\n    READY = AutoEnum.auto()\n    NOT_READY = AutoEnum.auto()\n    PAUSED = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkerStatus","title":"WorkerStatus","text":"

    Bases: AutoEnum

    Enumeration of worker statuses.

    Source code in prefect/client/schemas/objects.py
    class WorkerStatus(AutoEnum):\n    \"\"\"Enumeration of worker statuses.\"\"\"\n\n    ONLINE = AutoEnum.auto()\n    OFFLINE = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.DeploymentStatus","title":"DeploymentStatus","text":"

    Bases: AutoEnum

    Enumeration of deployment statuses.

    Source code in prefect/client/schemas/objects.py
    class DeploymentStatus(AutoEnum):\n    \"\"\"Enumeration of deployment statuses.\"\"\"\n\n    READY = AutoEnum.auto()\n    NOT_READY = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State","title":"State","text":"

    Bases: ObjectBaseModel, Generic[R]

    The state of a run.

    Source code in prefect/client/schemas/objects.py
    class State(ObjectBaseModel, Generic[R]):\n    \"\"\"\n    The state of a run.\n    \"\"\"\n\n    type: StateType\n    name: Optional[str] = Field(default=None)\n    timestamp: DateTimeTZ = Field(default_factory=lambda: pendulum.now(\"UTC\"))\n    message: Optional[str] = Field(default=None, example=\"Run started\")\n    state_details: StateDetails = Field(default_factory=StateDetails)\n    data: Union[\"BaseResult[R]\", \"DataDocument[R]\", Any] = Field(\n        default=None,\n    )\n\n    @overload\n    def result(self: \"State[R]\", raise_on_failure: bool = True) -> R:\n        ...\n\n    @overload\n    def result(self: \"State[R]\", raise_on_failure: bool = False) -> Union[R, Exception]:\n        ...\n\n    def result(self, raise_on_failure: bool = True, fetch: Optional[bool] = None):\n        \"\"\"\n        Retrieve the result attached to this state.\n\n        Args:\n            raise_on_failure: a boolean specifying whether to raise an exception\n                if the state is of type `FAILED` and the underlying data is an exception\n            fetch: a boolean specifying whether to resolve references to persisted\n                results into data. For synchronous users, this defaults to `True`.\n                For asynchronous users, this defaults to `False` for backwards\n                compatibility.\n\n        Raises:\n            TypeError: If the state is failed but the result is not an exception.\n\n        Returns:\n            The result of the run\n\n        Examples:\n            >>> from prefect import flow, task\n            >>> @task\n            >>> def my_task(x):\n            >>>     return x\n\n            Get the result from a task future in a flow\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     future = my_task(\"hello\")\n            >>>     state = future.wait()\n            >>>     result = state.result()\n            >>>     print(result)\n            >>> my_flow()\n            hello\n\n            Get the result from a flow state\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     return \"hello\"\n            >>> my_flow(return_state=True).result()\n            hello\n\n            Get the result from a failed state\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     raise ValueError(\"oh no!\")\n            >>> state = my_flow(return_state=True)  # Error is wrapped in FAILED state\n            >>> state.result()  # Raises `ValueError`\n\n            Get the result from a failed state without erroring\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     raise ValueError(\"oh no!\")\n            >>> state = my_flow(return_state=True)\n            >>> result = state.result(raise_on_failure=False)\n            >>> print(result)\n            ValueError(\"oh no!\")\n\n\n            Get the result from a flow state in an async context\n\n            >>> @flow\n            >>> async def my_flow():\n            >>>     return \"hello\"\n            >>> state = await my_flow(return_state=True)\n            >>> await state.result()\n            hello\n        \"\"\"\n        from prefect.states import get_state_result\n\n        return get_state_result(self, raise_on_failure=raise_on_failure, fetch=fetch)\n\n    def to_state_create(self):\n        \"\"\"\n        Convert this state to a `StateCreate` type which can be used to set the state of\n        a run in the API.\n\n        This method will drop this state's `data` if it is not a result type. Only\n        results should be sent to the API. Other data is only available locally.\n        \"\"\"\n        from prefect.client.schemas.actions import StateCreate\n        from prefect.results import BaseResult\n\n        return StateCreate(\n            type=self.type,\n            name=self.name,\n            message=self.message,\n            data=self.data if isinstance(self.data, BaseResult) else None,\n            state_details=self.state_details,\n        )\n\n    @validator(\"name\", always=True)\n    def default_name_from_type(cls, v, *, values, **kwargs):\n        \"\"\"If a name is not provided, use the type\"\"\"\n\n        # if `type` is not in `values` it means the `type` didn't pass its own\n        # validation check and an error will be raised after this function is called\n        if v is None and values.get(\"type\"):\n            v = \" \".join([v.capitalize() for v in values.get(\"type\").value.split(\"_\")])\n        return v\n\n    @root_validator\n    def default_scheduled_start_time(cls, values):\n        \"\"\"\n        TODO: This should throw an error instead of setting a default but is out of\n              scope for https://github.com/PrefectHQ/orion/pull/174/ and can be rolled\n              into work refactoring state initialization\n        \"\"\"\n        if values.get(\"type\") == StateType.SCHEDULED:\n            state_details = values.setdefault(\n                \"state_details\", cls.__fields__[\"state_details\"].get_default()\n            )\n            if not state_details.scheduled_time:\n                state_details.scheduled_time = pendulum.now(\"utc\")\n        return values\n\n    def is_scheduled(self) -> bool:\n        return self.type == StateType.SCHEDULED\n\n    def is_pending(self) -> bool:\n        return self.type == StateType.PENDING\n\n    def is_running(self) -> bool:\n        return self.type == StateType.RUNNING\n\n    def is_completed(self) -> bool:\n        return self.type == StateType.COMPLETED\n\n    def is_failed(self) -> bool:\n        return self.type == StateType.FAILED\n\n    def is_crashed(self) -> bool:\n        return self.type == StateType.CRASHED\n\n    def is_cancelled(self) -> bool:\n        return self.type == StateType.CANCELLED\n\n    def is_cancelling(self) -> bool:\n        return self.type == StateType.CANCELLING\n\n    def is_final(self) -> bool:\n        return self.type in {\n            StateType.CANCELLED,\n            StateType.FAILED,\n            StateType.COMPLETED,\n            StateType.CRASHED,\n        }\n\n    def is_paused(self) -> bool:\n        return self.type == StateType.PAUSED\n\n    def copy(self, *, update: dict = None, reset_fields: bool = False, **kwargs):\n        \"\"\"\n        Copying API models should return an object that could be inserted into the\n        database again. The 'timestamp' is reset using the default factory.\n        \"\"\"\n        update = update or {}\n        update.setdefault(\"timestamp\", self.__fields__[\"timestamp\"].get_default())\n        return super().copy(reset_fields=reset_fields, update=update, **kwargs)\n\n    def __repr__(self) -> str:\n        \"\"\"\n        Generates a complete state representation appropriate for introspection\n        and debugging, including the result:\n\n        `MyCompletedState(message=\"my message\", type=COMPLETED, result=...)`\n        \"\"\"\n        from prefect.deprecated.data_documents import DataDocument\n\n        if isinstance(self.data, DataDocument):\n            result = self.data.decode()\n        else:\n            result = self.data\n\n        display = dict(\n            message=repr(self.message),\n            type=str(self.type.value),\n            result=repr(result),\n        )\n\n        return f\"{self.name}({', '.join(f'{k}={v}' for k, v in display.items())})\"\n\n    def __str__(self) -> str:\n        \"\"\"\n        Generates a simple state representation appropriate for logging:\n\n        `MyCompletedState(\"my message\", type=COMPLETED)`\n        \"\"\"\n\n        display = []\n\n        if self.message:\n            display.append(repr(self.message))\n\n        if self.type.value.lower() != self.name.lower():\n            display.append(f\"type={self.type.value}\")\n\n        return f\"{self.name}({', '.join(display)})\"\n\n    def __hash__(self) -> int:\n        return hash(\n            (\n                getattr(self.state_details, \"flow_run_id\", None),\n                getattr(self.state_details, \"task_run_id\", None),\n                self.timestamp,\n                self.type,\n            )\n        )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State.result","title":"result","text":"

    Retrieve the result attached to this state.

    Parameters:

    Name Type Description Default raise_on_failure bool

    a boolean specifying whether to raise an exception if the state is of type FAILED and the underlying data is an exception

    True fetch Optional[bool]

    a boolean specifying whether to resolve references to persisted results into data. For synchronous users, this defaults to True. For asynchronous users, this defaults to False for backwards compatibility.

    None

    Raises:

    Type Description TypeError

    If the state is failed but the result is not an exception.

    Returns:

    Type Description

    The result of the run

    Examples:

    >>> from prefect import flow, task\n>>> @task\n>>> def my_task(x):\n>>>     return x\n

    Get the result from a task future in a flow

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task(\"hello\")\n>>>     state = future.wait()\n>>>     result = state.result()\n>>>     print(result)\n>>> my_flow()\nhello\n

    Get the result from a flow state

    >>> @flow\n>>> def my_flow():\n>>>     return \"hello\"\n>>> my_flow(return_state=True).result()\nhello\n

    Get the result from a failed state

    >>> @flow\n>>> def my_flow():\n>>>     raise ValueError(\"oh no!\")\n>>> state = my_flow(return_state=True)  # Error is wrapped in FAILED state\n>>> state.result()  # Raises `ValueError`\n

    Get the result from a failed state without erroring

    >>> @flow\n>>> def my_flow():\n>>>     raise ValueError(\"oh no!\")\n>>> state = my_flow(return_state=True)\n>>> result = state.result(raise_on_failure=False)\n>>> print(result)\nValueError(\"oh no!\")\n

    Get the result from a flow state in an async context

    >>> @flow\n>>> async def my_flow():\n>>>     return \"hello\"\n>>> state = await my_flow(return_state=True)\n>>> await state.result()\nhello\n
    Source code in prefect/client/schemas/objects.py
    def result(self, raise_on_failure: bool = True, fetch: Optional[bool] = None):\n    \"\"\"\n    Retrieve the result attached to this state.\n\n    Args:\n        raise_on_failure: a boolean specifying whether to raise an exception\n            if the state is of type `FAILED` and the underlying data is an exception\n        fetch: a boolean specifying whether to resolve references to persisted\n            results into data. For synchronous users, this defaults to `True`.\n            For asynchronous users, this defaults to `False` for backwards\n            compatibility.\n\n    Raises:\n        TypeError: If the state is failed but the result is not an exception.\n\n    Returns:\n        The result of the run\n\n    Examples:\n        >>> from prefect import flow, task\n        >>> @task\n        >>> def my_task(x):\n        >>>     return x\n\n        Get the result from a task future in a flow\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task(\"hello\")\n        >>>     state = future.wait()\n        >>>     result = state.result()\n        >>>     print(result)\n        >>> my_flow()\n        hello\n\n        Get the result from a flow state\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     return \"hello\"\n        >>> my_flow(return_state=True).result()\n        hello\n\n        Get the result from a failed state\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     raise ValueError(\"oh no!\")\n        >>> state = my_flow(return_state=True)  # Error is wrapped in FAILED state\n        >>> state.result()  # Raises `ValueError`\n\n        Get the result from a failed state without erroring\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     raise ValueError(\"oh no!\")\n        >>> state = my_flow(return_state=True)\n        >>> result = state.result(raise_on_failure=False)\n        >>> print(result)\n        ValueError(\"oh no!\")\n\n\n        Get the result from a flow state in an async context\n\n        >>> @flow\n        >>> async def my_flow():\n        >>>     return \"hello\"\n        >>> state = await my_flow(return_state=True)\n        >>> await state.result()\n        hello\n    \"\"\"\n    from prefect.states import get_state_result\n\n    return get_state_result(self, raise_on_failure=raise_on_failure, fetch=fetch)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State.to_state_create","title":"to_state_create","text":"

    Convert this state to a StateCreate type which can be used to set the state of a run in the API.

    This method will drop this state's data if it is not a result type. Only results should be sent to the API. Other data is only available locally.

    Source code in prefect/client/schemas/objects.py
    def to_state_create(self):\n    \"\"\"\n    Convert this state to a `StateCreate` type which can be used to set the state of\n    a run in the API.\n\n    This method will drop this state's `data` if it is not a result type. Only\n    results should be sent to the API. Other data is only available locally.\n    \"\"\"\n    from prefect.client.schemas.actions import StateCreate\n    from prefect.results import BaseResult\n\n    return StateCreate(\n        type=self.type,\n        name=self.name,\n        message=self.message,\n        data=self.data if isinstance(self.data, BaseResult) else None,\n        state_details=self.state_details,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State.default_name_from_type","title":"default_name_from_type","text":"

    If a name is not provided, use the type

    Source code in prefect/client/schemas/objects.py
    @validator(\"name\", always=True)\ndef default_name_from_type(cls, v, *, values, **kwargs):\n    \"\"\"If a name is not provided, use the type\"\"\"\n\n    # if `type` is not in `values` it means the `type` didn't pass its own\n    # validation check and an error will be raised after this function is called\n    if v is None and values.get(\"type\"):\n        v = \" \".join([v.capitalize() for v in values.get(\"type\").value.split(\"_\")])\n    return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State.default_scheduled_start_time","title":"default_scheduled_start_time","text":"This should throw an error instead of setting a default but is out of

    scope for https://github.com/PrefectHQ/orion/pull/174/ and can be rolled into work refactoring state initialization

    Source code in prefect/client/schemas/objects.py
    @root_validator\ndef default_scheduled_start_time(cls, values):\n    \"\"\"\n    TODO: This should throw an error instead of setting a default but is out of\n          scope for https://github.com/PrefectHQ/orion/pull/174/ and can be rolled\n          into work refactoring state initialization\n    \"\"\"\n    if values.get(\"type\") == StateType.SCHEDULED:\n        state_details = values.setdefault(\n            \"state_details\", cls.__fields__[\"state_details\"].get_default()\n        )\n        if not state_details.scheduled_time:\n            state_details.scheduled_time = pendulum.now(\"utc\")\n    return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunPolicy","title":"FlowRunPolicy","text":"

    Bases: PrefectBaseModel

    Defines of how a flow run should be orchestrated.

    Source code in prefect/client/schemas/objects.py
    class FlowRunPolicy(PrefectBaseModel):\n    \"\"\"Defines of how a flow run should be orchestrated.\"\"\"\n\n    max_retries: int = Field(\n        default=0,\n        description=(\n            \"The maximum number of retries. Field is not used. Please use `retries`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retry_delay_seconds: float = Field(\n        default=0,\n        description=(\n            \"The delay between retries. Field is not used. Please use `retry_delay`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retries: Optional[int] = Field(default=None, description=\"The number of retries.\")\n    retry_delay: Optional[int] = Field(\n        default=None, description=\"The delay time between retries, in seconds.\"\n    )\n    pause_keys: Optional[set] = Field(\n        default_factory=set, description=\"Tracks pauses this run has observed.\"\n    )\n    resuming: Optional[bool] = Field(\n        default=False, description=\"Indicates if this run is resuming from a pause.\"\n    )\n\n    @root_validator\n    def populate_deprecated_fields(cls, values):\n        \"\"\"\n        If deprecated fields are provided, populate the corresponding new fields\n        to preserve orchestration behavior.\n        \"\"\"\n        if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n            values[\"retries\"] = values[\"max_retries\"]\n        if (\n            not values.get(\"retry_delay\", None)\n            and values.get(\"retry_delay_seconds\", 0) != 0\n        ):\n            values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n        return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunPolicy.populate_deprecated_fields","title":"populate_deprecated_fields","text":"

    If deprecated fields are provided, populate the corresponding new fields to preserve orchestration behavior.

    Source code in prefect/client/schemas/objects.py
    @root_validator\ndef populate_deprecated_fields(cls, values):\n    \"\"\"\n    If deprecated fields are provided, populate the corresponding new fields\n    to preserve orchestration behavior.\n    \"\"\"\n    if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n        values[\"retries\"] = values[\"max_retries\"]\n    if (\n        not values.get(\"retry_delay\", None)\n        and values.get(\"retry_delay_seconds\", 0) != 0\n    ):\n        values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n    return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRun","title":"FlowRun","text":"

    Bases: ObjectBaseModel

    Source code in prefect/client/schemas/objects.py
    class FlowRun(ObjectBaseModel):\n    name: str = Field(\n        default_factory=lambda: generate_slug(2),\n        description=(\n            \"The name of the flow run. Defaults to a random slug if not specified.\"\n        ),\n        example=\"my-flow-run\",\n    )\n    flow_id: UUID = Field(default=..., description=\"The id of the flow being run.\")\n    state_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the flow run's current state.\"\n    )\n    deployment_id: Optional[UUID] = Field(\n        default=None,\n        description=(\n            \"The id of the deployment associated with this flow run, if available.\"\n        ),\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None, description=\"The work queue that handled this flow run.\"\n    )\n    flow_version: Optional[str] = Field(\n        default=None,\n        description=\"The version of the flow executed in this flow run.\",\n        example=\"1.0\",\n    )\n    parameters: dict = Field(\n        default_factory=dict, description=\"Parameters for the flow run.\"\n    )\n    idempotency_key: Optional[str] = Field(\n        default=None,\n        description=(\n            \"An optional idempotency key for the flow run. Used to ensure the same flow\"\n            \" run is not created multiple times.\"\n        ),\n    )\n    context: dict = Field(\n        default_factory=dict,\n        description=\"Additional context for the flow run.\",\n        example={\"my_var\": \"my_val\"},\n    )\n    empirical_policy: FlowRunPolicy = Field(\n        default_factory=FlowRunPolicy,\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags on the flow run\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    parent_task_run_id: Optional[UUID] = Field(\n        default=None,\n        description=(\n            \"If the flow run is a subflow, the id of the 'dummy' task in the parent\"\n            \" flow used to track subflow state.\"\n        ),\n    )\n    run_count: int = Field(\n        default=0, description=\"The number of times the flow run was executed.\"\n    )\n    expected_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The flow run's expected start time.\",\n    )\n    next_scheduled_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The next time the flow run is scheduled to start.\",\n    )\n    start_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual start time.\"\n    )\n    end_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual end time.\"\n    )\n    total_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=(\n            \"Total run time. If the flow run was executed multiple times, the time of\"\n            \" each run will be summed.\"\n        ),\n    )\n    estimated_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"A real-time estimate of the total run time.\",\n    )\n    estimated_start_time_delta: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"The difference between actual and expected start time.\",\n    )\n    auto_scheduled: bool = Field(\n        default=False,\n        description=\"Whether or not the flow run was automatically scheduled.\",\n    )\n    infrastructure_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining infrastructure to use this flow run.\",\n    )\n    infrastructure_pid: Optional[str] = Field(\n        default=None,\n        description=\"The id of the flow run as returned by an infrastructure block.\",\n    )\n    created_by: Optional[CreatedBy] = Field(\n        default=None,\n        description=\"Optional information about the creator of this flow run.\",\n    )\n    work_queue_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the run's work pool queue.\"\n    )\n\n    work_pool_id: Optional[UUID] = Field(\n        description=\"The work pool with which the queue is associated.\"\n    )\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the flow run's work pool.\",\n        example=\"my-work-pool\",\n    )\n    state: Optional[State] = Field(\n        default=None,\n        description=\"The state of the flow run.\",\n        example=State(type=StateType.COMPLETED),\n    )\n\n    def __eq__(self, other: Any) -> bool:\n        \"\"\"\n        Check for \"equality\" to another flow run schema\n\n        Estimates times are rolling and will always change with repeated queries for\n        a flow run so we ignore them during equality checks.\n        \"\"\"\n        if isinstance(other, FlowRun):\n            exclude_fields = {\"estimated_run_time\", \"estimated_start_time_delta\"}\n            return self.dict(exclude=exclude_fields) == other.dict(\n                exclude=exclude_fields\n            )\n        return super().__eq__(other)\n\n    @validator(\"name\", pre=True)\n    def set_default_name(cls, name):\n        return name or generate_slug(2)\n\n    # These are server-side optimizations and should not be present on client models\n    # TODO: Deprecate these fields\n\n    state_type: Optional[StateType] = Field(\n        default=None, description=\"The type of the current flow run state.\"\n    )\n    state_name: Optional[str] = Field(\n        default=None, description=\"The name of the current flow run state.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.TaskRunPolicy","title":"TaskRunPolicy","text":"

    Bases: PrefectBaseModel

    Defines of how a task run should retry.

    Source code in prefect/client/schemas/objects.py
    class TaskRunPolicy(PrefectBaseModel):\n    \"\"\"Defines of how a task run should retry.\"\"\"\n\n    max_retries: int = Field(\n        default=0,\n        description=(\n            \"The maximum number of retries. Field is not used. Please use `retries`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retry_delay_seconds: float = Field(\n        default=0,\n        description=(\n            \"The delay between retries. Field is not used. Please use `retry_delay`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retries: Optional[int] = Field(default=None, description=\"The number of retries.\")\n    retry_delay: Union[None, int, List[int]] = Field(\n        default=None,\n        description=\"A delay time or list of delay times between retries, in seconds.\",\n    )\n    retry_jitter_factor: Optional[float] = Field(\n        default=None, description=\"Determines the amount a retry should jitter\"\n    )\n\n    @root_validator\n    def populate_deprecated_fields(cls, values):\n        \"\"\"\n        If deprecated fields are provided, populate the corresponding new fields\n        to preserve orchestration behavior.\n        \"\"\"\n        if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n            values[\"retries\"] = values[\"max_retries\"]\n\n        if (\n            not values.get(\"retry_delay\", None)\n            and values.get(\"retry_delay_seconds\", 0) != 0\n        ):\n            values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n\n        return values\n\n    @validator(\"retry_delay\")\n    def validate_configured_retry_delays(cls, v):\n        if isinstance(v, list) and (len(v) > 50):\n            raise ValueError(\"Can not configure more than 50 retry delays per task.\")\n        return v\n\n    @validator(\"retry_jitter_factor\")\n    def validate_jitter_factor(cls, v):\n        if v is not None and v < 0:\n            raise ValueError(\"`retry_jitter_factor` must be >= 0.\")\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.TaskRunPolicy.populate_deprecated_fields","title":"populate_deprecated_fields","text":"

    If deprecated fields are provided, populate the corresponding new fields to preserve orchestration behavior.

    Source code in prefect/client/schemas/objects.py
    @root_validator\ndef populate_deprecated_fields(cls, values):\n    \"\"\"\n    If deprecated fields are provided, populate the corresponding new fields\n    to preserve orchestration behavior.\n    \"\"\"\n    if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n        values[\"retries\"] = values[\"max_retries\"]\n\n    if (\n        not values.get(\"retry_delay\", None)\n        and values.get(\"retry_delay_seconds\", 0) != 0\n    ):\n        values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n\n    return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.TaskRunInput","title":"TaskRunInput","text":"

    Bases: PrefectBaseModel

    Base class for classes that represent inputs to task runs, which could include, constants, parameters, or other task runs.

    Source code in prefect/client/schemas/objects.py
    class TaskRunInput(PrefectBaseModel):\n    \"\"\"\n    Base class for classes that represent inputs to task runs, which\n    could include, constants, parameters, or other task runs.\n    \"\"\"\n\n    # freeze TaskRunInputs to allow them to be placed in sets\n    class Config:\n        frozen = True\n\n    input_type: str\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.TaskRunResult","title":"TaskRunResult","text":"

    Bases: TaskRunInput

    Represents a task run result input to another task run.

    Source code in prefect/client/schemas/objects.py
    class TaskRunResult(TaskRunInput):\n    \"\"\"Represents a task run result input to another task run.\"\"\"\n\n    input_type: Literal[\"task_run\"] = \"task_run\"\n    id: UUID\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Parameter","title":"Parameter","text":"

    Bases: TaskRunInput

    Represents a parameter input to a task run.

    Source code in prefect/client/schemas/objects.py
    class Parameter(TaskRunInput):\n    \"\"\"Represents a parameter input to a task run.\"\"\"\n\n    input_type: Literal[\"parameter\"] = \"parameter\"\n    name: str\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Constant","title":"Constant","text":"

    Bases: TaskRunInput

    Represents constant input value to a task run.

    Source code in prefect/client/schemas/objects.py
    class Constant(TaskRunInput):\n    \"\"\"Represents constant input value to a task run.\"\"\"\n\n    input_type: Literal[\"constant\"] = \"constant\"\n    type: str\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Workspace","title":"Workspace","text":"

    Bases: PrefectBaseModel

    A Prefect Cloud workspace.

    Expected payload for each workspace returned by the me/workspaces route.

    Source code in prefect/client/schemas/objects.py
    class Workspace(PrefectBaseModel):\n    \"\"\"\n    A Prefect Cloud workspace.\n\n    Expected payload for each workspace returned by the `me/workspaces` route.\n    \"\"\"\n\n    account_id: UUID = Field(..., description=\"The account id of the workspace.\")\n    account_name: str = Field(..., description=\"The account name.\")\n    account_handle: str = Field(..., description=\"The account's unique handle.\")\n    workspace_id: UUID = Field(..., description=\"The workspace id.\")\n    workspace_name: str = Field(..., description=\"The workspace name.\")\n    workspace_description: str = Field(..., description=\"Description of the workspace.\")\n    workspace_handle: str = Field(..., description=\"The workspace's unique handle.\")\n\n    class Config:\n        extra = \"ignore\"\n\n    @property\n    def handle(self) -> str:\n        \"\"\"\n        The full handle of the workspace as `account_handle` / `workspace_handle`\n        \"\"\"\n        return self.account_handle + \"/\" + self.workspace_handle\n\n    def api_url(self) -> str:\n        \"\"\"\n        Generate the API URL for accessing this workspace\n        \"\"\"\n        return (\n            f\"{PREFECT_CLOUD_API_URL.value()}\"\n            f\"/accounts/{self.account_id}\"\n            f\"/workspaces/{self.workspace_id}\"\n        )\n\n    def ui_url(self) -> str:\n        \"\"\"\n        Generate the UI URL for accessing this workspace\n        \"\"\"\n        return (\n            f\"{PREFECT_CLOUD_UI_URL.value()}\"\n            f\"/account/{self.account_id}\"\n            f\"/workspace/{self.workspace_id}\"\n        )\n\n    def __hash__(self):\n        return hash(self.handle)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Workspace.handle","title":"handle: str property","text":"

    The full handle of the workspace as account_handle / workspace_handle

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Workspace.api_url","title":"api_url","text":"

    Generate the API URL for accessing this workspace

    Source code in prefect/client/schemas/objects.py
    def api_url(self) -> str:\n    \"\"\"\n    Generate the API URL for accessing this workspace\n    \"\"\"\n    return (\n        f\"{PREFECT_CLOUD_API_URL.value()}\"\n        f\"/accounts/{self.account_id}\"\n        f\"/workspaces/{self.workspace_id}\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Workspace.ui_url","title":"ui_url","text":"

    Generate the UI URL for accessing this workspace

    Source code in prefect/client/schemas/objects.py
    def ui_url(self) -> str:\n    \"\"\"\n    Generate the UI URL for accessing this workspace\n    \"\"\"\n    return (\n        f\"{PREFECT_CLOUD_UI_URL.value()}\"\n        f\"/account/{self.account_id}\"\n        f\"/workspace/{self.workspace_id}\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockType","title":"BlockType","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block type

    Source code in prefect/client/schemas/objects.py
    class BlockType(ObjectBaseModel):\n    \"\"\"An ORM representation of a block type\"\"\"\n\n    name: str = Field(default=..., description=\"A block type's name\")\n    slug: str = Field(default=..., description=\"A block type's slug\")\n    logo_url: Optional[HttpUrl] = Field(\n        default=None, description=\"Web URL for the block type's logo\"\n    )\n    documentation_url: Optional[HttpUrl] = Field(\n        default=None, description=\"Web URL for the block type's documentation\"\n    )\n    description: Optional[str] = Field(\n        default=None,\n        description=\"A short blurb about the corresponding block's intended use\",\n    )\n    code_example: Optional[str] = Field(\n        default=None,\n        description=\"A code snippet demonstrating use of the corresponding block\",\n    )\n    is_protected: bool = Field(\n        default=False, description=\"Protected block types cannot be modified via API.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockDocument","title":"BlockDocument","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block document.

    Source code in prefect/client/schemas/objects.py
    class BlockDocument(ObjectBaseModel):\n    \"\"\"An ORM representation of a block document.\"\"\"\n\n    name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The block document's name. Not required for anonymous block documents.\"\n        ),\n    )\n    data: dict = Field(default_factory=dict, description=\"The block document's data\")\n    block_schema_id: UUID = Field(default=..., description=\"A block schema ID\")\n    block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The associated block schema\"\n    )\n    block_type_id: UUID = Field(default=..., description=\"A block type ID\")\n    block_type_name: Optional[str] = Field(None, description=\"A block type name\")\n    block_type: Optional[BlockType] = Field(\n        default=None, description=\"The associated block type\"\n    )\n    block_document_references: Dict[str, Dict[str, Any]] = Field(\n        default_factory=dict, description=\"Record of the block document's references\"\n    )\n    is_anonymous: bool = Field(\n        default=False,\n        description=(\n            \"Whether the block is anonymous (anonymous blocks are usually created by\"\n            \" Prefect automatically)\"\n        ),\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        # the BlockDocumentCreate subclass allows name=None\n        # and will inherit this validator\n        if v is not None:\n            raise_on_name_with_banned_characters(v)\n        return v\n\n    @root_validator\n    def validate_name_is_present_if_not_anonymous(cls, values):\n        # anonymous blocks may have no name prior to actually being\n        # stored in the database\n        if not values.get(\"is_anonymous\") and not values.get(\"name\"):\n            raise ValueError(\"Names must be provided for block documents.\")\n        return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Flow","title":"Flow","text":"

    Bases: ObjectBaseModel

    An ORM representation of flow data.

    Source code in prefect/client/schemas/objects.py
    class Flow(ObjectBaseModel):\n    \"\"\"An ORM representation of flow data.\"\"\"\n\n    name: str = Field(\n        default=..., description=\"The name of the flow\", example=\"my-flow\"\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of flow tags\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunnerSettings","title":"FlowRunnerSettings","text":"

    Bases: PrefectBaseModel

    An API schema for passing details about the flow runner.

    This schema is agnostic to the types and configuration provided by clients

    Source code in prefect/client/schemas/objects.py
    class FlowRunnerSettings(PrefectBaseModel):\n    \"\"\"\n    An API schema for passing details about the flow runner.\n\n    This schema is agnostic to the types and configuration provided by clients\n    \"\"\"\n\n    type: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The type of the flow runner which can be used by the client for\"\n            \" dispatching.\"\n        ),\n    )\n    config: Optional[dict] = Field(\n        default=None, description=\"The configuration for the given flow runner type.\"\n    )\n\n    # The following is required for composite compatibility in the ORM\n\n    def __init__(self, type: str = None, config: dict = None, **kwargs) -> None:\n        # Pydantic does not support positional arguments so they must be converted to\n        # keyword arguments\n        super().__init__(type=type, config=config, **kwargs)\n\n    def __composite_values__(self):\n        return self.type, self.config\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Deployment","title":"Deployment","text":"

    Bases: ObjectBaseModel

    An ORM representation of deployment data.

    Source code in prefect/client/schemas/objects.py
    class Deployment(ObjectBaseModel):\n    \"\"\"An ORM representation of deployment data.\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the deployment.\")\n    version: Optional[str] = Field(\n        default=None, description=\"An optional version for the deployment.\"\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"A description for the deployment.\"\n    )\n    flow_id: UUID = Field(\n        default=..., description=\"The flow id associated with the deployment.\"\n    )\n    schedule: Optional[SCHEDULE_TYPES] = Field(\n        default=None, description=\"A schedule for the deployment.\"\n    )\n    is_schedule_active: bool = Field(\n        default=True, description=\"Whether or not the deployment schedule is active.\"\n    )\n    infra_overrides: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Overrides to apply to the base infrastructure block at runtime.\",\n    )\n    parameters: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Parameters for flow runs scheduled by the deployment.\",\n    )\n    pull_steps: Optional[List[dict]] = Field(\n        default=None,\n        description=\"Pull steps for cloning and running this deployment.\",\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags for the deployment\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The work queue for the deployment. If no work queue is set, work will not\"\n            \" be scheduled.\"\n        ),\n    )\n    last_polled: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The last time the deployment was polled for status updates.\",\n    )\n    parameter_openapi_schema: Optional[Dict[str, Any]] = Field(\n        default=None,\n        description=\"The parameter schema of the flow, including defaults.\",\n    )\n    path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the working directory for the workflow, relative to remote\"\n            \" storage or an absolute path.\"\n        ),\n    )\n    entrypoint: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the entrypoint for the workflow, relative to the `path`.\"\n        ),\n    )\n    manifest_path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the flow's manifest file, relative to the chosen storage.\"\n        ),\n    )\n    storage_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining storage used for this flow.\",\n    )\n    infrastructure_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining infrastructure to use for flow runs.\",\n    )\n    created_by: Optional[CreatedBy] = Field(\n        default=None,\n        description=\"Optional information about the creator of this deployment.\",\n    )\n    updated_by: Optional[UpdatedBy] = Field(\n        default=None,\n        description=\"Optional information about the updater of this deployment.\",\n    )\n    work_queue_id: UUID = Field(\n        default=None,\n        description=(\n            \"The id of the work pool queue to which this deployment is assigned.\"\n        ),\n    )\n    enforce_parameter_schema: bool = Field(\n        default=False,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.ConcurrencyLimit","title":"ConcurrencyLimit","text":"

    Bases: ObjectBaseModel

    An ORM representation of a concurrency limit.

    Source code in prefect/client/schemas/objects.py
    class ConcurrencyLimit(ObjectBaseModel):\n    \"\"\"An ORM representation of a concurrency limit.\"\"\"\n\n    tag: str = Field(\n        default=..., description=\"A tag the concurrency limit is applied to.\"\n    )\n    concurrency_limit: int = Field(default=..., description=\"The concurrency limit.\")\n    active_slots: List[UUID] = Field(\n        default_factory=list,\n        description=\"A list of active run ids using a concurrency slot\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockSchema","title":"BlockSchema","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block schema.

    Source code in prefect/client/schemas/objects.py
    class BlockSchema(ObjectBaseModel):\n    \"\"\"An ORM representation of a block schema.\"\"\"\n\n    checksum: str = Field(default=..., description=\"The block schema's unique checksum\")\n    fields: dict = Field(\n        default_factory=dict, description=\"The block schema's field schema\"\n    )\n    block_type_id: Optional[UUID] = Field(default=..., description=\"A block type ID\")\n    block_type: Optional[BlockType] = Field(\n        default=None, description=\"The associated block type\"\n    )\n    capabilities: List[str] = Field(\n        default_factory=list,\n        description=\"A list of Block capabilities\",\n    )\n    version: str = Field(\n        default=DEFAULT_BLOCK_SCHEMA_VERSION,\n        description=\"Human readable identifier for the block schema\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockSchemaReference","title":"BlockSchemaReference","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block schema reference.

    Source code in prefect/client/schemas/objects.py
    class BlockSchemaReference(ObjectBaseModel):\n    \"\"\"An ORM representation of a block schema reference.\"\"\"\n\n    parent_block_schema_id: UUID = Field(\n        default=..., description=\"ID of block schema the reference is nested within\"\n    )\n    parent_block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The block schema the reference is nested within\"\n    )\n    reference_block_schema_id: UUID = Field(\n        default=..., description=\"ID of the nested block schema\"\n    )\n    reference_block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The nested block schema\"\n    )\n    name: str = Field(\n        default=..., description=\"The name that the reference is nested under\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockDocumentReference","title":"BlockDocumentReference","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block document reference.

    Source code in prefect/client/schemas/objects.py
    class BlockDocumentReference(ObjectBaseModel):\n    \"\"\"An ORM representation of a block document reference.\"\"\"\n\n    parent_block_document_id: UUID = Field(\n        default=..., description=\"ID of block document the reference is nested within\"\n    )\n    parent_block_document: Optional[BlockDocument] = Field(\n        default=None, description=\"The block document the reference is nested within\"\n    )\n    reference_block_document_id: UUID = Field(\n        default=..., description=\"ID of the nested block document\"\n    )\n    reference_block_document: Optional[BlockDocument] = Field(\n        default=None, description=\"The nested block document\"\n    )\n    name: str = Field(\n        default=..., description=\"The name that the reference is nested under\"\n    )\n\n    @root_validator\n    def validate_parent_and_ref_are_different(cls, values):\n        parent_id = values.get(\"parent_block_document_id\")\n        ref_id = values.get(\"reference_block_document_id\")\n        if parent_id and ref_id and parent_id == ref_id:\n            raise ValueError(\n                \"`parent_block_document_id` and `reference_block_document_id` cannot be\"\n                \" the same\"\n            )\n        return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.SavedSearchFilter","title":"SavedSearchFilter","text":"

    Bases: PrefectBaseModel

    A filter for a saved search model. Intended for use by the Prefect UI.

    Source code in prefect/client/schemas/objects.py
    class SavedSearchFilter(PrefectBaseModel):\n    \"\"\"A filter for a saved search model. Intended for use by the Prefect UI.\"\"\"\n\n    object: str = Field(default=..., description=\"The object over which to filter.\")\n    property: str = Field(\n        default=..., description=\"The property of the object on which to filter.\"\n    )\n    type: str = Field(default=..., description=\"The type of the property.\")\n    operation: str = Field(\n        default=...,\n        description=\"The operator to apply to the object. For example, `equals`.\",\n    )\n    value: Any = Field(\n        default=..., description=\"A JSON-compatible value for the filter.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.SavedSearch","title":"SavedSearch","text":"

    Bases: ObjectBaseModel

    An ORM representation of saved search data. Represents a set of filter criteria.

    Source code in prefect/client/schemas/objects.py
    class SavedSearch(ObjectBaseModel):\n    \"\"\"An ORM representation of saved search data. Represents a set of filter criteria.\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the saved search.\")\n    filters: List[SavedSearchFilter] = Field(\n        default_factory=list, description=\"The filter set for the saved search.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Log","title":"Log","text":"

    Bases: ObjectBaseModel

    An ORM representation of log data.

    Source code in prefect/client/schemas/objects.py
    class Log(ObjectBaseModel):\n    \"\"\"An ORM representation of log data.\"\"\"\n\n    name: str = Field(default=..., description=\"The logger name.\")\n    level: int = Field(default=..., description=\"The log level.\")\n    message: str = Field(default=..., description=\"The log message.\")\n    timestamp: DateTimeTZ = Field(default=..., description=\"The log timestamp.\")\n    flow_run_id: Optional[UUID] = Field(\n        default=None, description=\"The flow run ID associated with the log.\"\n    )\n    task_run_id: Optional[UUID] = Field(\n        default=None, description=\"The task run ID associated with the log.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.QueueFilter","title":"QueueFilter","text":"

    Bases: PrefectBaseModel

    Filter criteria definition for a work queue.

    Source code in prefect/client/schemas/objects.py
    class QueueFilter(PrefectBaseModel):\n    \"\"\"Filter criteria definition for a work queue.\"\"\"\n\n    tags: Optional[List[str]] = Field(\n        default=None,\n        description=\"Only include flow runs with these tags in the work queue.\",\n    )\n    deployment_ids: Optional[List[UUID]] = Field(\n        default=None,\n        description=\"Only include flow runs from these deployments in the work queue.\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkQueue","title":"WorkQueue","text":"

    Bases: ObjectBaseModel

    An ORM representation of a work queue

    Source code in prefect/client/schemas/objects.py
    class WorkQueue(ObjectBaseModel):\n    \"\"\"An ORM representation of a work queue\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the work queue.\")\n    description: Optional[str] = Field(\n        default=\"\", description=\"An optional description for the work queue.\"\n    )\n    is_paused: bool = Field(\n        default=False, description=\"Whether or not the work queue is paused.\"\n    )\n    concurrency_limit: Optional[conint(ge=0)] = Field(\n        default=None, description=\"An optional concurrency limit for the work queue.\"\n    )\n    priority: conint(ge=1) = Field(\n        default=1,\n        description=(\n            \"The queue's priority. Lower values are higher priority (1 is the highest).\"\n        ),\n    )\n    work_pool_name: Optional[str] = Field(default=None)\n    # Will be required after a future migration\n    work_pool_id: Optional[UUID] = Field(\n        description=\"The work pool with which the queue is associated.\"\n    )\n    filter: Optional[QueueFilter] = Field(\n        default=None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n    last_polled: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The last time an agent polled this queue for work.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkQueueHealthPolicy","title":"WorkQueueHealthPolicy","text":"

    Bases: PrefectBaseModel

    Source code in prefect/client/schemas/objects.py
    class WorkQueueHealthPolicy(PrefectBaseModel):\n    maximum_late_runs: Optional[int] = Field(\n        default=0,\n        description=(\n            \"The maximum number of late runs in the work queue before it is deemed\"\n            \" unhealthy. Defaults to `0`.\"\n        ),\n    )\n    maximum_seconds_since_last_polled: Optional[int] = Field(\n        default=60,\n        description=(\n            \"The maximum number of time in seconds elapsed since work queue has been\"\n            \" polled before it is deemed unhealthy. Defaults to `60`.\"\n        ),\n    )\n\n    def evaluate_health_status(\n        self, late_runs_count: int, last_polled: Optional[DateTimeTZ] = None\n    ) -> bool:\n        \"\"\"\n        Given empirical information about the state of the work queue, evaluate its health status.\n\n        Args:\n            late_runs: the count of late runs for the work queue.\n            last_polled: the last time the work queue was polled, if available.\n\n        Returns:\n            bool: whether or not the work queue is healthy.\n        \"\"\"\n        healthy = True\n        if (\n            self.maximum_late_runs is not None\n            and late_runs_count > self.maximum_late_runs\n        ):\n            healthy = False\n\n        if self.maximum_seconds_since_last_polled is not None:\n            if (\n                last_polled is None\n                or pendulum.now(\"UTC\").diff(last_polled).in_seconds()\n                > self.maximum_seconds_since_last_polled\n            ):\n                healthy = False\n\n        return healthy\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkQueueHealthPolicy.evaluate_health_status","title":"evaluate_health_status","text":"

    Given empirical information about the state of the work queue, evaluate its health status.

    Parameters:

    Name Type Description Default late_runs

    the count of late runs for the work queue.

    required last_polled Optional[DateTimeTZ]

    the last time the work queue was polled, if available.

    None

    Returns:

    Name Type Description bool bool

    whether or not the work queue is healthy.

    Source code in prefect/client/schemas/objects.py
    def evaluate_health_status(\n    self, late_runs_count: int, last_polled: Optional[DateTimeTZ] = None\n) -> bool:\n    \"\"\"\n    Given empirical information about the state of the work queue, evaluate its health status.\n\n    Args:\n        late_runs: the count of late runs for the work queue.\n        last_polled: the last time the work queue was polled, if available.\n\n    Returns:\n        bool: whether or not the work queue is healthy.\n    \"\"\"\n    healthy = True\n    if (\n        self.maximum_late_runs is not None\n        and late_runs_count > self.maximum_late_runs\n    ):\n        healthy = False\n\n    if self.maximum_seconds_since_last_polled is not None:\n        if (\n            last_polled is None\n            or pendulum.now(\"UTC\").diff(last_polled).in_seconds()\n            > self.maximum_seconds_since_last_polled\n        ):\n            healthy = False\n\n    return healthy\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunNotificationPolicy","title":"FlowRunNotificationPolicy","text":"

    Bases: ObjectBaseModel

    An ORM representation of a flow run notification.

    Source code in prefect/client/schemas/objects.py
    class FlowRunNotificationPolicy(ObjectBaseModel):\n    \"\"\"An ORM representation of a flow run notification.\"\"\"\n\n    is_active: bool = Field(\n        default=True, description=\"Whether the policy is currently active\"\n    )\n    state_names: List[str] = Field(\n        default=..., description=\"The flow run states that trigger notifications\"\n    )\n    tags: List[str] = Field(\n        default=...,\n        description=\"The flow run tags that trigger notifications (set [] to disable)\",\n    )\n    block_document_id: UUID = Field(\n        default=..., description=\"The block document ID used for sending notifications\"\n    )\n    message_template: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A templatable notification message. Use {braces} to add variables.\"\n            \" Valid variables include:\"\n            f\" {listrepr(sorted(FLOW_RUN_NOTIFICATION_TEMPLATE_KWARGS), sep=', ')}\"\n        ),\n        example=(\n            \"Flow run {flow_run_name} with id {flow_run_id} entered state\"\n            \" {flow_run_state_name}.\"\n        ),\n    )\n\n    @validator(\"message_template\")\n    def validate_message_template_variables(cls, v):\n        if v is not None:\n            try:\n                v.format(**{k: \"test\" for k in FLOW_RUN_NOTIFICATION_TEMPLATE_KWARGS})\n            except KeyError as exc:\n                raise ValueError(f\"Invalid template variable provided: '{exc.args[0]}'\")\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Agent","title":"Agent","text":"

    Bases: ObjectBaseModel

    An ORM representation of an agent

    Source code in prefect/client/schemas/objects.py
    class Agent(ObjectBaseModel):\n    \"\"\"An ORM representation of an agent\"\"\"\n\n    name: str = Field(\n        default_factory=lambda: generate_slug(2),\n        description=(\n            \"The name of the agent. If a name is not provided, it will be\"\n            \" auto-generated.\"\n        ),\n    )\n    work_queue_id: UUID = Field(\n        default=..., description=\"The work queue with which the agent is associated.\"\n    )\n    last_activity_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The last time this agent polled for work.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkPool","title":"WorkPool","text":"

    Bases: ObjectBaseModel

    An ORM representation of a work pool

    Source code in prefect/client/schemas/objects.py
    class WorkPool(ObjectBaseModel):\n    \"\"\"An ORM representation of a work pool\"\"\"\n\n    name: str = Field(\n        description=\"The name of the work pool.\",\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"A description of the work pool.\"\n    )\n    type: str = Field(description=\"The work pool type.\")\n    base_job_template: Dict[str, Any] = Field(\n        default_factory=dict, description=\"The work pool's base job template.\"\n    )\n    is_paused: bool = Field(\n        default=False,\n        description=\"Pausing the work pool stops the delivery of all work.\",\n    )\n    concurrency_limit: Optional[conint(ge=0)] = Field(\n        default=None, description=\"A concurrency limit for the work pool.\"\n    )\n    status: Optional[WorkPoolStatus] = Field(\n        default=None, description=\"The current status of the work pool.\"\n    )\n\n    # this required field has a default of None so that the custom validator\n    # below will be called and produce a more helpful error message\n    default_queue_id: UUID = Field(\n        None, description=\"The id of the pool's default queue.\"\n    )\n\n    @property\n    def is_push_pool(self) -> bool:\n        return self.type.endswith(\":push\")\n\n    @property\n    def is_managed_pool(self) -> bool:\n        return self.type.endswith(\":managed\")\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n\n    @validator(\"default_queue_id\", always=True)\n    def helpful_error_for_missing_default_queue_id(cls, v):\n        \"\"\"\n        Default queue ID is required because all pools must have a default queue\n        ID, but it represents a circular foreign key relationship to a\n        WorkQueue (which can't be created until the work pool exists).\n        Therefore, while this field can *technically* be null, it shouldn't be.\n        This should only be an issue when creating new pools, as reading\n        existing ones will always have this field populated. This custom error\n        message will help users understand that they should use the\n        `actions.WorkPoolCreate` model in that case.\n        \"\"\"\n        if v is None:\n            raise ValueError(\n                \"`default_queue_id` is a required field. If you are \"\n                \"creating a new WorkPool and don't have a queue \"\n                \"ID yet, use the `actions.WorkPoolCreate` model instead.\"\n            )\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkPool.helpful_error_for_missing_default_queue_id","title":"helpful_error_for_missing_default_queue_id","text":"

    Default queue ID is required because all pools must have a default queue ID, but it represents a circular foreign key relationship to a WorkQueue (which can't be created until the work pool exists). Therefore, while this field can technically be null, it shouldn't be. This should only be an issue when creating new pools, as reading existing ones will always have this field populated. This custom error message will help users understand that they should use the actions.WorkPoolCreate model in that case.

    Source code in prefect/client/schemas/objects.py
    @validator(\"default_queue_id\", always=True)\ndef helpful_error_for_missing_default_queue_id(cls, v):\n    \"\"\"\n    Default queue ID is required because all pools must have a default queue\n    ID, but it represents a circular foreign key relationship to a\n    WorkQueue (which can't be created until the work pool exists).\n    Therefore, while this field can *technically* be null, it shouldn't be.\n    This should only be an issue when creating new pools, as reading\n    existing ones will always have this field populated. This custom error\n    message will help users understand that they should use the\n    `actions.WorkPoolCreate` model in that case.\n    \"\"\"\n    if v is None:\n        raise ValueError(\n            \"`default_queue_id` is a required field. If you are \"\n            \"creating a new WorkPool and don't have a queue \"\n            \"ID yet, use the `actions.WorkPoolCreate` model instead.\"\n        )\n    return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Worker","title":"Worker","text":"

    Bases: ObjectBaseModel

    An ORM representation of a worker

    Source code in prefect/client/schemas/objects.py
    class Worker(ObjectBaseModel):\n    \"\"\"An ORM representation of a worker\"\"\"\n\n    name: str = Field(description=\"The name of the worker.\")\n    work_pool_id: UUID = Field(\n        description=\"The work pool with which the queue is associated.\"\n    )\n    last_heartbeat_time: datetime.datetime = Field(\n        None, description=\"The last time the worker process sent a heartbeat.\"\n    )\n    heartbeat_interval_seconds: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The number of seconds to expect between heartbeats sent by the worker.\"\n        ),\n    )\n    status: WorkerStatus = Field(\n        WorkerStatus.OFFLINE,\n        description=\"Current status of the worker.\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunInput","title":"FlowRunInput","text":"

    Bases: ObjectBaseModel

    Source code in prefect/client/schemas/objects.py
    class FlowRunInput(ObjectBaseModel):\n    flow_run_id: UUID = Field(description=\"The flow run ID associated with the input.\")\n    key: str = Field(description=\"The key of the input.\")\n    value: str = Field(description=\"The value of the input.\")\n    sender: Optional[str] = Field(description=\"The sender of the input.\")\n\n    @property\n    def decoded_value(self) -> Any:\n        \"\"\"\n        Decode the value of the input.\n\n        Returns:\n            Any: the decoded value\n        \"\"\"\n        return orjson.loads(self.value)\n\n    @validator(\"key\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_alphanumeric_dashes_only(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunInput.decoded_value","title":"decoded_value: Any property","text":"

    Decode the value of the input.

    Returns:

    Name Type Description Any Any

    the decoded value

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_4","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses","title":"prefect.client.schemas.responses","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.SetStateStatus","title":"SetStateStatus","text":"

    Bases: AutoEnum

    Enumerates return statuses for setting run states.

    Source code in prefect/client/schemas/responses.py
    class SetStateStatus(AutoEnum):\n    \"\"\"Enumerates return statuses for setting run states.\"\"\"\n\n    ACCEPT = AutoEnum.auto()\n    REJECT = AutoEnum.auto()\n    ABORT = AutoEnum.auto()\n    WAIT = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.StateAcceptDetails","title":"StateAcceptDetails","text":"

    Bases: PrefectBaseModel

    Details associated with an ACCEPT state transition.

    Source code in prefect/client/schemas/responses.py
    class StateAcceptDetails(PrefectBaseModel):\n    \"\"\"Details associated with an ACCEPT state transition.\"\"\"\n\n    type: Literal[\"accept_details\"] = Field(\n        default=\"accept_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.StateRejectDetails","title":"StateRejectDetails","text":"

    Bases: PrefectBaseModel

    Details associated with a REJECT state transition.

    Source code in prefect/client/schemas/responses.py
    class StateRejectDetails(PrefectBaseModel):\n    \"\"\"Details associated with a REJECT state transition.\"\"\"\n\n    type: Literal[\"reject_details\"] = Field(\n        default=\"reject_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition was rejected.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.StateAbortDetails","title":"StateAbortDetails","text":"

    Bases: PrefectBaseModel

    Details associated with an ABORT state transition.

    Source code in prefect/client/schemas/responses.py
    class StateAbortDetails(PrefectBaseModel):\n    \"\"\"Details associated with an ABORT state transition.\"\"\"\n\n    type: Literal[\"abort_details\"] = Field(\n        default=\"abort_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition was aborted.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.StateWaitDetails","title":"StateWaitDetails","text":"

    Bases: PrefectBaseModel

    Details associated with a WAIT state transition.

    Source code in prefect/client/schemas/responses.py
    class StateWaitDetails(PrefectBaseModel):\n    \"\"\"Details associated with a WAIT state transition.\"\"\"\n\n    type: Literal[\"wait_details\"] = Field(\n        default=\"wait_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    delay_seconds: int = Field(\n        default=...,\n        description=(\n            \"The length of time in seconds the client should wait before transitioning\"\n            \" states.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition should wait.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.HistoryResponseState","title":"HistoryResponseState","text":"

    Bases: PrefectBaseModel

    Represents a single state's history over an interval.

    Source code in prefect/client/schemas/responses.py
    class HistoryResponseState(PrefectBaseModel):\n    \"\"\"Represents a single state's history over an interval.\"\"\"\n\n    state_type: objects.StateType = Field(default=..., description=\"The state type.\")\n    state_name: str = Field(default=..., description=\"The state name.\")\n    count_runs: int = Field(\n        default=...,\n        description=\"The number of runs in the specified state during the interval.\",\n    )\n    sum_estimated_run_time: datetime.timedelta = Field(\n        default=...,\n        description=\"The total estimated run time of all runs during the interval.\",\n    )\n    sum_estimated_lateness: datetime.timedelta = Field(\n        default=...,\n        description=(\n            \"The sum of differences between actual and expected start time during the\"\n            \" interval.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.HistoryResponse","title":"HistoryResponse","text":"

    Bases: PrefectBaseModel

    Represents a history of aggregation states over an interval

    Source code in prefect/client/schemas/responses.py
    class HistoryResponse(PrefectBaseModel):\n    \"\"\"Represents a history of aggregation states over an interval\"\"\"\n\n    interval_start: DateTimeTZ = Field(\n        default=..., description=\"The start date of the interval.\"\n    )\n    interval_end: DateTimeTZ = Field(\n        default=..., description=\"The end date of the interval.\"\n    )\n    states: List[HistoryResponseState] = Field(\n        default=..., description=\"A list of state histories during the interval.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.OrchestrationResult","title":"OrchestrationResult","text":"

    Bases: PrefectBaseModel

    A container for the output of state orchestration.

    Source code in prefect/client/schemas/responses.py
    class OrchestrationResult(PrefectBaseModel):\n    \"\"\"\n    A container for the output of state orchestration.\n    \"\"\"\n\n    state: Optional[objects.State]\n    status: SetStateStatus\n    details: StateResponseDetails\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.FlowRunResponse","title":"FlowRunResponse","text":"

    Bases: ObjectBaseModel

    Source code in prefect/client/schemas/responses.py
    @copy_model_fields\nclass FlowRunResponse(ObjectBaseModel):\n    name: str = FieldFrom(objects.FlowRun)\n    flow_id: UUID = FieldFrom(objects.FlowRun)\n    state_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    deployment_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    work_queue_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    work_queue_name: Optional[str] = FieldFrom(objects.FlowRun)\n    flow_version: Optional[str] = FieldFrom(objects.FlowRun)\n    parameters: dict = FieldFrom(objects.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(objects.FlowRun)\n    context: dict = FieldFrom(objects.FlowRun)\n    empirical_policy: objects.FlowRunPolicy = FieldFrom(objects.FlowRun)\n    tags: List[str] = FieldFrom(objects.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    state_type: Optional[objects.StateType] = FieldFrom(objects.FlowRun)\n    state_name: Optional[str] = FieldFrom(objects.FlowRun)\n    run_count: int = FieldFrom(objects.FlowRun)\n    expected_start_time: Optional[DateTimeTZ] = FieldFrom(objects.FlowRun)\n    next_scheduled_start_time: Optional[DateTimeTZ] = FieldFrom(objects.FlowRun)\n    start_time: Optional[DateTimeTZ] = FieldFrom(objects.FlowRun)\n    end_time: Optional[DateTimeTZ] = FieldFrom(objects.FlowRun)\n    total_run_time: datetime.timedelta = FieldFrom(objects.FlowRun)\n    estimated_run_time: datetime.timedelta = FieldFrom(objects.FlowRun)\n    estimated_start_time_delta: datetime.timedelta = FieldFrom(objects.FlowRun)\n    auto_scheduled: bool = FieldFrom(objects.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    infrastructure_pid: Optional[str] = FieldFrom(objects.FlowRun)\n    created_by: Optional[CreatedBy] = FieldFrom(objects.FlowRun)\n    work_pool_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the flow run's work pool.\",\n        example=\"my-work-pool\",\n    )\n    state: Optional[objects.State] = FieldFrom(objects.FlowRun)\n\n    def __eq__(self, other: Any) -> bool:\n        \"\"\"\n        Check for \"equality\" to another flow run schema\n\n        Estimates times are rolling and will always change with repeated queries for\n        a flow run so we ignore them during equality checks.\n        \"\"\"\n        if isinstance(other, FlowRunResponse):\n            exclude_fields = {\"estimated_run_time\", \"estimated_start_time_delta\"}\n            return self.dict(exclude=exclude_fields) == other.dict(\n                exclude=exclude_fields\n            )\n        return super().__eq__(other)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_5","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules","title":"prefect.client.schemas.schedules","text":"

    Schedule schemas

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.IntervalSchedule","title":"IntervalSchedule","text":"

    Bases: PrefectBaseModel

    A schedule formed by adding interval increments to an anchor_date. If no anchor_date is supplied, the current UTC time is used. If a timezone-naive datetime is provided for anchor_date, it is assumed to be in the schedule's timezone (or UTC). Even if supplied with an IANA timezone, anchor dates are always stored as UTC offsets, so a timezone can be provided to determine localization behaviors like DST boundary handling. If none is provided it will be inferred from the anchor date.

    NOTE: If the IntervalSchedule anchor_date or timezone is provided in a DST-observing timezone, then the schedule will adjust itself appropriately. Intervals greater than 24 hours will follow DST conventions, while intervals of less than 24 hours will follow UTC intervals. For example, an hourly schedule will fire every UTC hour, even across DST boundaries. When clocks are set back, this will result in two runs that appear to both be scheduled for 1am local time, even though they are an hour apart in UTC time. For longer intervals, like a daily schedule, the interval schedule will adjust for DST boundaries so that the clock-hour remains constant. This means that a daily schedule that always fires at 9am will observe DST and continue to fire at 9am in the local time zone.

    Parameters:

    Name Type Description Default interval timedelta

    an interval to schedule on

    required anchor_date DateTimeTZ

    an anchor date to schedule increments against; if not provided, the current timestamp will be used

    required timezone str

    a valid timezone string

    required Source code in prefect/client/schemas/schedules.py
    class IntervalSchedule(PrefectBaseModel):\n    \"\"\"\n    A schedule formed by adding `interval` increments to an `anchor_date`. If no\n    `anchor_date` is supplied, the current UTC time is used.  If a\n    timezone-naive datetime is provided for `anchor_date`, it is assumed to be\n    in the schedule's timezone (or UTC). Even if supplied with an IANA timezone,\n    anchor dates are always stored as UTC offsets, so a `timezone` can be\n    provided to determine localization behaviors like DST boundary handling. If\n    none is provided it will be inferred from the anchor date.\n\n    NOTE: If the `IntervalSchedule` `anchor_date` or `timezone` is provided in a\n    DST-observing timezone, then the schedule will adjust itself appropriately.\n    Intervals greater than 24 hours will follow DST conventions, while intervals\n    of less than 24 hours will follow UTC intervals. For example, an hourly\n    schedule will fire every UTC hour, even across DST boundaries. When clocks\n    are set back, this will result in two runs that *appear* to both be\n    scheduled for 1am local time, even though they are an hour apart in UTC\n    time. For longer intervals, like a daily schedule, the interval schedule\n    will adjust for DST boundaries so that the clock-hour remains constant. This\n    means that a daily schedule that always fires at 9am will observe DST and\n    continue to fire at 9am in the local time zone.\n\n    Args:\n        interval (datetime.timedelta): an interval to schedule on\n        anchor_date (DateTimeTZ, optional): an anchor date to schedule increments against;\n            if not provided, the current timestamp will be used\n        timezone (str, optional): a valid timezone string\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n        exclude_none = True\n\n    interval: datetime.timedelta\n    anchor_date: DateTimeTZ = None\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n\n    @validator(\"interval\")\n    def interval_must_be_positive(cls, v):\n        if v.total_seconds() <= 0:\n            raise ValueError(\"The interval must be positive\")\n        return v\n\n    @validator(\"anchor_date\", always=True)\n    def default_anchor_date(cls, v):\n        if v is None:\n            return pendulum.now(\"UTC\")\n        return pendulum.instance(v)\n\n    @validator(\"timezone\", always=True)\n    def default_timezone(cls, v, *, values, **kwargs):\n        # if was provided, make sure its a valid IANA string\n        if v and v not in pendulum.tz.timezones:\n            raise ValueError(f'Invalid timezone: \"{v}\"')\n\n        # otherwise infer the timezone from the anchor date\n        elif v is None and values.get(\"anchor_date\"):\n            tz = values[\"anchor_date\"].tz.name\n            if tz in pendulum.tz.timezones:\n                return tz\n            # sometimes anchor dates have \"timezones\" that are UTC offsets\n            # like \"-04:00\". This happens when parsing ISO8601 strings.\n            # In this case we, the correct inferred localization is \"UTC\".\n            else:\n                return \"UTC\"\n\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.CronSchedule","title":"CronSchedule","text":"

    Bases: PrefectBaseModel

    Cron schedule

    NOTE: If the timezone is a DST-observing one, then the schedule will adjust itself appropriately. Cron's rules for DST are based on schedule times, not intervals. This means that an hourly cron schedule will fire on every new schedule hour, not every elapsed hour; for example, when clocks are set back this will result in a two-hour pause as the schedule will fire the first time 1am is reached and the first time 2am is reached, 120 minutes later. Longer schedules, such as one that fires at 9am every morning, will automatically adjust for DST.

    Parameters:

    Name Type Description Default cron str

    a valid cron string

    required timezone str

    a valid timezone string in IANA tzdata format (for example, America/New_York).

    required day_or bool

    Control how croniter handles day and day_of_week entries. Defaults to True, matching cron which connects those values using OR. If the switch is set to False, the values are connected using AND. This behaves like fcron and enables you to e.g. define a job that executes each 2nd friday of a month by setting the days of month and the weekday.

    required Source code in prefect/client/schemas/schedules.py
    class CronSchedule(PrefectBaseModel):\n    \"\"\"\n    Cron schedule\n\n    NOTE: If the timezone is a DST-observing one, then the schedule will adjust\n    itself appropriately. Cron's rules for DST are based on schedule times, not\n    intervals. This means that an hourly cron schedule will fire on every new\n    schedule hour, not every elapsed hour; for example, when clocks are set back\n    this will result in a two-hour pause as the schedule will fire *the first\n    time* 1am is reached and *the first time* 2am is reached, 120 minutes later.\n    Longer schedules, such as one that fires at 9am every morning, will\n    automatically adjust for DST.\n\n    Args:\n        cron (str): a valid cron string\n        timezone (str): a valid timezone string in IANA tzdata format (for example,\n            America/New_York).\n        day_or (bool, optional): Control how croniter handles `day` and `day_of_week`\n            entries. Defaults to True, matching cron which connects those values using\n            OR. If the switch is set to False, the values are connected using AND. This\n            behaves like fcron and enables you to e.g. define a job that executes each\n            2nd friday of a month by setting the days of month and the weekday.\n\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    cron: str = Field(default=..., example=\"0 0 * * *\")\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n    day_or: bool = Field(\n        default=True,\n        description=(\n            \"Control croniter behavior for handling day and day_of_week entries.\"\n        ),\n    )\n\n    @validator(\"timezone\")\n    def valid_timezone(cls, v):\n        if v and v not in pendulum.tz.timezones:\n            raise ValueError(\n                f'Invalid timezone: \"{v}\" (specify in IANA tzdata format, for example,'\n                \" America/New_York)\"\n            )\n        return v\n\n    @validator(\"cron\")\n    def valid_cron_string(cls, v):\n        # croniter allows \"random\" and \"hashed\" expressions\n        # which we do not support https://github.com/kiorky/croniter\n        if not croniter.is_valid(v):\n            raise ValueError(f'Invalid cron string: \"{v}\"')\n        elif any(c for c in v.split() if c.casefold() in [\"R\", \"H\", \"r\", \"h\"]):\n            raise ValueError(\n                f'Random and Hashed expressions are unsupported, received: \"{v}\"'\n            )\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.RRuleSchedule","title":"RRuleSchedule","text":"

    Bases: PrefectBaseModel

    RRule schedule, based on the iCalendar standard (RFC 5545) as implemented in dateutils.rrule.

    RRules are appropriate for any kind of calendar-date manipulation, including irregular intervals, repetition, exclusions, week day or day-of-month adjustments, and more.

    Note that as a calendar-oriented standard, RRuleSchedules are sensitive to to the initial timezone provided. A 9am daily schedule with a daylight saving time-aware start date will maintain a local 9am time through DST boundaries; a 9am daily schedule with a UTC start date will maintain a 9am UTC time.

    Parameters:

    Name Type Description Default rrule str

    a valid RRule string

    required timezone str

    a valid timezone string

    required Source code in prefect/client/schemas/schedules.py
    class RRuleSchedule(PrefectBaseModel):\n    \"\"\"\n    RRule schedule, based on the iCalendar standard\n    ([RFC 5545](https://datatracker.ietf.org/doc/html/rfc5545)) as\n    implemented in `dateutils.rrule`.\n\n    RRules are appropriate for any kind of calendar-date manipulation, including\n    irregular intervals, repetition, exclusions, week day or day-of-month\n    adjustments, and more.\n\n    Note that as a calendar-oriented standard, `RRuleSchedules` are sensitive to\n    to the initial timezone provided. A 9am daily schedule with a daylight saving\n    time-aware start date will maintain a local 9am time through DST boundaries;\n    a 9am daily schedule with a UTC start date will maintain a 9am UTC time.\n\n    Args:\n        rrule (str): a valid RRule string\n        timezone (str, optional): a valid timezone string\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    rrule: str\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n\n    @validator(\"rrule\")\n    def validate_rrule_str(cls, v):\n        # attempt to parse the rrule string as an rrule object\n        # this will error if the string is invalid\n        try:\n            dateutil.rrule.rrulestr(v, cache=True)\n        except ValueError as exc:\n            # rrules errors are a mix of cryptic and informative\n            # so reraise to be clear that the string was invalid\n            raise ValueError(f'Invalid RRule string \"{v}\": {exc}')\n        if len(v) > MAX_RRULE_LENGTH:\n            raise ValueError(\n                f'Invalid RRule string \"{v[:40]}...\"\\n'\n                f\"Max length is {MAX_RRULE_LENGTH}, got {len(v)}\"\n            )\n        return v\n\n    @classmethod\n    def from_rrule(cls, rrule: dateutil.rrule.rrule):\n        if isinstance(rrule, dateutil.rrule.rrule):\n            if rrule._dtstart.tzinfo is not None:\n                timezone = rrule._dtstart.tzinfo.name\n            else:\n                timezone = \"UTC\"\n            return RRuleSchedule(rrule=str(rrule), timezone=timezone)\n        elif isinstance(rrule, dateutil.rrule.rruleset):\n            dtstarts = [rr._dtstart for rr in rrule._rrule if rr._dtstart is not None]\n            unique_dstarts = set(pendulum.instance(d).in_tz(\"UTC\") for d in dtstarts)\n            unique_timezones = set(d.tzinfo for d in dtstarts if d.tzinfo is not None)\n\n            if len(unique_timezones) > 1:\n                raise ValueError(\n                    f\"rruleset has too many dtstart timezones: {unique_timezones}\"\n                )\n\n            if len(unique_dstarts) > 1:\n                raise ValueError(f\"rruleset has too many dtstarts: {unique_dstarts}\")\n\n            if unique_dstarts and unique_timezones:\n                timezone = dtstarts[0].tzinfo.name\n            else:\n                timezone = \"UTC\"\n\n            rruleset_string = \"\"\n            if rrule._rrule:\n                rruleset_string += \"\\n\".join(str(r) for r in rrule._rrule)\n            if rrule._exrule:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"\\n\".join(str(r) for r in rrule._exrule).replace(\n                    \"RRULE\", \"EXRULE\"\n                )\n            if rrule._rdate:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"RDATE:\" + \",\".join(\n                    rd.strftime(\"%Y%m%dT%H%M%SZ\") for rd in rrule._rdate\n                )\n            if rrule._exdate:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"EXDATE:\" + \",\".join(\n                    exd.strftime(\"%Y%m%dT%H%M%SZ\") for exd in rrule._exdate\n                )\n            return RRuleSchedule(rrule=rruleset_string, timezone=timezone)\n        else:\n            raise ValueError(f\"Invalid RRule object: {rrule}\")\n\n    def to_rrule(self) -> dateutil.rrule.rrule:\n        \"\"\"\n        Since rrule doesn't properly serialize/deserialize timezones, we localize dates\n        here\n        \"\"\"\n        rrule = dateutil.rrule.rrulestr(\n            self.rrule,\n            dtstart=DEFAULT_ANCHOR_DATE,\n            cache=True,\n        )\n        timezone = dateutil.tz.gettz(self.timezone)\n        if isinstance(rrule, dateutil.rrule.rrule):\n            kwargs = dict(dtstart=rrule._dtstart.replace(tzinfo=timezone))\n            if rrule._until:\n                kwargs.update(\n                    until=rrule._until.replace(tzinfo=timezone),\n                )\n            return rrule.replace(**kwargs)\n        elif isinstance(rrule, dateutil.rrule.rruleset):\n            # update rrules\n            localized_rrules = []\n            for rr in rrule._rrule:\n                kwargs = dict(dtstart=rr._dtstart.replace(tzinfo=timezone))\n                if rr._until:\n                    kwargs.update(\n                        until=rr._until.replace(tzinfo=timezone),\n                    )\n                localized_rrules.append(rr.replace(**kwargs))\n            rrule._rrule = localized_rrules\n\n            # update exrules\n            localized_exrules = []\n            for exr in rrule._exrule:\n                kwargs = dict(dtstart=exr._dtstart.replace(tzinfo=timezone))\n                if exr._until:\n                    kwargs.update(\n                        until=exr._until.replace(tzinfo=timezone),\n                    )\n                localized_exrules.append(exr.replace(**kwargs))\n            rrule._exrule = localized_exrules\n\n            # update rdates\n            localized_rdates = []\n            for rd in rrule._rdate:\n                localized_rdates.append(rd.replace(tzinfo=timezone))\n            rrule._rdate = localized_rdates\n\n            # update exdates\n            localized_exdates = []\n            for exd in rrule._exdate:\n                localized_exdates.append(exd.replace(tzinfo=timezone))\n            rrule._exdate = localized_exdates\n\n            return rrule\n\n    @validator(\"timezone\", always=True)\n    def valid_timezone(cls, v):\n        if v and v not in pytz.all_timezones_set:\n            raise ValueError(f'Invalid timezone: \"{v}\"')\n        elif v is None:\n            return \"UTC\"\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.RRuleSchedule.to_rrule","title":"to_rrule","text":"

    Since rrule doesn't properly serialize/deserialize timezones, we localize dates here

    Source code in prefect/client/schemas/schedules.py
    def to_rrule(self) -> dateutil.rrule.rrule:\n    \"\"\"\n    Since rrule doesn't properly serialize/deserialize timezones, we localize dates\n    here\n    \"\"\"\n    rrule = dateutil.rrule.rrulestr(\n        self.rrule,\n        dtstart=DEFAULT_ANCHOR_DATE,\n        cache=True,\n    )\n    timezone = dateutil.tz.gettz(self.timezone)\n    if isinstance(rrule, dateutil.rrule.rrule):\n        kwargs = dict(dtstart=rrule._dtstart.replace(tzinfo=timezone))\n        if rrule._until:\n            kwargs.update(\n                until=rrule._until.replace(tzinfo=timezone),\n            )\n        return rrule.replace(**kwargs)\n    elif isinstance(rrule, dateutil.rrule.rruleset):\n        # update rrules\n        localized_rrules = []\n        for rr in rrule._rrule:\n            kwargs = dict(dtstart=rr._dtstart.replace(tzinfo=timezone))\n            if rr._until:\n                kwargs.update(\n                    until=rr._until.replace(tzinfo=timezone),\n                )\n            localized_rrules.append(rr.replace(**kwargs))\n        rrule._rrule = localized_rrules\n\n        # update exrules\n        localized_exrules = []\n        for exr in rrule._exrule:\n            kwargs = dict(dtstart=exr._dtstart.replace(tzinfo=timezone))\n            if exr._until:\n                kwargs.update(\n                    until=exr._until.replace(tzinfo=timezone),\n                )\n            localized_exrules.append(exr.replace(**kwargs))\n        rrule._exrule = localized_exrules\n\n        # update rdates\n        localized_rdates = []\n        for rd in rrule._rdate:\n            localized_rdates.append(rd.replace(tzinfo=timezone))\n        rrule._rdate = localized_rdates\n\n        # update exdates\n        localized_exdates = []\n        for exd in rrule._exdate:\n            localized_exdates.append(exd.replace(tzinfo=timezone))\n        rrule._exdate = localized_exdates\n\n        return rrule\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.construct_schedule","title":"construct_schedule","text":"

    Construct a schedule from the provided arguments.

    Parameters:

    Name Type Description Default interval Optional[Union[int, float, timedelta]]

    An interval on which to schedule runs. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None anchor_date Optional[Union[datetime, str]]

    The start date for an interval schedule.

    None cron Optional[str]

    A cron schedule for runs.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None timezone Optional[str]

    A timezone to use for the schedule. Defaults to UTC.

    None Source code in prefect/client/schemas/schedules.py
    def construct_schedule(\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    anchor_date: Optional[Union[datetime.datetime, str]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    timezone: Optional[str] = None,\n) -> SCHEDULE_TYPES:\n    \"\"\"\n    Construct a schedule from the provided arguments.\n\n    Args:\n        interval: An interval on which to schedule runs. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        anchor_date: The start date for an interval schedule.\n        cron: A cron schedule for runs.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        timezone: A timezone to use for the schedule. Defaults to UTC.\n    \"\"\"\n    num_schedules = sum(1 for entry in (interval, cron, rrule) if entry is not None)\n    if num_schedules > 1:\n        raise ValueError(\"Only one of interval, cron, or rrule can be provided.\")\n\n    if anchor_date and not interval:\n        raise ValueError(\n            \"An anchor date can only be provided with an interval schedule\"\n        )\n\n    if timezone and not (interval or cron or rrule):\n        raise ValueError(\n            \"A timezone can only be provided with interval, cron, or rrule\"\n        )\n\n    schedule = None\n    if interval:\n        if isinstance(interval, (int, float)):\n            interval = datetime.timedelta(seconds=interval)\n        schedule = IntervalSchedule(\n            interval=interval, anchor_date=anchor_date, timezone=timezone\n        )\n    elif cron:\n        schedule = CronSchedule(cron=cron, timezone=timezone)\n    elif rrule:\n        schedule = RRuleSchedule(rrule=rrule, timezone=timezone)\n\n    if schedule is None:\n        raise ValueError(\"Either interval, cron, or rrule must be provided\")\n\n    return schedule\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_6","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting","title":"prefect.client.schemas.sorting","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.FlowRunSort","title":"FlowRunSort","text":"

    Bases: AutoEnum

    Defines flow run sorting options.

    Source code in prefect/client/schemas/sorting.py
    class FlowRunSort(AutoEnum):\n    \"\"\"Defines flow run sorting options.\"\"\"\n\n    ID_DESC = AutoEnum.auto()\n    START_TIME_ASC = AutoEnum.auto()\n    START_TIME_DESC = AutoEnum.auto()\n    EXPECTED_START_TIME_ASC = AutoEnum.auto()\n    EXPECTED_START_TIME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n    NEXT_SCHEDULED_START_TIME_ASC = AutoEnum.auto()\n    END_TIME_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.TaskRunSort","title":"TaskRunSort","text":"

    Bases: AutoEnum

    Defines task run sorting options.

    Source code in prefect/client/schemas/sorting.py
    class TaskRunSort(AutoEnum):\n    \"\"\"Defines task run sorting options.\"\"\"\n\n    ID_DESC = AutoEnum.auto()\n    EXPECTED_START_TIME_ASC = AutoEnum.auto()\n    EXPECTED_START_TIME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n    NEXT_SCHEDULED_START_TIME_ASC = AutoEnum.auto()\n    END_TIME_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.LogSort","title":"LogSort","text":"

    Bases: AutoEnum

    Defines log sorting options.

    Source code in prefect/client/schemas/sorting.py
    class LogSort(AutoEnum):\n    \"\"\"Defines log sorting options.\"\"\"\n\n    TIMESTAMP_ASC = AutoEnum.auto()\n    TIMESTAMP_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.FlowSort","title":"FlowSort","text":"

    Bases: AutoEnum

    Defines flow sorting options.

    Source code in prefect/client/schemas/sorting.py
    class FlowSort(AutoEnum):\n    \"\"\"Defines flow sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.DeploymentSort","title":"DeploymentSort","text":"

    Bases: AutoEnum

    Defines deployment sorting options.

    Source code in prefect/client/schemas/sorting.py
    class DeploymentSort(AutoEnum):\n    \"\"\"Defines deployment sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.ArtifactSort","title":"ArtifactSort","text":"

    Bases: AutoEnum

    Defines artifact sorting options.

    Source code in prefect/client/schemas/sorting.py
    class ArtifactSort(AutoEnum):\n    \"\"\"Defines artifact sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    ID_DESC = AutoEnum.auto()\n    KEY_DESC = AutoEnum.auto()\n    KEY_ASC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.ArtifactCollectionSort","title":"ArtifactCollectionSort","text":"

    Bases: AutoEnum

    Defines artifact collection sorting options.

    Source code in prefect/client/schemas/sorting.py
    class ArtifactCollectionSort(AutoEnum):\n    \"\"\"Defines artifact collection sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    ID_DESC = AutoEnum.auto()\n    KEY_DESC = AutoEnum.auto()\n    KEY_ASC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.VariableSort","title":"VariableSort","text":"

    Bases: AutoEnum

    Defines variables sorting options.

    Source code in prefect/client/schemas/sorting.py
    class VariableSort(AutoEnum):\n    \"\"\"Defines variables sorting options.\"\"\"\n\n    CREATED_DESC = \"CREATED_DESC\"\n    UPDATED_DESC = \"UPDATED_DESC\"\n    NAME_DESC = \"NAME_DESC\"\n    NAME_ASC = \"NAME_ASC\"\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.BlockDocumentSort","title":"BlockDocumentSort","text":"

    Bases: AutoEnum

    Defines block document sorting options.

    Source code in prefect/client/schemas/sorting.py
    class BlockDocumentSort(AutoEnum):\n    \"\"\"Defines block document sorting options.\"\"\"\n\n    NAME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    BLOCK_TYPE_AND_NAME_ASC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/utilities/","title":"utilities","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/utilities/#prefect.client.utilities","title":"prefect.client.utilities","text":"

    Utilities for working with clients.

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/utilities/#prefect.client.utilities.inject_client","title":"inject_client","text":"

    Simple helper to provide a context managed client to a asynchronous function.

    The decorated function must take a client kwarg and if a client is passed when called it will be used instead of creating a new one, but it will not be context managed as it is assumed that the caller is managing the context.

    Source code in prefect/client/utilities.py
    def inject_client(fn):\n    \"\"\"\n    Simple helper to provide a context managed client to a asynchronous function.\n\n    The decorated function _must_ take a `client` kwarg and if a client is passed when\n    called it will be used instead of creating a new one, but it will not be context\n    managed as it is assumed that the caller is managing the context.\n    \"\"\"\n\n    @wraps(fn)\n    async def with_injected_client(*args, **kwargs):\n        from prefect.client.orchestration import get_client\n        from prefect.context import FlowRunContext, TaskRunContext\n\n        flow_run_context = FlowRunContext.get()\n        task_run_context = TaskRunContext.get()\n        client = None\n        client_context = asyncnullcontext()\n\n        if \"client\" in kwargs and kwargs[\"client\"] is not None:\n            # Client provided in kwargs\n            client = kwargs[\"client\"]\n        elif flow_run_context and flow_run_context.client._loop == get_running_loop():\n            # Client available from flow run context\n            client = flow_run_context.client\n        elif task_run_context and task_run_context.client._loop == get_running_loop():\n            # Client available from task run context\n            client = task_run_context.client\n\n        else:\n            # A new client is needed\n            client_context = get_client()\n\n        # Removes existing client to allow it to be set by setdefault below\n        kwargs.pop(\"client\", None)\n\n        async with client_context as new_client:\n            kwargs.setdefault(\"client\", new_client or client)\n            return await fn(*args, **kwargs)\n\n    return with_injected_client\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/concurrency/asyncio/","title":"asyncio","text":"","tags":["Python API","concurrency","asyncio"]},{"location":"api-ref/prefect/concurrency/asyncio/#prefect.concurrency.asyncio","title":"prefect.concurrency.asyncio","text":"","tags":["Python API","concurrency","asyncio"]},{"location":"api-ref/prefect/concurrency/asyncio/#prefect.concurrency.asyncio.ConcurrencySlotAcquisitionError","title":"ConcurrencySlotAcquisitionError","text":"

    Bases: Exception

    Raised when an unhandlable occurs while acquiring concurrency slots.

    Source code in prefect/concurrency/asyncio.py
    class ConcurrencySlotAcquisitionError(Exception):\n    \"\"\"Raised when an unhandlable occurs while acquiring concurrency slots.\"\"\"\n
    ","tags":["Python API","concurrency","asyncio"]},{"location":"api-ref/prefect/concurrency/asyncio/#prefect.concurrency.asyncio.rate_limit","title":"rate_limit async","text":"

    Block execution until an occupy number of slots of the concurrency limits given in names are acquired. Requires that all given concurrency limits have a slot decay.

    Parameters:

    Name Type Description Default names Union[str, List[str]]

    The names of the concurrency limits to acquire slots from.

    required occupy int

    The number of slots to acquire and hold from each limit.

    1 Source code in prefect/concurrency/asyncio.py
    async def rate_limit(names: Union[str, List[str]], occupy: int = 1):\n    \"\"\"Block execution until an `occupy` number of slots of the concurrency\n    limits given in `names` are acquired. Requires that all given concurrency\n    limits have a slot decay.\n\n    Args:\n        names: The names of the concurrency limits to acquire slots from.\n        occupy: The number of slots to acquire and hold from each limit.\n    \"\"\"\n    names = names if isinstance(names, list) else [names]\n    limits = await _acquire_concurrency_slots(names, occupy, mode=\"rate_limit\")\n    _emit_concurrency_acquisition_events(limits, occupy)\n
    ","tags":["Python API","concurrency","asyncio"]},{"location":"api-ref/prefect/concurrency/common/","title":"common","text":"","tags":["Python API","concurrency","common"]},{"location":"api-ref/prefect/concurrency/common/#prefect.concurrency.common","title":"prefect.concurrency.common","text":"","tags":["Python API","concurrency","common"]},{"location":"api-ref/prefect/concurrency/events/","title":"events","text":"","tags":["Python API","concurrency"]},{"location":"api-ref/prefect/concurrency/events/#prefect.concurrency.events","title":"prefect.concurrency.events","text":"","tags":["Python API","concurrency"]},{"location":"api-ref/prefect/concurrency/services/","title":"services","text":"","tags":["Python API","concurrency"]},{"location":"api-ref/prefect/concurrency/services/#prefect.concurrency.services","title":"prefect.concurrency.services","text":"","tags":["Python API","concurrency"]},{"location":"api-ref/prefect/concurrency/sync/","title":"sync","text":"","tags":["Python API","concurrency","sync"]},{"location":"api-ref/prefect/concurrency/sync/#prefect.concurrency.sync","title":"prefect.concurrency.sync","text":"","tags":["Python API","concurrency","sync"]},{"location":"api-ref/prefect/concurrency/sync/#prefect.concurrency.sync.rate_limit","title":"rate_limit","text":"

    Block execution until an occupy number of slots of the concurrency limits given in names are acquired. Requires that all given concurrency limits have a slot decay.

    Parameters:

    Name Type Description Default names Union[str, List[str]]

    The names of the concurrency limits to acquire slots from.

    required occupy int

    The number of slots to acquire and hold from each limit.

    1 Source code in prefect/concurrency/sync.py
    def rate_limit(names: Union[str, List[str]], occupy: int = 1):\n    \"\"\"Block execution until an `occupy` number of slots of the concurrency\n    limits given in `names` are acquired. Requires that all given concurrency\n    limits have a slot decay.\n\n    Args:\n        names: The names of the concurrency limits to acquire slots from.\n        occupy: The number of slots to acquire and hold from each limit.\n    \"\"\"\n    names = names if isinstance(names, list) else [names]\n    limits = _call_async_function_from_sync(\n        _acquire_concurrency_slots, names, occupy, mode=\"rate_limit\"\n    )\n    _emit_concurrency_acquisition_events(limits, occupy)\n
    ","tags":["Python API","concurrency","sync"]},{"location":"api-ref/prefect/deployments/base/","title":"base","text":"","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base","title":"prefect.deployments.base","text":"

    Core primitives for managing Prefect projects. Projects provide a minimally opinionated build system for managing flows and deployments.

    To get started, follow along with the deloyments tutorial.

    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.configure_project_by_recipe","title":"configure_project_by_recipe","text":"

    Given a recipe name, returns a dictionary representing base configuration options.

    Parameters:

    Name Type Description Default recipe str

    the name of the recipe to use

    required formatting_kwargs dict

    additional keyword arguments to format the recipe

    {}

    Raises:

    Type Description ValueError

    if provided recipe name does not exist.

    Source code in prefect/deployments/base.py
    def configure_project_by_recipe(recipe: str, **formatting_kwargs) -> dict:\n    \"\"\"\n    Given a recipe name, returns a dictionary representing base configuration options.\n\n    Args:\n        recipe (str): the name of the recipe to use\n        formatting_kwargs (dict, optional): additional keyword arguments to format the recipe\n\n    Raises:\n        ValueError: if provided recipe name does not exist.\n    \"\"\"\n    # load the recipe\n    recipe_path = Path(__file__).parent / \"recipes\" / recipe / \"prefect.yaml\"\n\n    if not recipe_path.exists():\n        raise ValueError(f\"Unknown recipe {recipe!r} provided.\")\n\n    with recipe_path.open(mode=\"r\") as f:\n        config = yaml.safe_load(f)\n\n    config = apply_values(\n        template=config, values=formatting_kwargs, remove_notset=False\n    )\n\n    return config\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.create_default_prefect_yaml","title":"create_default_prefect_yaml","text":"

    Creates default prefect.yaml file in the provided path if one does not already exist; returns boolean specifying whether a file was created.

    Parameters:

    Name Type Description Default name str

    the name of the project; if not provided, the current directory name will be used

    None contents dict

    a dictionary of contents to write to the file; if not provided, defaults will be used

    None Source code in prefect/deployments/base.py
    def create_default_prefect_yaml(\n    path: str, name: str = None, contents: dict = None\n) -> bool:\n    \"\"\"\n    Creates default `prefect.yaml` file in the provided path if one does not already exist;\n    returns boolean specifying whether a file was created.\n\n    Args:\n        name (str, optional): the name of the project; if not provided, the current directory name\n            will be used\n        contents (dict, optional): a dictionary of contents to write to the file; if not provided,\n            defaults will be used\n    \"\"\"\n    path = Path(path)\n    prefect_file = path / \"prefect.yaml\"\n    if prefect_file.exists():\n        return False\n    default_file = Path(__file__).parent / \"templates\" / \"prefect.yaml\"\n\n    with default_file.open(mode=\"r\") as df:\n        default_contents = yaml.safe_load(df)\n\n    import prefect\n\n    contents[\"prefect-version\"] = prefect.__version__\n    contents[\"name\"] = name\n\n    with prefect_file.open(mode=\"w\") as f:\n        # write header\n        f.write(\n            \"# Welcome to your prefect.yaml file! You can use this file for storing and\"\n            \" managing\\n# configuration for deploying your flows. We recommend\"\n            \" committing this file to source\\n# control along with your flow code.\\n\\n\"\n        )\n\n        f.write(\"# Generic metadata about this project\\n\")\n        yaml.dump({\"name\": contents[\"name\"]}, f, sort_keys=False)\n        yaml.dump({\"prefect-version\": contents[\"prefect-version\"]}, f, sort_keys=False)\n        f.write(\"\\n\")\n\n        # build\n        f.write(\"# build section allows you to manage and build docker images\\n\")\n        yaml.dump(\n            {\"build\": contents.get(\"build\", default_contents.get(\"build\"))},\n            f,\n            sort_keys=False,\n        )\n        f.write(\"\\n\")\n\n        # push\n        f.write(\n            \"# push section allows you to manage if and how this project is uploaded to\"\n            \" remote locations\\n\"\n        )\n        yaml.dump(\n            {\"push\": contents.get(\"push\", default_contents.get(\"push\"))},\n            f,\n            sort_keys=False,\n        )\n        f.write(\"\\n\")\n\n        # pull\n        f.write(\n            \"# pull section allows you to provide instructions for cloning this project\"\n            \" in remote locations\\n\"\n        )\n        yaml.dump(\n            {\"pull\": contents.get(\"pull\", default_contents.get(\"pull\"))},\n            f,\n            sort_keys=False,\n        )\n        f.write(\"\\n\")\n\n        # deployments\n        f.write(\n            \"# the deployments section allows you to provide configuration for\"\n            \" deploying flows\\n\"\n        )\n        yaml.dump(\n            {\n                \"deployments\": contents.get(\n                    \"deployments\", default_contents.get(\"deployments\")\n                )\n            },\n            f,\n            sort_keys=False,\n        )\n    return True\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.find_prefect_directory","title":"find_prefect_directory","text":"

    Given a path, recurses upward looking for .prefect/ directories.

    Once found, returns absolute path to the ./prefect directory, which is assumed to reside within the root for the current project.

    If one is never found, None is returned.

    Source code in prefect/deployments/base.py
    def find_prefect_directory(path: Path = None) -> Optional[Path]:\n    \"\"\"\n    Given a path, recurses upward looking for .prefect/ directories.\n\n    Once found, returns absolute path to the ./prefect directory, which is assumed to reside within the\n    root for the current project.\n\n    If one is never found, `None` is returned.\n    \"\"\"\n    path = Path(path or \".\").resolve()\n    parent = path.parent.resolve()\n    while path != parent:\n        prefect_dir = path.joinpath(\".prefect\")\n        if prefect_dir.is_dir():\n            return prefect_dir\n\n        path = parent.resolve()\n        parent = path.parent.resolve()\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.initialize_project","title":"initialize_project","text":"

    Initializes a basic project structure with base files. If no name is provided, the name of the current directory is used. If no recipe is provided, one is inferred.

    Parameters:

    Name Type Description Default name str

    the name of the project; if not provided, the current directory name

    None recipe str

    the name of the recipe to use; if not provided, one is inferred

    None inputs dict

    a dictionary of inputs to use when formatting the recipe

    None

    Returns:

    Type Description List[str]

    List[str]: a list of files / directories that were created

    Source code in prefect/deployments/base.py
    def initialize_project(\n    name: str = None, recipe: str = None, inputs: dict = None\n) -> List[str]:\n    \"\"\"\n    Initializes a basic project structure with base files.  If no name is provided, the name\n    of the current directory is used.  If no recipe is provided, one is inferred.\n\n    Args:\n        name (str, optional): the name of the project; if not provided, the current directory name\n        recipe (str, optional): the name of the recipe to use; if not provided, one is inferred\n        inputs (dict, optional): a dictionary of inputs to use when formatting the recipe\n\n    Returns:\n        List[str]: a list of files / directories that were created\n    \"\"\"\n    # determine if in git repo or use directory name as a default\n    is_git_based = False\n    formatting_kwargs = {\"directory\": str(Path(\".\").absolute().resolve())}\n    dir_name = os.path.basename(os.getcwd())\n\n    remote_url = _get_git_remote_origin_url()\n    if remote_url:\n        formatting_kwargs[\"repository\"] = remote_url\n        is_git_based = True\n        branch = _get_git_branch()\n        formatting_kwargs[\"branch\"] = branch or \"main\"\n\n    formatting_kwargs[\"name\"] = dir_name\n\n    has_dockerfile = Path(\"Dockerfile\").exists()\n\n    if has_dockerfile:\n        formatting_kwargs[\"dockerfile\"] = \"Dockerfile\"\n    elif recipe is not None and \"docker\" in recipe:\n        formatting_kwargs[\"dockerfile\"] = \"auto\"\n\n    # hand craft a pull step\n    if is_git_based and recipe is None:\n        if has_dockerfile:\n            recipe = \"docker-git\"\n        else:\n            recipe = \"git\"\n    elif recipe is None and has_dockerfile:\n        recipe = \"docker\"\n    elif recipe is None:\n        recipe = \"local\"\n\n    formatting_kwargs.update(inputs or {})\n    configuration = configure_project_by_recipe(recipe=recipe, **formatting_kwargs)\n\n    project_name = name or dir_name\n\n    files = []\n    if create_default_ignore_file(\".\"):\n        files.append(\".prefectignore\")\n    if create_default_prefect_yaml(\".\", name=project_name, contents=configuration):\n        files.append(\"prefect.yaml\")\n    if set_prefect_hidden_dir():\n        files.append(\".prefect/\")\n\n    return files\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.register_flow","title":"register_flow async","text":"

    Register a flow with this project from an entrypoint.

    Parameters:

    Name Type Description Default entrypoint str

    the entrypoint to the flow to register

    required force bool

    whether or not to overwrite an existing flow with the same name

    False

    Raises:

    Type Description ValueError

    if force is False and registration would overwrite an existing flow

    Source code in prefect/deployments/base.py
    async def register_flow(entrypoint: str, force: bool = False):\n    \"\"\"\n    Register a flow with this project from an entrypoint.\n\n    Args:\n        entrypoint (str): the entrypoint to the flow to register\n        force (bool, optional): whether or not to overwrite an existing flow with the same name\n\n    Raises:\n        ValueError: if `force` is `False` and registration would overwrite an existing flow\n    \"\"\"\n    try:\n        fpath, obj_name = entrypoint.rsplit(\":\", 1)\n    except ValueError as exc:\n        if str(exc) == \"not enough values to unpack (expected 2, got 1)\":\n            missing_flow_name_msg = (\n                \"Your flow entrypoint must include the name of the function that is\"\n                f\" the entrypoint to your flow.\\nTry {entrypoint}:<flow_name> as your\"\n                f\" entrypoint. If you meant to specify '{entrypoint}' as the deployment\"\n                f\" name, try `prefect deploy -n {entrypoint}`.\"\n            )\n            raise ValueError(missing_flow_name_msg)\n        else:\n            raise exc\n\n    flow = await run_sync_in_worker_thread(load_flow_from_entrypoint, entrypoint)\n\n    fpath = Path(fpath).absolute()\n    prefect_dir = find_prefect_directory()\n    if not prefect_dir:\n        raise FileNotFoundError(\n            \"No .prefect directory could be found - run `prefect project\"\n            \" init` to create one.\"\n        )\n\n    entrypoint = f\"{fpath.relative_to(prefect_dir.parent)!s}:{obj_name}\"\n\n    flows_file = prefect_dir / \"flows.json\"\n    if flows_file.exists():\n        with flows_file.open(mode=\"r\") as f:\n            flows = json.load(f)\n    else:\n        flows = {}\n\n    ## quality control\n    if flow.name in flows and flows[flow.name] != entrypoint:\n        if not force:\n            raise ValueError(\n                f\"Conflicting entry found for flow with name {flow.name!r}.\\nExisting\"\n                f\" entrypoint: {flows[flow.name]}\\nAttempted entrypoint:\"\n                f\" {entrypoint}\\n\\nYou can try removing the existing entry for\"\n                f\" {flow.name!r} from your [yellow]~/.prefect/flows.json[/yellow].\"\n            )\n\n    flows[flow.name] = entrypoint\n\n    with flows_file.open(mode=\"w\") as f:\n        json.dump(flows, f, sort_keys=True, indent=2)\n\n    return flow\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.set_prefect_hidden_dir","title":"set_prefect_hidden_dir","text":"

    Creates default .prefect/ directory if one does not already exist. Returns boolean specifying whether or not a directory was created.

    If a path is provided, the directory will be created in that location.

    Source code in prefect/deployments/base.py
    def set_prefect_hidden_dir(path: str = None) -> bool:\n    \"\"\"\n    Creates default `.prefect/` directory if one does not already exist.\n    Returns boolean specifying whether or not a directory was created.\n\n    If a path is provided, the directory will be created in that location.\n    \"\"\"\n    path = Path(path or \".\") / \".prefect\"\n\n    # use exists so that we dont accidentally overwrite a file\n    if path.exists():\n        return False\n    path.mkdir(mode=0o0700)\n    return True\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/deployments/","title":"deployments","text":"","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments","title":"prefect.deployments.deployments","text":"

    Objects for specifying deployments and utilities for loading flows from deployments.

    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment","title":"Deployment","text":"

    Bases: BaseModel

    A Prefect Deployment definition, used for specifying and building deployments.

    Parameters:

    Name Type Description Default name

    A name for the deployment (required).

    required version

    An optional version for the deployment; defaults to the flow's version

    required description

    An optional description of the deployment; defaults to the flow's description

    required tags

    An optional list of tags to associate with this deployment; note that tags are used only for organizational purposes. For delegating work to agents, see work_queue_name.

    required schedule

    A schedule to run this deployment on, once registered

    required is_schedule_active

    Whether or not the schedule is active

    required work_queue_name

    The work queue that will handle this deployment's runs

    required work_pool_name

    The work pool for the deployment

    required flow_name

    The name of the flow this deployment encapsulates

    required parameters

    A dictionary of parameter values to pass to runs created from this deployment

    required infrastructure

    An optional infrastructure block used to configure infrastructure for runs; if not provided, will default to running this deployment in Agent subprocesses

    required infra_overrides

    A dictionary of dot delimited infrastructure overrides that will be applied at runtime; for example env.CONFIG_KEY=config_value or namespace='prefect'

    required storage

    An optional remote storage block used to store and retrieve this workflow; if not provided, will default to referencing this flow by its local path

    required path

    The path to the working directory for the workflow, relative to remote storage or, if stored on a local filesystem, an absolute path

    required entrypoint

    The path to the entrypoint for the workflow, always relative to the path

    required parameter_openapi_schema

    The parameter schema of the flow, including defaults.

    required enforce_parameter_schema

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    required
    Create a new deployment using configuration defaults for an imported flow:\n\n>>> from my_project.flows import my_flow\n>>> from prefect.deployments import Deployment\n>>>\n>>> deployment = Deployment.build_from_flow(\n...     flow=my_flow,\n...     name=\"example\",\n...     version=\"1\",\n...     tags=[\"demo\"],\n>>> )\n>>> deployment.apply()\n\nCreate a new deployment with custom storage and an infrastructure override:\n\n>>> from my_project.flows import my_flow\n>>> from prefect.deployments import Deployment\n>>> from prefect.filesystems import S3\n\n>>> storage = S3.load(\"dev-bucket\") # load a pre-defined block\n>>> deployment = Deployment.build_from_flow(\n...     flow=my_flow,\n...     name=\"s3-example\",\n...     version=\"2\",\n...     tags=[\"aws\"],\n...     storage=storage,\n...     infra_overrides=dict(\"env.PREFECT_LOGGING_LEVEL\"=\"DEBUG\"),\n>>> )\n>>> deployment.apply()\n
    Source code in prefect/deployments/deployments.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None and x != DEFAULT_AGENT_WORK_POOL_NAME,\n)\nclass Deployment(BaseModel):\n    \"\"\"\n    A Prefect Deployment definition, used for specifying and building deployments.\n\n    Args:\n        name: A name for the deployment (required).\n        version: An optional version for the deployment; defaults to the flow's version\n        description: An optional description of the deployment; defaults to the flow's\n            description\n        tags: An optional list of tags to associate with this deployment; note that tags\n            are used only for organizational purposes. For delegating work to agents,\n            see `work_queue_name`.\n        schedule: A schedule to run this deployment on, once registered\n        is_schedule_active: Whether or not the schedule is active\n        work_queue_name: The work queue that will handle this deployment's runs\n        work_pool_name: The work pool for the deployment\n        flow_name: The name of the flow this deployment encapsulates\n        parameters: A dictionary of parameter values to pass to runs created from this\n            deployment\n        infrastructure: An optional infrastructure block used to configure\n            infrastructure for runs; if not provided, will default to running this\n            deployment in Agent subprocesses\n        infra_overrides: A dictionary of dot delimited infrastructure overrides that\n            will be applied at runtime; for example `env.CONFIG_KEY=config_value` or\n            `namespace='prefect'`\n        storage: An optional remote storage block used to store and retrieve this\n            workflow; if not provided, will default to referencing this flow by its\n            local path\n        path: The path to the working directory for the workflow, relative to remote\n            storage or, if stored on a local filesystem, an absolute path\n        entrypoint: The path to the entrypoint for the workflow, always relative to the\n            `path`\n        parameter_openapi_schema: The parameter schema of the flow, including defaults.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n\n    Examples:\n\n        Create a new deployment using configuration defaults for an imported flow:\n\n        >>> from my_project.flows import my_flow\n        >>> from prefect.deployments import Deployment\n        >>>\n        >>> deployment = Deployment.build_from_flow(\n        ...     flow=my_flow,\n        ...     name=\"example\",\n        ...     version=\"1\",\n        ...     tags=[\"demo\"],\n        >>> )\n        >>> deployment.apply()\n\n        Create a new deployment with custom storage and an infrastructure override:\n\n        >>> from my_project.flows import my_flow\n        >>> from prefect.deployments import Deployment\n        >>> from prefect.filesystems import S3\n\n        >>> storage = S3.load(\"dev-bucket\") # load a pre-defined block\n        >>> deployment = Deployment.build_from_flow(\n        ...     flow=my_flow,\n        ...     name=\"s3-example\",\n        ...     version=\"2\",\n        ...     tags=[\"aws\"],\n        ...     storage=storage,\n        ...     infra_overrides=dict(\"env.PREFECT_LOGGING_LEVEL\"=\"DEBUG\"),\n        >>> )\n        >>> deployment.apply()\n\n    \"\"\"\n\n    class Config:\n        json_encoders = {SecretDict: lambda v: v.dict()}\n        validate_assignment = True\n        extra = \"forbid\"\n\n    @property\n    def _editable_fields(self) -> List[str]:\n        editable_fields = [\n            \"name\",\n            \"description\",\n            \"version\",\n            \"work_queue_name\",\n            \"work_pool_name\",\n            \"tags\",\n            \"parameters\",\n            \"schedule\",\n            \"is_schedule_active\",\n            \"infra_overrides\",\n        ]\n\n        # if infrastructure is baked as a pre-saved block, then\n        # editing its fields will not update anything\n        if self.infrastructure._block_document_id:\n            return editable_fields\n        else:\n            return editable_fields + [\"infrastructure\"]\n\n    @property\n    def location(self) -> str:\n        \"\"\"\n        The 'location' that this deployment points to is given by `path` alone\n        in the case of no remote storage, and otherwise by `storage.basepath / path`.\n\n        The underlying flow entrypoint is interpreted relative to this location.\n        \"\"\"\n        location = \"\"\n        if self.storage:\n            location = (\n                self.storage.basepath + \"/\"\n                if not self.storage.basepath.endswith(\"/\")\n                else \"\"\n            )\n        if self.path:\n            location += self.path\n        return location\n\n    @sync_compatible\n    async def to_yaml(self, path: Path) -> None:\n        yaml_dict = self._yaml_dict()\n        schema = self.schema()\n\n        with open(path, \"w\") as f:\n            # write header\n            f.write(\n                \"###\\n### A complete description of a Prefect Deployment for flow\"\n                f\" {self.flow_name!r}\\n###\\n\"\n            )\n\n            # write editable fields\n            for field in self._editable_fields:\n                # write any comments\n                if schema[\"properties\"][field].get(\"yaml_comment\"):\n                    f.write(f\"# {schema['properties'][field]['yaml_comment']}\\n\")\n                # write the field\n                yaml.dump({field: yaml_dict[field]}, f, sort_keys=False)\n\n            # write non-editable fields\n            f.write(\"\\n###\\n### DO NOT EDIT BELOW THIS LINE\\n###\\n\")\n            yaml.dump(\n                {k: v for k, v in yaml_dict.items() if k not in self._editable_fields},\n                f,\n                sort_keys=False,\n            )\n\n    def _yaml_dict(self) -> dict:\n        \"\"\"\n        Returns a YAML-compatible representation of this deployment as a dictionary.\n        \"\"\"\n        # avoids issues with UUIDs showing up in YAML\n        all_fields = json.loads(\n            self.json(\n                exclude={\n                    \"storage\": {\"_filesystem\", \"filesystem\", \"_remote_file_system\"}\n                }\n            )\n        )\n        if all_fields[\"storage\"]:\n            all_fields[\"storage\"][\n                \"_block_type_slug\"\n            ] = self.storage.get_block_type_slug()\n        if all_fields[\"infrastructure\"]:\n            all_fields[\"infrastructure\"][\n                \"_block_type_slug\"\n            ] = self.infrastructure.get_block_type_slug()\n        return all_fields\n\n    # top level metadata\n    name: str = Field(..., description=\"The name of the deployment.\")\n    description: Optional[str] = Field(\n        default=None, description=\"An optional description of the deployment.\"\n    )\n    version: Optional[str] = Field(\n        default=None, description=\"An optional version for the deployment.\"\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"One of more tags to apply to this deployment.\",\n    )\n    schedule: SCHEDULE_TYPES = None\n    is_schedule_active: Optional[bool] = Field(\n        default=None, description=\"Whether or not the schedule is active.\"\n    )\n    flow_name: Optional[str] = Field(default=None, description=\"The name of the flow.\")\n    work_queue_name: Optional[str] = Field(\n        \"default\",\n        description=\"The work queue for the deployment.\",\n        yaml_comment=\"The work queue that will handle this deployment's runs\",\n    )\n    work_pool_name: Optional[str] = Field(\n        default=None, description=\"The work pool for the deployment\"\n    )\n    # flow data\n    parameters: Dict[str, Any] = Field(default_factory=dict)\n    manifest_path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the flow's manifest file, relative to the chosen storage.\"\n        ),\n    )\n    infrastructure: Infrastructure = Field(default_factory=Process)\n    infra_overrides: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Overrides to apply to the base infrastructure block at runtime.\",\n    )\n    storage: Optional[Block] = Field(\n        None,\n        help=\"The remote storage to use for this workflow.\",\n    )\n    path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the working directory for the workflow, relative to remote\"\n            \" storage or an absolute path.\"\n        ),\n    )\n    entrypoint: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the entrypoint for the workflow, relative to the `path`.\"\n        ),\n    )\n    parameter_openapi_schema: ParameterSchema = Field(\n        default_factory=ParameterSchema,\n        description=\"The parameter schema of the flow, including defaults.\",\n    )\n    timestamp: datetime = Field(default_factory=partial(pendulum.now, \"UTC\"))\n    triggers: List[DeploymentTrigger] = Field(\n        default_factory=list,\n        description=\"The triggers that should cause this deployment to run.\",\n    )\n    # defaults to None to allow for backwards compatibility\n    enforce_parameter_schema: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Whether or not the Prefect API should enforce the parameter schema for\"\n            \" this deployment.\"\n        ),\n    )\n\n    @validator(\"infrastructure\", pre=True)\n    def infrastructure_must_have_capabilities(cls, value):\n        if isinstance(value, dict):\n            if \"_block_type_slug\" in value:\n                # Replace private attribute with public for dispatch\n                value[\"block_type_slug\"] = value.pop(\"_block_type_slug\")\n            block = Block(**value)\n        elif value is None:\n            return value\n        else:\n            block = value\n\n        if \"run-infrastructure\" not in block.get_block_capabilities():\n            raise ValueError(\n                \"Infrastructure block must have 'run-infrastructure' capabilities.\"\n            )\n        return block\n\n    @validator(\"storage\", pre=True)\n    def storage_must_have_capabilities(cls, value):\n        if isinstance(value, dict):\n            block_type = Block.get_block_class_from_key(value.pop(\"_block_type_slug\"))\n            block = block_type(**value)\n        elif value is None:\n            return value\n        else:\n            block = value\n\n        capabilities = block.get_block_capabilities()\n        if \"get-directory\" not in capabilities:\n            raise ValueError(\n                \"Remote Storage block must have 'get-directory' capabilities.\"\n            )\n        return block\n\n    @validator(\"parameter_openapi_schema\", pre=True)\n    def handle_openapi_schema(cls, value):\n        \"\"\"\n        This method ensures setting a value of `None` is handled gracefully.\n        \"\"\"\n        if value is None:\n            return ParameterSchema()\n        return value\n\n    @validator(\"triggers\")\n    def validate_automation_names(cls, field_value, values, field, config):\n        \"\"\"Ensure that each trigger has a name for its automation if none is provided.\"\"\"\n        for i, trigger in enumerate(field_value, start=1):\n            if trigger.name is None:\n                trigger.name = f\"{values['name']}__automation_{i}\"\n\n        return field_value\n\n    @classmethod\n    @sync_compatible\n    async def load_from_yaml(cls, path: str):\n        data = yaml.safe_load(await anyio.Path(path).read_bytes())\n        # load blocks from server to ensure secret values are properly hydrated\n        if data.get(\"storage\"):\n            block_doc_name = data[\"storage\"].get(\"_block_document_name\")\n            # if no doc name, this block is not stored on the server\n            if block_doc_name:\n                block_slug = data[\"storage\"][\"_block_type_slug\"]\n                block = await Block.load(f\"{block_slug}/{block_doc_name}\")\n                data[\"storage\"] = block\n\n        if data.get(\"infrastructure\"):\n            block_doc_name = data[\"infrastructure\"].get(\"_block_document_name\")\n            # if no doc name, this block is not stored on the server\n            if block_doc_name:\n                block_slug = data[\"infrastructure\"][\"_block_type_slug\"]\n                block = await Block.load(f\"{block_slug}/{block_doc_name}\")\n                data[\"infrastructure\"] = block\n\n            return cls(**data)\n\n    @sync_compatible\n    async def load(self) -> bool:\n        \"\"\"\n        Queries the API for a deployment with this name for this flow, and if found,\n        prepopulates any settings that were not set at initialization.\n\n        Returns a boolean specifying whether a load was successful or not.\n\n        Raises:\n            - ValueError: if both name and flow name are not set\n        \"\"\"\n        if not self.name or not self.flow_name:\n            raise ValueError(\"Both a deployment name and flow name must be provided.\")\n        async with get_client() as client:\n            try:\n                deployment = await client.read_deployment_by_name(\n                    f\"{self.flow_name}/{self.name}\"\n                )\n                if deployment.storage_document_id:\n                    Block._from_block_document(\n                        await client.read_block_document(deployment.storage_document_id)\n                    )\n\n                excluded_fields = self.__fields_set__.union(\n                    {\n                        \"infrastructure\",\n                        \"storage\",\n                        \"timestamp\",\n                        \"triggers\",\n                        \"enforce_parameter_schema\",\n                    }\n                )\n                for field in set(self.__fields__.keys()) - excluded_fields:\n                    new_value = getattr(deployment, field)\n                    setattr(self, field, new_value)\n\n                if \"infrastructure\" not in self.__fields_set__:\n                    if deployment.infrastructure_document_id:\n                        self.infrastructure = Block._from_block_document(\n                            await client.read_block_document(\n                                deployment.infrastructure_document_id\n                            )\n                        )\n                if \"storage\" not in self.__fields_set__:\n                    if deployment.storage_document_id:\n                        self.storage = Block._from_block_document(\n                            await client.read_block_document(\n                                deployment.storage_document_id\n                            )\n                        )\n            except ObjectNotFound:\n                return False\n        return True\n\n    @sync_compatible\n    async def update(self, ignore_none: bool = False, **kwargs):\n        \"\"\"\n        Performs an in-place update with the provided settings.\n\n        Args:\n            ignore_none: if True, all `None` values are ignored when performing the\n                update\n        \"\"\"\n        unknown_keys = set(kwargs.keys()) - set(self.dict().keys())\n        if unknown_keys:\n            raise ValueError(\n                f\"Received unexpected attributes: {', '.join(unknown_keys)}\"\n            )\n        for key, value in kwargs.items():\n            if ignore_none and value is None:\n                continue\n            setattr(self, key, value)\n\n    @sync_compatible\n    async def upload_to_storage(\n        self, storage_block: str = None, ignore_file: str = \".prefectignore\"\n    ) -> Optional[int]:\n        \"\"\"\n        Uploads the workflow this deployment represents using a provided storage block;\n        if no block is provided, defaults to configuring self for local storage.\n\n        Args:\n            storage_block: a string reference a remote storage block slug `$type/$name`;\n                if provided, used to upload the workflow's project\n            ignore_file: an optional path to a `.prefectignore` file that specifies\n                filename patterns to ignore when uploading to remote storage; if not\n                provided, looks for `.prefectignore` in the current working directory\n        \"\"\"\n        file_count = None\n        if storage_block:\n            storage = await Block.load(storage_block)\n\n            if \"put-directory\" not in storage.get_block_capabilities():\n                raise BlockMissingCapabilities(\n                    f\"Storage block {storage!r} missing 'put-directory' capability.\"\n                )\n\n            self.storage = storage\n\n            # upload current directory to storage location\n            file_count = await self.storage.put_directory(\n                ignore_file=ignore_file, to_path=self.path\n            )\n        elif self.storage:\n            if \"put-directory\" not in self.storage.get_block_capabilities():\n                raise BlockMissingCapabilities(\n                    f\"Storage block {self.storage!r} missing 'put-directory'\"\n                    \" capability.\"\n                )\n\n            file_count = await self.storage.put_directory(\n                ignore_file=ignore_file, to_path=self.path\n            )\n\n        # persists storage now in case it contains secret values\n        if self.storage and not self.storage._block_document_id:\n            await self.storage._save(is_anonymous=True)\n\n        return file_count\n\n    @sync_compatible\n    async def apply(\n        self, upload: bool = False, work_queue_concurrency: int = None\n    ) -> UUID:\n        \"\"\"\n        Registers this deployment with the API and returns the deployment's ID.\n\n        Args:\n            upload: if True, deployment files are automatically uploaded to remote\n                storage\n            work_queue_concurrency: If provided, sets the concurrency limit on the\n                deployment's work queue\n        \"\"\"\n        if not self.name or not self.flow_name:\n            raise ValueError(\"Both a deployment name and flow name must be set.\")\n        async with get_client() as client:\n            # prep IDs\n            flow_id = await client.create_flow_from_name(self.flow_name)\n\n            infrastructure_document_id = self.infrastructure._block_document_id\n            if not infrastructure_document_id:\n                # if not building off a block, will create an anonymous block\n                self.infrastructure = self.infrastructure.copy()\n                infrastructure_document_id = await self.infrastructure._save(\n                    is_anonymous=True,\n                )\n\n            if upload:\n                await self.upload_to_storage()\n\n            if self.work_queue_name and work_queue_concurrency is not None:\n                try:\n                    res = await client.create_work_queue(\n                        name=self.work_queue_name, work_pool_name=self.work_pool_name\n                    )\n                except ObjectAlreadyExists:\n                    res = await client.read_work_queue_by_name(\n                        name=self.work_queue_name, work_pool_name=self.work_pool_name\n                    )\n                await client.update_work_queue(\n                    res.id, concurrency_limit=work_queue_concurrency\n                )\n\n            # we assume storage was already saved\n            storage_document_id = getattr(self.storage, \"_block_document_id\", None)\n            deployment_id = await client.create_deployment(\n                flow_id=flow_id,\n                name=self.name,\n                work_queue_name=self.work_queue_name,\n                work_pool_name=self.work_pool_name,\n                version=self.version,\n                schedule=self.schedule,\n                is_schedule_active=self.is_schedule_active,\n                parameters=self.parameters,\n                description=self.description,\n                tags=self.tags,\n                manifest_path=self.manifest_path,  # allows for backwards YAML compat\n                path=self.path,\n                entrypoint=self.entrypoint,\n                infra_overrides=self.infra_overrides,\n                storage_document_id=storage_document_id,\n                infrastructure_document_id=infrastructure_document_id,\n                parameter_openapi_schema=self.parameter_openapi_schema.dict(),\n                enforce_parameter_schema=self.enforce_parameter_schema,\n            )\n\n            if client.server_type == ServerType.CLOUD:\n                # The triggers defined in the deployment spec are, essentially,\n                # anonymous and attempting truly sync them with cloud is not\n                # feasible. Instead, we remove all automations that are owned\n                # by the deployment, meaning that they were created via this\n                # mechanism below, and then recreate them.\n                await client.delete_resource_owned_automations(\n                    f\"prefect.deployment.{deployment_id}\"\n                )\n                for trigger in self.triggers:\n                    trigger.set_deployment_id(deployment_id)\n                    await client.create_automation(trigger.as_automation())\n\n            return deployment_id\n\n    @classmethod\n    @sync_compatible\n    async def build_from_flow(\n        cls,\n        flow: Flow,\n        name: str,\n        output: str = None,\n        skip_upload: bool = False,\n        ignore_file: str = \".prefectignore\",\n        apply: bool = False,\n        load_existing: bool = True,\n        **kwargs,\n    ) -> \"Deployment\":\n        \"\"\"\n        Configure a deployment for a given flow.\n\n        Args:\n            flow: A flow function to deploy\n            name: A name for the deployment\n            output (optional): if provided, the full deployment specification will be\n                written as a YAML file in the location specified by `output`\n            skip_upload: if True, deployment files are not automatically uploaded to\n                remote storage\n            ignore_file: an optional path to a `.prefectignore` file that specifies\n                filename patterns to ignore when uploading to remote storage; if not\n                provided, looks for `.prefectignore` in the current working directory\n            apply: if True, the deployment is automatically registered with the API\n            load_existing: if True, load any settings that may already be configured for\n                the named deployment server-side (e.g., schedules, default parameter\n                values, etc.)\n            **kwargs: other keyword arguments to pass to the constructor for the\n                `Deployment` class\n        \"\"\"\n        if not name:\n            raise ValueError(\"A deployment name must be provided.\")\n\n        # note that `deployment.load` only updates settings that were *not*\n        # provided at initialization\n        deployment = cls(name=name, **kwargs)\n        deployment.flow_name = flow.name\n        if not deployment.entrypoint:\n            ## first see if an entrypoint can be determined\n            flow_file = getattr(flow, \"__globals__\", {}).get(\"__file__\")\n            mod_name = getattr(flow, \"__module__\", None)\n            if not flow_file:\n                if not mod_name:\n                    # todo, check if the file location was manually set already\n                    raise ValueError(\"Could not determine flow's file location.\")\n                module = importlib.import_module(mod_name)\n                flow_file = getattr(module, \"__file__\", None)\n                if not flow_file:\n                    raise ValueError(\"Could not determine flow's file location.\")\n\n            # set entrypoint\n            entry_path = Path(flow_file).absolute().relative_to(Path(\".\").absolute())\n            deployment.entrypoint = f\"{entry_path}:{flow.fn.__name__}\"\n\n        if load_existing:\n            await deployment.load()\n\n        # set a few attributes for this flow object\n        deployment.parameter_openapi_schema = parameter_schema(flow)\n\n        # ensure the ignore file exists\n        if not Path(ignore_file).exists():\n            Path(ignore_file).touch()\n\n        if not deployment.version:\n            deployment.version = flow.version\n        if not deployment.description:\n            deployment.description = flow.description\n\n        # proxy for whether infra is docker-based\n        is_docker_based = hasattr(deployment.infrastructure, \"image\")\n\n        if not deployment.storage and not is_docker_based and not deployment.path:\n            deployment.path = str(Path(\".\").absolute())\n        elif not deployment.storage and is_docker_based:\n            # only update if a path is not already set\n            if not deployment.path:\n                deployment.path = \"/opt/prefect/flows\"\n\n        if not skip_upload:\n            if (\n                deployment.storage\n                and \"put-directory\" in deployment.storage.get_block_capabilities()\n            ):\n                await deployment.upload_to_storage(ignore_file=ignore_file)\n\n        if output:\n            await deployment.to_yaml(output)\n\n        if apply:\n            await deployment.apply()\n\n        return deployment\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.location","title":"location: str property","text":"

    The 'location' that this deployment points to is given by path alone in the case of no remote storage, and otherwise by storage.basepath / path.

    The underlying flow entrypoint is interpreted relative to this location.

    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.apply","title":"apply async","text":"

    Registers this deployment with the API and returns the deployment's ID.

    Parameters:

    Name Type Description Default upload bool

    if True, deployment files are automatically uploaded to remote storage

    False work_queue_concurrency int

    If provided, sets the concurrency limit on the deployment's work queue

    None Source code in prefect/deployments/deployments.py
    @sync_compatible\nasync def apply(\n    self, upload: bool = False, work_queue_concurrency: int = None\n) -> UUID:\n    \"\"\"\n    Registers this deployment with the API and returns the deployment's ID.\n\n    Args:\n        upload: if True, deployment files are automatically uploaded to remote\n            storage\n        work_queue_concurrency: If provided, sets the concurrency limit on the\n            deployment's work queue\n    \"\"\"\n    if not self.name or not self.flow_name:\n        raise ValueError(\"Both a deployment name and flow name must be set.\")\n    async with get_client() as client:\n        # prep IDs\n        flow_id = await client.create_flow_from_name(self.flow_name)\n\n        infrastructure_document_id = self.infrastructure._block_document_id\n        if not infrastructure_document_id:\n            # if not building off a block, will create an anonymous block\n            self.infrastructure = self.infrastructure.copy()\n            infrastructure_document_id = await self.infrastructure._save(\n                is_anonymous=True,\n            )\n\n        if upload:\n            await self.upload_to_storage()\n\n        if self.work_queue_name and work_queue_concurrency is not None:\n            try:\n                res = await client.create_work_queue(\n                    name=self.work_queue_name, work_pool_name=self.work_pool_name\n                )\n            except ObjectAlreadyExists:\n                res = await client.read_work_queue_by_name(\n                    name=self.work_queue_name, work_pool_name=self.work_pool_name\n                )\n            await client.update_work_queue(\n                res.id, concurrency_limit=work_queue_concurrency\n            )\n\n        # we assume storage was already saved\n        storage_document_id = getattr(self.storage, \"_block_document_id\", None)\n        deployment_id = await client.create_deployment(\n            flow_id=flow_id,\n            name=self.name,\n            work_queue_name=self.work_queue_name,\n            work_pool_name=self.work_pool_name,\n            version=self.version,\n            schedule=self.schedule,\n            is_schedule_active=self.is_schedule_active,\n            parameters=self.parameters,\n            description=self.description,\n            tags=self.tags,\n            manifest_path=self.manifest_path,  # allows for backwards YAML compat\n            path=self.path,\n            entrypoint=self.entrypoint,\n            infra_overrides=self.infra_overrides,\n            storage_document_id=storage_document_id,\n            infrastructure_document_id=infrastructure_document_id,\n            parameter_openapi_schema=self.parameter_openapi_schema.dict(),\n            enforce_parameter_schema=self.enforce_parameter_schema,\n        )\n\n        if client.server_type == ServerType.CLOUD:\n            # The triggers defined in the deployment spec are, essentially,\n            # anonymous and attempting truly sync them with cloud is not\n            # feasible. Instead, we remove all automations that are owned\n            # by the deployment, meaning that they were created via this\n            # mechanism below, and then recreate them.\n            await client.delete_resource_owned_automations(\n                f\"prefect.deployment.{deployment_id}\"\n            )\n            for trigger in self.triggers:\n                trigger.set_deployment_id(deployment_id)\n                await client.create_automation(trigger.as_automation())\n\n        return deployment_id\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.build_from_flow","title":"build_from_flow async classmethod","text":"

    Configure a deployment for a given flow.

    Parameters:

    Name Type Description Default flow Flow

    A flow function to deploy

    required name str

    A name for the deployment

    required output optional

    if provided, the full deployment specification will be written as a YAML file in the location specified by output

    None skip_upload bool

    if True, deployment files are not automatically uploaded to remote storage

    False ignore_file str

    an optional path to a .prefectignore file that specifies filename patterns to ignore when uploading to remote storage; if not provided, looks for .prefectignore in the current working directory

    '.prefectignore' apply bool

    if True, the deployment is automatically registered with the API

    False load_existing bool

    if True, load any settings that may already be configured for the named deployment server-side (e.g., schedules, default parameter values, etc.)

    True **kwargs

    other keyword arguments to pass to the constructor for the Deployment class

    {} Source code in prefect/deployments/deployments.py
    @classmethod\n@sync_compatible\nasync def build_from_flow(\n    cls,\n    flow: Flow,\n    name: str,\n    output: str = None,\n    skip_upload: bool = False,\n    ignore_file: str = \".prefectignore\",\n    apply: bool = False,\n    load_existing: bool = True,\n    **kwargs,\n) -> \"Deployment\":\n    \"\"\"\n    Configure a deployment for a given flow.\n\n    Args:\n        flow: A flow function to deploy\n        name: A name for the deployment\n        output (optional): if provided, the full deployment specification will be\n            written as a YAML file in the location specified by `output`\n        skip_upload: if True, deployment files are not automatically uploaded to\n            remote storage\n        ignore_file: an optional path to a `.prefectignore` file that specifies\n            filename patterns to ignore when uploading to remote storage; if not\n            provided, looks for `.prefectignore` in the current working directory\n        apply: if True, the deployment is automatically registered with the API\n        load_existing: if True, load any settings that may already be configured for\n            the named deployment server-side (e.g., schedules, default parameter\n            values, etc.)\n        **kwargs: other keyword arguments to pass to the constructor for the\n            `Deployment` class\n    \"\"\"\n    if not name:\n        raise ValueError(\"A deployment name must be provided.\")\n\n    # note that `deployment.load` only updates settings that were *not*\n    # provided at initialization\n    deployment = cls(name=name, **kwargs)\n    deployment.flow_name = flow.name\n    if not deployment.entrypoint:\n        ## first see if an entrypoint can be determined\n        flow_file = getattr(flow, \"__globals__\", {}).get(\"__file__\")\n        mod_name = getattr(flow, \"__module__\", None)\n        if not flow_file:\n            if not mod_name:\n                # todo, check if the file location was manually set already\n                raise ValueError(\"Could not determine flow's file location.\")\n            module = importlib.import_module(mod_name)\n            flow_file = getattr(module, \"__file__\", None)\n            if not flow_file:\n                raise ValueError(\"Could not determine flow's file location.\")\n\n        # set entrypoint\n        entry_path = Path(flow_file).absolute().relative_to(Path(\".\").absolute())\n        deployment.entrypoint = f\"{entry_path}:{flow.fn.__name__}\"\n\n    if load_existing:\n        await deployment.load()\n\n    # set a few attributes for this flow object\n    deployment.parameter_openapi_schema = parameter_schema(flow)\n\n    # ensure the ignore file exists\n    if not Path(ignore_file).exists():\n        Path(ignore_file).touch()\n\n    if not deployment.version:\n        deployment.version = flow.version\n    if not deployment.description:\n        deployment.description = flow.description\n\n    # proxy for whether infra is docker-based\n    is_docker_based = hasattr(deployment.infrastructure, \"image\")\n\n    if not deployment.storage and not is_docker_based and not deployment.path:\n        deployment.path = str(Path(\".\").absolute())\n    elif not deployment.storage and is_docker_based:\n        # only update if a path is not already set\n        if not deployment.path:\n            deployment.path = \"/opt/prefect/flows\"\n\n    if not skip_upload:\n        if (\n            deployment.storage\n            and \"put-directory\" in deployment.storage.get_block_capabilities()\n        ):\n            await deployment.upload_to_storage(ignore_file=ignore_file)\n\n    if output:\n        await deployment.to_yaml(output)\n\n    if apply:\n        await deployment.apply()\n\n    return deployment\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.handle_openapi_schema","title":"handle_openapi_schema","text":"

    This method ensures setting a value of None is handled gracefully.

    Source code in prefect/deployments/deployments.py
    @validator(\"parameter_openapi_schema\", pre=True)\ndef handle_openapi_schema(cls, value):\n    \"\"\"\n    This method ensures setting a value of `None` is handled gracefully.\n    \"\"\"\n    if value is None:\n        return ParameterSchema()\n    return value\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.load","title":"load async","text":"

    Queries the API for a deployment with this name for this flow, and if found, prepopulates any settings that were not set at initialization.

    Returns a boolean specifying whether a load was successful or not.

    Raises:

    Type Description -ValueError

    if both name and flow name are not set

    Source code in prefect/deployments/deployments.py
    @sync_compatible\nasync def load(self) -> bool:\n    \"\"\"\n    Queries the API for a deployment with this name for this flow, and if found,\n    prepopulates any settings that were not set at initialization.\n\n    Returns a boolean specifying whether a load was successful or not.\n\n    Raises:\n        - ValueError: if both name and flow name are not set\n    \"\"\"\n    if not self.name or not self.flow_name:\n        raise ValueError(\"Both a deployment name and flow name must be provided.\")\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(\n                f\"{self.flow_name}/{self.name}\"\n            )\n            if deployment.storage_document_id:\n                Block._from_block_document(\n                    await client.read_block_document(deployment.storage_document_id)\n                )\n\n            excluded_fields = self.__fields_set__.union(\n                {\n                    \"infrastructure\",\n                    \"storage\",\n                    \"timestamp\",\n                    \"triggers\",\n                    \"enforce_parameter_schema\",\n                }\n            )\n            for field in set(self.__fields__.keys()) - excluded_fields:\n                new_value = getattr(deployment, field)\n                setattr(self, field, new_value)\n\n            if \"infrastructure\" not in self.__fields_set__:\n                if deployment.infrastructure_document_id:\n                    self.infrastructure = Block._from_block_document(\n                        await client.read_block_document(\n                            deployment.infrastructure_document_id\n                        )\n                    )\n            if \"storage\" not in self.__fields_set__:\n                if deployment.storage_document_id:\n                    self.storage = Block._from_block_document(\n                        await client.read_block_document(\n                            deployment.storage_document_id\n                        )\n                    )\n        except ObjectNotFound:\n            return False\n    return True\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.update","title":"update async","text":"

    Performs an in-place update with the provided settings.

    Parameters:

    Name Type Description Default ignore_none bool

    if True, all None values are ignored when performing the update

    False Source code in prefect/deployments/deployments.py
    @sync_compatible\nasync def update(self, ignore_none: bool = False, **kwargs):\n    \"\"\"\n    Performs an in-place update with the provided settings.\n\n    Args:\n        ignore_none: if True, all `None` values are ignored when performing the\n            update\n    \"\"\"\n    unknown_keys = set(kwargs.keys()) - set(self.dict().keys())\n    if unknown_keys:\n        raise ValueError(\n            f\"Received unexpected attributes: {', '.join(unknown_keys)}\"\n        )\n    for key, value in kwargs.items():\n        if ignore_none and value is None:\n            continue\n        setattr(self, key, value)\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.upload_to_storage","title":"upload_to_storage async","text":"

    Uploads the workflow this deployment represents using a provided storage block; if no block is provided, defaults to configuring self for local storage.

    Parameters:

    Name Type Description Default storage_block str

    a string reference a remote storage block slug $type/$name; if provided, used to upload the workflow's project

    None ignore_file str

    an optional path to a .prefectignore file that specifies filename patterns to ignore when uploading to remote storage; if not provided, looks for .prefectignore in the current working directory

    '.prefectignore' Source code in prefect/deployments/deployments.py
    @sync_compatible\nasync def upload_to_storage(\n    self, storage_block: str = None, ignore_file: str = \".prefectignore\"\n) -> Optional[int]:\n    \"\"\"\n    Uploads the workflow this deployment represents using a provided storage block;\n    if no block is provided, defaults to configuring self for local storage.\n\n    Args:\n        storage_block: a string reference a remote storage block slug `$type/$name`;\n            if provided, used to upload the workflow's project\n        ignore_file: an optional path to a `.prefectignore` file that specifies\n            filename patterns to ignore when uploading to remote storage; if not\n            provided, looks for `.prefectignore` in the current working directory\n    \"\"\"\n    file_count = None\n    if storage_block:\n        storage = await Block.load(storage_block)\n\n        if \"put-directory\" not in storage.get_block_capabilities():\n            raise BlockMissingCapabilities(\n                f\"Storage block {storage!r} missing 'put-directory' capability.\"\n            )\n\n        self.storage = storage\n\n        # upload current directory to storage location\n        file_count = await self.storage.put_directory(\n            ignore_file=ignore_file, to_path=self.path\n        )\n    elif self.storage:\n        if \"put-directory\" not in self.storage.get_block_capabilities():\n            raise BlockMissingCapabilities(\n                f\"Storage block {self.storage!r} missing 'put-directory'\"\n                \" capability.\"\n            )\n\n        file_count = await self.storage.put_directory(\n            ignore_file=ignore_file, to_path=self.path\n        )\n\n    # persists storage now in case it contains secret values\n    if self.storage and not self.storage._block_document_id:\n        await self.storage._save(is_anonymous=True)\n\n    return file_count\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.validate_automation_names","title":"validate_automation_names","text":"

    Ensure that each trigger has a name for its automation if none is provided.

    Source code in prefect/deployments/deployments.py
    @validator(\"triggers\")\ndef validate_automation_names(cls, field_value, values, field, config):\n    \"\"\"Ensure that each trigger has a name for its automation if none is provided.\"\"\"\n    for i, trigger in enumerate(field_value, start=1):\n        if trigger.name is None:\n            trigger.name = f\"{values['name']}__automation_{i}\"\n\n    return field_value\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.load_deployments_from_yaml","title":"load_deployments_from_yaml","text":"

    Load deployments from a yaml file.

    Source code in prefect/deployments/deployments.py
    def load_deployments_from_yaml(\n    path: str,\n) -> PrefectObjectRegistry:\n    \"\"\"\n    Load deployments from a yaml file.\n    \"\"\"\n    with open(path, \"r\") as f:\n        contents = f.read()\n\n    # Parse into a yaml tree to retrieve separate documents\n    nodes = yaml.compose_all(contents)\n\n    with PrefectObjectRegistry(capture_failures=True) as registry:\n        for node in nodes:\n            with tmpchdir(path):\n                deployment_dict = yaml.safe_load(yaml.serialize(node))\n                # The return value is not necessary, just instantiating the Deployment\n                # is enough to get it recorded on the registry\n                parse_obj_as(Deployment, deployment_dict)\n\n    return registry\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.load_flow_from_flow_run","title":"load_flow_from_flow_run async","text":"

    Load a flow from the location/script provided in a deployment's storage document.

    If ignore_storage=True is provided, no pull from remote storage occurs. This flag is largely for testing, and assumes the flow is already available locally.

    Source code in prefect/deployments/deployments.py
    @inject_client\nasync def load_flow_from_flow_run(\n    flow_run: FlowRun,\n    client: PrefectClient,\n    ignore_storage: bool = False,\n    storage_base_path: Optional[str] = None,\n) -> Flow:\n    \"\"\"\n    Load a flow from the location/script provided in a deployment's storage document.\n\n    If `ignore_storage=True` is provided, no pull from remote storage occurs.  This flag\n    is largely for testing, and assumes the flow is already available locally.\n    \"\"\"\n    deployment = await client.read_deployment(flow_run.deployment_id)\n    logger = flow_run_logger(flow_run)\n\n    runner_storage_base_path = storage_base_path or os.environ.get(\n        \"PREFECT__STORAGE_BASE_PATH\"\n    )\n\n    if not ignore_storage and not deployment.pull_steps:\n        sys.path.insert(0, \".\")\n        if deployment.storage_document_id:\n            storage_document = await client.read_block_document(\n                deployment.storage_document_id\n            )\n            storage_block = Block._from_block_document(storage_document)\n        else:\n            basepath = deployment.path or Path(deployment.manifest_path).parent\n            if runner_storage_base_path:\n                basepath = str(basepath).replace(\n                    \"$STORAGE_BASE_PATH\", runner_storage_base_path\n                )\n            storage_block = LocalFileSystem(basepath=basepath)\n\n        from_path = (\n            str(deployment.path).replace(\"$STORAGE_BASE_PATH\", runner_storage_base_path)\n            if runner_storage_base_path and deployment.path\n            else deployment.path\n        )\n        logger.info(f\"Downloading flow code from storage at {from_path!r}\")\n        await storage_block.get_directory(from_path=from_path, local_path=\".\")\n\n    if deployment.pull_steps:\n        logger.debug(f\"Running {len(deployment.pull_steps)} deployment pull steps\")\n        output = await run_steps(deployment.pull_steps)\n        if output.get(\"directory\"):\n            logger.debug(f\"Changing working directory to {output['directory']!r}\")\n            os.chdir(output[\"directory\"])\n\n    import_path = relative_path_to_current_platform(deployment.entrypoint)\n    logger.debug(f\"Importing flow code from '{import_path}'\")\n\n    # for backwards compat\n    if deployment.manifest_path:\n        with open(deployment.manifest_path, \"r\") as f:\n            import_path = json.load(f)[\"import_path\"]\n            import_path = (\n                Path(deployment.manifest_path).parent / import_path\n            ).absolute()\n    flow = await run_sync_in_worker_thread(load_flow_from_entrypoint, str(import_path))\n    return flow\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.run_deployment","title":"run_deployment async","text":"

    Create a flow run for a deployment and return it after completion or a timeout.

    This function will return when the created flow run enters any terminal state or the timeout is reached. If the timeout is reached and the flow run has not reached a terminal state, it will still be returned. When using a timeout, we suggest checking the state of the flow run if completion is important moving forward.

    Parameters:

    Name Type Description Default name Union[str, UUID]

    The deployment id or deployment name in the form: <slugified-flow-name>/<slugified-deployment-name>

    required parameters Optional[dict]

    Parameter overrides for this flow run. Merged with the deployment defaults.

    None scheduled_time Optional[datetime]

    The time to schedule the flow run for, defaults to scheduling the flow run to start now.

    None flow_run_name Optional[str]

    A name for the created flow run

    None timeout Optional[float]

    The amount of time to wait for the flow run to complete before returning. Setting timeout to 0 will return the flow run immediately. Setting timeout to None will allow this function to poll indefinitely. Defaults to None

    None poll_interval Optional[float]

    The number of seconds between polls

    5 tags Optional[Iterable[str]]

    A list of tags to associate with this flow run; note that tags are used only for organizational purposes.

    None idempotency_key Optional[str]

    A unique value to recognize retries of the same run, and prevent creating multiple flow runs.

    None work_queue_name Optional[str]

    The name of a work queue to use for this run. Defaults to the default work queue for the deployment.

    None Source code in prefect/deployments/deployments.py
    @sync_compatible\n@inject_client\nasync def run_deployment(\n    name: Union[str, UUID],\n    client: Optional[PrefectClient] = None,\n    parameters: Optional[dict] = None,\n    scheduled_time: Optional[datetime] = None,\n    flow_run_name: Optional[str] = None,\n    timeout: Optional[float] = None,\n    poll_interval: Optional[float] = 5,\n    tags: Optional[Iterable[str]] = None,\n    idempotency_key: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n):\n    \"\"\"\n    Create a flow run for a deployment and return it after completion or a timeout.\n\n    This function will return when the created flow run enters any terminal state or\n    the timeout is reached. If the timeout is reached and the flow run has not reached\n    a terminal state, it will still be returned. When using a timeout, we suggest\n    checking the state of the flow run if completion is important moving forward.\n\n    Args:\n        name: The deployment id or deployment name in the form:\n            `<slugified-flow-name>/<slugified-deployment-name>`\n        parameters: Parameter overrides for this flow run. Merged with the deployment\n            defaults.\n        scheduled_time: The time to schedule the flow run for, defaults to scheduling\n            the flow run to start now.\n        flow_run_name: A name for the created flow run\n        timeout: The amount of time to wait for the flow run to complete before\n            returning. Setting `timeout` to 0 will return the flow run immediately.\n            Setting `timeout` to None will allow this function to poll indefinitely.\n            Defaults to None\n        poll_interval: The number of seconds between polls\n        tags: A list of tags to associate with this flow run; note that tags are used\n            only for organizational purposes.\n        idempotency_key: A unique value to recognize retries of the same run, and\n            prevent creating multiple flow runs.\n        work_queue_name: The name of a work queue to use for this run. Defaults to\n            the default work queue for the deployment.\n    \"\"\"\n    if timeout is not None and timeout < 0:\n        raise ValueError(\"`timeout` cannot be negative\")\n\n    if scheduled_time is None:\n        scheduled_time = pendulum.now(\"UTC\")\n\n    parameters = parameters or {}\n\n    deployment_id = None\n\n    if isinstance(name, UUID):\n        deployment_id = name\n    else:\n        try:\n            deployment_id = UUID(name)\n        except ValueError:\n            pass\n\n    if deployment_id:\n        deployment = await client.read_deployment(deployment_id=deployment_id)\n    else:\n        deployment = await client.read_deployment_by_name(name)\n\n    flow_run_ctx = FlowRunContext.get()\n    task_run_ctx = TaskRunContext.get()\n    if flow_run_ctx or task_run_ctx:\n        # This was called from a flow. Link the flow run as a subflow.\n        from prefect.engine import (\n            Pending,\n            _dynamic_key_for_task_run,\n            collect_task_run_inputs,\n        )\n\n        task_inputs = {\n            k: await collect_task_run_inputs(v) for k, v in parameters.items()\n        }\n\n        if deployment_id:\n            flow = await client.read_flow(deployment.flow_id)\n            deployment_name = f\"{flow.name}/{deployment.name}\"\n        else:\n            deployment_name = name\n\n        # Generate a task in the parent flow run to represent the result of the subflow\n        dummy_task = Task(\n            name=deployment_name,\n            fn=lambda: None,\n            version=deployment.version,\n        )\n        # Override the default task key to include the deployment name\n        dummy_task.task_key = f\"{__name__}.run_deployment.{slugify(deployment_name)}\"\n        flow_run_id = (\n            flow_run_ctx.flow_run.id\n            if flow_run_ctx\n            else task_run_ctx.task_run.flow_run_id\n        )\n        dynamic_key = (\n            _dynamic_key_for_task_run(flow_run_ctx, dummy_task)\n            if flow_run_ctx\n            else task_run_ctx.task_run.dynamic_key\n        )\n        parent_task_run = await client.create_task_run(\n            task=dummy_task,\n            flow_run_id=flow_run_id,\n            dynamic_key=dynamic_key,\n            task_inputs=task_inputs,\n            state=Pending(),\n        )\n        parent_task_run_id = parent_task_run.id\n    else:\n        parent_task_run_id = None\n\n    flow_run = await client.create_flow_run_from_deployment(\n        deployment.id,\n        parameters=parameters,\n        state=Scheduled(scheduled_time=scheduled_time),\n        name=flow_run_name,\n        tags=tags,\n        idempotency_key=idempotency_key,\n        parent_task_run_id=parent_task_run_id,\n        work_queue_name=work_queue_name,\n    )\n\n    flow_run_id = flow_run.id\n\n    if timeout == 0:\n        return flow_run\n\n    with anyio.move_on_after(timeout):\n        while True:\n            flow_run = await client.read_flow_run(flow_run_id)\n            flow_state = flow_run.state\n            if flow_state and flow_state.is_final():\n                return flow_run\n            await anyio.sleep(poll_interval)\n\n    return flow_run\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/runner/","title":"runner","text":"","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner","title":"prefect.deployments.runner","text":"

    Objects for creating and configuring deployments for flows using serve functionality.

    Example
    import time\nfrom prefect import flow, serve\n\n\n@flow\ndef slow_flow(sleep: int = 60):\n    \"Sleepy flow - sleeps the provided amount of time (in seconds).\"\n    time.sleep(sleep)\n\n\n@flow\ndef fast_flow():\n    \"Fastest flow this side of the Mississippi.\"\n    return\n\n\nif __name__ == \"__main__\":\n    # to_deployment creates RunnerDeployment instances\n    slow_deploy = slow_flow.to_deployment(name=\"sleeper\", interval=45)\n    fast_deploy = fast_flow.to_deployment(name=\"fast\")\n\n    serve(slow_deploy, fast_deploy)\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.DeploymentApplyError","title":"DeploymentApplyError","text":"

    Bases: RuntimeError

    Raised when an error occurs while applying a deployment.

    Source code in prefect/deployments/runner.py
    class DeploymentApplyError(RuntimeError):\n    \"\"\"\n    Raised when an error occurs while applying a deployment.\n    \"\"\"\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.DeploymentImage","title":"DeploymentImage","text":"

    Configuration used to build and push a Docker image for a deployment.

    Attributes:

    Name Type Description name

    The name of the Docker image to build, including the registry and repository.

    tag

    The tag to apply to the built image.

    dockerfile

    The path to the Dockerfile to use for building the image. If not provided, a default Dockerfile will be generated.

    **build_kwargs

    Additional keyword arguments to pass to the Docker build request. See the docker-py documentation for more information.

    Source code in prefect/deployments/runner.py
    class DeploymentImage:\n    \"\"\"\n    Configuration used to build and push a Docker image for a deployment.\n\n    Attributes:\n        name: The name of the Docker image to build, including the registry and\n            repository.\n        tag: The tag to apply to the built image.\n        dockerfile: The path to the Dockerfile to use for building the image. If\n            not provided, a default Dockerfile will be generated.\n        **build_kwargs: Additional keyword arguments to pass to the Docker build request.\n            See the [`docker-py` documentation](https://docker-py.readthedocs.io/en/stable/images.html#docker.models.images.ImageCollection.build)\n            for more information.\n\n    \"\"\"\n\n    def __init__(self, name, tag=None, dockerfile=\"auto\", **build_kwargs):\n        image_name, image_tag = parse_image_tag(name)\n        if tag and image_tag:\n            raise ValueError(\n                f\"Only one tag can be provided - both {image_tag!r} and {tag!r} were\"\n                \" provided as tags.\"\n            )\n        namespace, repository = split_repository_path(image_name)\n        # if the provided image name does not include a namespace (registry URL or user/org name),\n        # use the default namespace\n        if not namespace:\n            namespace = PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE.value()\n        # join the namespace and repository to create the full image name\n        # ignore namespace if it is None\n        self.name = \"/\".join(filter(None, [namespace, repository]))\n        self.tag = tag or image_tag or slugify(pendulum.now(\"utc\").isoformat())\n        self.dockerfile = dockerfile\n        self.build_kwargs = build_kwargs\n\n    @property\n    def reference(self):\n        return f\"{self.name}:{self.tag}\"\n\n    def build(self):\n        full_image_name = self.reference\n        build_kwargs = self.build_kwargs.copy()\n        build_kwargs[\"context\"] = Path.cwd()\n        build_kwargs[\"tag\"] = full_image_name\n        build_kwargs[\"pull\"] = build_kwargs.get(\"pull\", True)\n\n        if self.dockerfile == \"auto\":\n            with generate_default_dockerfile():\n                build_image(**build_kwargs)\n        else:\n            build_kwargs[\"dockerfile\"] = self.dockerfile\n            build_image(**build_kwargs)\n\n    def push(self):\n        with docker_client() as client:\n            events = client.api.push(\n                repository=self.name, tag=self.tag, stream=True, decode=True\n            )\n            for event in events:\n                if \"error\" in event:\n                    raise PushError(event[\"error\"])\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment","title":"RunnerDeployment","text":"

    Bases: BaseModel

    A Prefect RunnerDeployment definition, used for specifying and building deployments.

    Attributes:

    Name Type Description name str

    A name for the deployment (required).

    version Optional[str]

    An optional version for the deployment; defaults to the flow's version

    description Optional[str]

    An optional description of the deployment; defaults to the flow's description

    tags List[str]

    An optional list of tags to associate with this deployment; note that tags are used only for organizational purposes. For delegating work to agents, see work_queue_name.

    schedule Optional[SCHEDULE_TYPES]

    A schedule to run this deployment on, once registered

    is_schedule_active Optional[bool]

    Whether or not the schedule is active

    parameters Dict[str, Any]

    A dictionary of parameter values to pass to runs created from this deployment

    path Dict[str, Any]

    The path to the working directory for the workflow, relative to remote storage or, if stored on a local filesystem, an absolute path

    entrypoint Optional[str]

    The path to the entrypoint for the workflow, always relative to the path

    parameter_openapi_schema Optional[str]

    The parameter schema of the flow, including defaults.

    enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    job_variables Dict[str, Any]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    Source code in prefect/deployments/runner.py
    class RunnerDeployment(BaseModel):\n    \"\"\"\n    A Prefect RunnerDeployment definition, used for specifying and building deployments.\n\n    Attributes:\n        name: A name for the deployment (required).\n        version: An optional version for the deployment; defaults to the flow's version\n        description: An optional description of the deployment; defaults to the flow's\n            description\n        tags: An optional list of tags to associate with this deployment; note that tags\n            are used only for organizational purposes. For delegating work to agents,\n            see `work_queue_name`.\n        schedule: A schedule to run this deployment on, once registered\n        is_schedule_active: Whether or not the schedule is active\n        parameters: A dictionary of parameter values to pass to runs created from this\n            deployment\n        path: The path to the working directory for the workflow, relative to remote\n            storage or, if stored on a local filesystem, an absolute path\n        entrypoint: The path to the entrypoint for the workflow, always relative to the\n            `path`\n        parameter_openapi_schema: The parameter schema of the flow, including defaults.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n    \"\"\"\n\n    class Config:\n        arbitrary_types_allowed = True\n\n    name: str = Field(..., description=\"The name of the deployment.\")\n    flow_name: Optional[str] = Field(\n        None, description=\"The name of the underlying flow; typically inferred.\"\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"An optional description of the deployment.\"\n    )\n    version: Optional[str] = Field(\n        default=None, description=\"An optional version for the deployment.\"\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"One of more tags to apply to this deployment.\",\n    )\n    schedule: Optional[SCHEDULE_TYPES] = None\n    is_schedule_active: Optional[bool] = Field(\n        default=None, description=\"Whether or not the schedule is active.\"\n    )\n    parameters: Dict[str, Any] = Field(default_factory=dict)\n    entrypoint: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the entrypoint for the workflow, relative to the `path`.\"\n        ),\n    )\n    triggers: List[DeploymentTrigger] = Field(\n        default_factory=list,\n        description=\"The triggers that should cause this deployment to run.\",\n    )\n    enforce_parameter_schema: bool = Field(\n        default=False,\n        description=(\n            \"Whether or not the Prefect API should enforce the parameter schema for\"\n            \" this deployment.\"\n        ),\n    )\n    storage: Optional[RunnerStorage] = Field(\n        default=None,\n        description=(\n            \"The storage object used to retrieve flow code for this deployment.\"\n        ),\n    )\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The name of the work pool to use for this deployment. Only used when\"\n            \" the deployment is registered with a built runner.\"\n        ),\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The name of the work queue to use for this deployment. Only used when\"\n            \" the deployment is registered with a built runner.\"\n        ),\n    )\n    job_variables: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=(\n            \"Job variables used to override the default values of a work pool\"\n            \" base job template. Only used when the deployment is registered with\"\n            \" a built runner.\"\n        ),\n    )\n    _path: Optional[str] = PrivateAttr(\n        default=None,\n    )\n    _parameter_openapi_schema: ParameterSchema = PrivateAttr(\n        default_factory=ParameterSchema,\n    )\n\n    @validator(\"triggers\", allow_reuse=True)\n    def validate_automation_names(cls, field_value, values, field, config):\n        \"\"\"Ensure that each trigger has a name for its automation if none is provided.\"\"\"\n        for i, trigger in enumerate(field_value, start=1):\n            if trigger.name is None:\n                trigger.name = f\"{values['name']}__automation_{i}\"\n\n        return field_value\n\n    @sync_compatible\n    async def apply(\n        self, work_pool_name: Optional[str] = None, image: Optional[str] = None\n    ) -> UUID:\n        \"\"\"\n        Registers this deployment with the API and returns the deployment's ID.\n\n        Args:\n            work_pool_name: The name of the work pool to use for this\n                deployment.\n            image: The registry, name, and tag of the Docker image to\n                use for this deployment. Only used when the deployment is\n                deployed to a work pool.\n\n        Returns:\n            The ID of the created deployment.\n        \"\"\"\n        work_pool_name = work_pool_name or self.work_pool_name\n\n        if image and not work_pool_name:\n            raise ValueError(\n                \"An image can only be provided when registering a deployment with a\"\n                \" work pool.\"\n            )\n\n        if self.work_queue_name and not work_pool_name:\n            raise ValueError(\n                \"A work queue can only be provided when registering a deployment with\"\n                \" a work pool.\"\n            )\n\n        if self.job_variables and not work_pool_name:\n            raise ValueError(\n                \"Job variables can only be provided when registering a deployment\"\n                \" with a work pool.\"\n            )\n\n        async with get_client() as client:\n            flow_id = await client.create_flow_from_name(self.flow_name)\n\n            create_payload = dict(\n                flow_id=flow_id,\n                name=self.name,\n                work_queue_name=self.work_queue_name,\n                work_pool_name=work_pool_name,\n                version=self.version,\n                schedule=self.schedule,\n                is_schedule_active=self.is_schedule_active,\n                parameters=self.parameters,\n                description=self.description,\n                tags=self.tags,\n                path=self._path,\n                entrypoint=self.entrypoint,\n                storage_document_id=None,\n                infrastructure_document_id=None,\n                parameter_openapi_schema=self._parameter_openapi_schema.dict(),\n                enforce_parameter_schema=self.enforce_parameter_schema,\n            )\n\n            if work_pool_name:\n                create_payload[\"infra_overrides\"] = self.job_variables\n                if image:\n                    create_payload[\"infra_overrides\"][\"image\"] = image\n                create_payload[\"path\"] = None if self.storage else self._path\n                create_payload[\"pull_steps\"] = (\n                    [self.storage.to_pull_step()] if self.storage else []\n                )\n\n            try:\n                deployment_id = await client.create_deployment(**create_payload)\n            except Exception as exc:\n                if isinstance(exc, PrefectHTTPStatusError):\n                    detail = exc.response.json().get(\"detail\")\n                    if detail:\n                        raise DeploymentApplyError(detail) from exc\n                raise DeploymentApplyError(\n                    f\"Error while applying deployment: {str(exc)}\"\n                ) from exc\n\n            if client.server_type == ServerType.CLOUD:\n                # The triggers defined in the deployment spec are, essentially,\n                # anonymous and attempting truly sync them with cloud is not\n                # feasible. Instead, we remove all automations that are owned\n                # by the deployment, meaning that they were created via this\n                # mechanism below, and then recreate them.\n                await client.delete_resource_owned_automations(\n                    f\"prefect.deployment.{deployment_id}\"\n                )\n                for trigger in self.triggers:\n                    trigger.set_deployment_id(deployment_id)\n                    await client.create_automation(trigger.as_automation())\n\n            return deployment_id\n\n    @staticmethod\n    def _construct_schedule(\n        interval: Optional[Union[int, float, timedelta]] = None,\n        anchor_date: Optional[Union[datetime, str]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        timezone: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n    ) -> Optional[SCHEDULE_TYPES]:\n        \"\"\"\n        Construct a schedule from the provided arguments.\n\n        This is a single path for all serve schedules. If schedule is provided,\n        it is returned. Otherwise, the other arguments are used to construct a schedule.\n\n        Args:\n            interval: An interval on which to schedule runs. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            anchor_date: The start date for an interval schedule.\n            cron: A cron schedule for runs.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            timezone: A timezone to use for the schedule.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n        \"\"\"\n        num_schedules = sum(\n            1 for entry in (interval, cron, rrule, schedule) if entry is not None\n        )\n        if num_schedules > 1:\n            raise ValueError(\n                \"Only one of interval, cron, rrule, or schedule can be provided.\"\n            )\n\n        if schedule:\n            return schedule\n        elif interval or cron or rrule:\n            return construct_schedule(\n                interval=interval,\n                cron=cron,\n                rrule=rrule,\n                timezone=timezone,\n                anchor_date=anchor_date,\n            )\n        else:\n            return None\n\n    def _set_defaults_from_flow(self, flow: \"Flow\"):\n        self._parameter_openapi_schema = parameter_schema(flow)\n\n        if not self.version:\n            self.version = flow.version\n        if not self.description:\n            self.description = flow.description\n\n    @classmethod\n    def from_flow(\n        cls,\n        flow: \"Flow\",\n        name: str,\n        interval: Optional[Union[int, float, timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        work_pool_name: Optional[str] = None,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[Dict[str, Any]] = None,\n    ) -> \"RunnerDeployment\":\n        \"\"\"\n        Configure a deployment for a given flow.\n\n        Args:\n            flow: A flow function to deploy\n            name: A name for the deployment\n            interval: An interval on which to execute the current flow. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this flow.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            triggers: A list of triggers that should kick of a run of this flow.\n            parameters: A dictionary of default parameter values to pass to runs of this flow.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for this deployment.\n            work_pool_name: The name of the work pool to use for this deployment.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n                available settings.\n        \"\"\"\n        schedule = cls._construct_schedule(\n            interval=interval, cron=cron, rrule=rrule, schedule=schedule\n        )\n\n        job_variables = job_variables or {}\n\n        deployment = cls(\n            name=Path(name).stem,\n            flow_name=flow.name,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags or [],\n            triggers=triggers or [],\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n\n        if not deployment.entrypoint:\n            no_file_location_error = (\n                \"Flows defined interactively cannot be deployed. Check out the\"\n                \" quickstart guide for help getting started:\"\n                \" https://docs.prefect.io/latest/getting-started/quickstart\"\n            )\n            ## first see if an entrypoint can be determined\n            flow_file = getattr(flow, \"__globals__\", {}).get(\"__file__\")\n            mod_name = getattr(flow, \"__module__\", None)\n            if not flow_file:\n                if not mod_name:\n                    raise ValueError(no_file_location_error)\n                try:\n                    module = importlib.import_module(mod_name)\n                    flow_file = getattr(module, \"__file__\", None)\n                except ModuleNotFoundError as exc:\n                    if \"__prefect_loader__\" in str(exc):\n                        raise ValueError(\n                            \"Cannot create a RunnerDeployment from a flow that has been\"\n                            \" loaded from an entrypoint. To deploy a flow via\"\n                            \" entrypoint, use RunnerDeployment.from_entrypoint instead.\"\n                        )\n                    raise ValueError(no_file_location_error)\n                if not flow_file:\n                    raise ValueError(no_file_location_error)\n\n            # set entrypoint\n            entry_path = Path(flow_file).absolute().relative_to(Path.cwd().absolute())\n            deployment.entrypoint = f\"{entry_path}:{flow.fn.__name__}\"\n\n        if not deployment._path:\n            deployment._path = \".\"\n\n        cls._set_defaults_from_flow(deployment, flow)\n\n        return deployment\n\n    @classmethod\n    def from_entrypoint(\n        cls,\n        entrypoint: str,\n        name: str,\n        interval: Optional[Union[int, float, timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        work_pool_name: Optional[str] = None,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[Dict[str, Any]] = None,\n    ) -> \"RunnerDeployment\":\n        \"\"\"\n        Configure a deployment for a given flow located at a given entrypoint.\n\n        Args:\n            entrypoint:  The path to a file containing a flow and the name of the flow function in\n                the format `./path/to/file.py:flow_func_name`.\n            name: A name for the deployment\n            interval: An interval on which to execute the current flow. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this flow.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            triggers: A list of triggers that should kick of a run of this flow.\n            parameters: A dictionary of default parameter values to pass to runs of this flow.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for this deployment.\n            work_pool_name: The name of the work pool to use for this deployment.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n                available settings.\n        \"\"\"\n        from prefect.flows import load_flow_from_entrypoint\n\n        job_variables = job_variables or {}\n        flow = load_flow_from_entrypoint(entrypoint)\n\n        schedule = cls._construct_schedule(\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n        )\n\n        deployment = cls(\n            name=Path(name).stem,\n            flow_name=flow.name,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags or [],\n            triggers=triggers or [],\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            entrypoint=entrypoint,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n        deployment._path = str(Path.cwd())\n\n        cls._set_defaults_from_flow(deployment, flow)\n\n        return deployment\n\n    @classmethod\n    @sync_compatible\n    async def from_storage(\n        cls,\n        storage: RunnerStorage,\n        entrypoint: str,\n        name: str,\n        interval: Optional[Union[int, float, timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        work_pool_name: Optional[str] = None,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[Dict[str, Any]] = None,\n    ):\n        \"\"\"\n        Create a RunnerDeployment from a flow located at a given entrypoint and stored in a\n        local storage location.\n\n        Args:\n            entrypoint:  The path to a file containing a flow and the name of the flow function in\n                the format `./path/to/file.py:flow_func_name`.\n            name: A name for the deployment\n            storage: A storage object to use for retrieving flow code. If not provided, a\n                URL must be provided.\n            interval: An interval on which to execute the current flow. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this flow.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            triggers: A list of triggers that should kick of a run of this flow.\n            parameters: A dictionary of default parameter values to pass to runs of this flow.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for this deployment.\n            work_pool_name: The name of the work pool to use for this deployment.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n                available settings.\n        \"\"\"\n        from prefect.flows import load_flow_from_entrypoint\n\n        schedule = cls._construct_schedule(\n            interval=interval, cron=cron, rrule=rrule, schedule=schedule\n        )\n        job_variables = job_variables or {}\n\n        with tempfile.TemporaryDirectory() as tmpdir:\n            storage.set_base_path(Path(tmpdir))\n            await storage.pull_code()\n\n            full_entrypoint = str(storage.destination / entrypoint)\n            flow = await from_async.wait_for_call_in_new_thread(\n                create_call(load_flow_from_entrypoint, full_entrypoint)\n            )\n\n        deployment = cls(\n            name=Path(name).stem,\n            flow_name=flow.name,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags or [],\n            triggers=triggers or [],\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            entrypoint=entrypoint,\n            enforce_parameter_schema=enforce_parameter_schema,\n            storage=storage,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n        deployment._path = str(storage.destination).replace(\n            tmpdir, \"$STORAGE_BASE_PATH\"\n        )\n\n        cls._set_defaults_from_flow(deployment, flow)\n\n        return deployment\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.apply","title":"apply async","text":"

    Registers this deployment with the API and returns the deployment's ID.

    Parameters:

    Name Type Description Default work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None image Optional[str]

    The registry, name, and tag of the Docker image to use for this deployment. Only used when the deployment is deployed to a work pool.

    None

    Returns:

    Type Description UUID

    The ID of the created deployment.

    Source code in prefect/deployments/runner.py
    @sync_compatible\nasync def apply(\n    self, work_pool_name: Optional[str] = None, image: Optional[str] = None\n) -> UUID:\n    \"\"\"\n    Registers this deployment with the API and returns the deployment's ID.\n\n    Args:\n        work_pool_name: The name of the work pool to use for this\n            deployment.\n        image: The registry, name, and tag of the Docker image to\n            use for this deployment. Only used when the deployment is\n            deployed to a work pool.\n\n    Returns:\n        The ID of the created deployment.\n    \"\"\"\n    work_pool_name = work_pool_name or self.work_pool_name\n\n    if image and not work_pool_name:\n        raise ValueError(\n            \"An image can only be provided when registering a deployment with a\"\n            \" work pool.\"\n        )\n\n    if self.work_queue_name and not work_pool_name:\n        raise ValueError(\n            \"A work queue can only be provided when registering a deployment with\"\n            \" a work pool.\"\n        )\n\n    if self.job_variables and not work_pool_name:\n        raise ValueError(\n            \"Job variables can only be provided when registering a deployment\"\n            \" with a work pool.\"\n        )\n\n    async with get_client() as client:\n        flow_id = await client.create_flow_from_name(self.flow_name)\n\n        create_payload = dict(\n            flow_id=flow_id,\n            name=self.name,\n            work_queue_name=self.work_queue_name,\n            work_pool_name=work_pool_name,\n            version=self.version,\n            schedule=self.schedule,\n            is_schedule_active=self.is_schedule_active,\n            parameters=self.parameters,\n            description=self.description,\n            tags=self.tags,\n            path=self._path,\n            entrypoint=self.entrypoint,\n            storage_document_id=None,\n            infrastructure_document_id=None,\n            parameter_openapi_schema=self._parameter_openapi_schema.dict(),\n            enforce_parameter_schema=self.enforce_parameter_schema,\n        )\n\n        if work_pool_name:\n            create_payload[\"infra_overrides\"] = self.job_variables\n            if image:\n                create_payload[\"infra_overrides\"][\"image\"] = image\n            create_payload[\"path\"] = None if self.storage else self._path\n            create_payload[\"pull_steps\"] = (\n                [self.storage.to_pull_step()] if self.storage else []\n            )\n\n        try:\n            deployment_id = await client.create_deployment(**create_payload)\n        except Exception as exc:\n            if isinstance(exc, PrefectHTTPStatusError):\n                detail = exc.response.json().get(\"detail\")\n                if detail:\n                    raise DeploymentApplyError(detail) from exc\n            raise DeploymentApplyError(\n                f\"Error while applying deployment: {str(exc)}\"\n            ) from exc\n\n        if client.server_type == ServerType.CLOUD:\n            # The triggers defined in the deployment spec are, essentially,\n            # anonymous and attempting truly sync them with cloud is not\n            # feasible. Instead, we remove all automations that are owned\n            # by the deployment, meaning that they were created via this\n            # mechanism below, and then recreate them.\n            await client.delete_resource_owned_automations(\n                f\"prefect.deployment.{deployment_id}\"\n            )\n            for trigger in self.triggers:\n                trigger.set_deployment_id(deployment_id)\n                await client.create_automation(trigger.as_automation())\n\n        return deployment_id\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.from_entrypoint","title":"from_entrypoint classmethod","text":"

    Configure a deployment for a given flow located at a given entrypoint.

    Parameters:

    Name Type Description Default entrypoint str

    The path to a file containing a flow and the name of the flow function in the format ./path/to/file.py:flow_func_name.

    required name str

    A name for the deployment

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the current flow. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this flow.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object of when to execute runs of this flow. Used for advanced scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that should kick of a run of this flow.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this flow.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    False work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[Dict[str, Any]]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    None Source code in prefect/deployments/runner.py
    @classmethod\ndef from_entrypoint(\n    cls,\n    entrypoint: str,\n    name: str,\n    interval: Optional[Union[int, float, timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    work_pool_name: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[Dict[str, Any]] = None,\n) -> \"RunnerDeployment\":\n    \"\"\"\n    Configure a deployment for a given flow located at a given entrypoint.\n\n    Args:\n        entrypoint:  The path to a file containing a flow and the name of the flow function in\n            the format `./path/to/file.py:flow_func_name`.\n        name: A name for the deployment\n        interval: An interval on which to execute the current flow. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this flow.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        schedule: A schedule object of when to execute runs of this flow. Used for\n            advanced scheduling options like timezone.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        triggers: A list of triggers that should kick of a run of this flow.\n        parameters: A dictionary of default parameter values to pass to runs of this flow.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n    \"\"\"\n    from prefect.flows import load_flow_from_entrypoint\n\n    job_variables = job_variables or {}\n    flow = load_flow_from_entrypoint(entrypoint)\n\n    schedule = cls._construct_schedule(\n        interval=interval,\n        cron=cron,\n        rrule=rrule,\n        schedule=schedule,\n    )\n\n    deployment = cls(\n        name=Path(name).stem,\n        flow_name=flow.name,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        tags=tags or [],\n        triggers=triggers or [],\n        parameters=parameters or {},\n        description=description,\n        version=version,\n        entrypoint=entrypoint,\n        enforce_parameter_schema=enforce_parameter_schema,\n        work_pool_name=work_pool_name,\n        work_queue_name=work_queue_name,\n        job_variables=job_variables,\n    )\n    deployment._path = str(Path.cwd())\n\n    cls._set_defaults_from_flow(deployment, flow)\n\n    return deployment\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.from_flow","title":"from_flow classmethod","text":"

    Configure a deployment for a given flow.

    Parameters:

    Name Type Description Default flow Flow

    A flow function to deploy

    required name str

    A name for the deployment

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the current flow. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this flow.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object of when to execute runs of this flow. Used for advanced scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that should kick of a run of this flow.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this flow.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    False work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[Dict[str, Any]]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    None Source code in prefect/deployments/runner.py
    @classmethod\ndef from_flow(\n    cls,\n    flow: \"Flow\",\n    name: str,\n    interval: Optional[Union[int, float, timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    work_pool_name: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[Dict[str, Any]] = None,\n) -> \"RunnerDeployment\":\n    \"\"\"\n    Configure a deployment for a given flow.\n\n    Args:\n        flow: A flow function to deploy\n        name: A name for the deployment\n        interval: An interval on which to execute the current flow. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this flow.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        schedule: A schedule object of when to execute runs of this flow. Used for\n            advanced scheduling options like timezone.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        triggers: A list of triggers that should kick of a run of this flow.\n        parameters: A dictionary of default parameter values to pass to runs of this flow.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n    \"\"\"\n    schedule = cls._construct_schedule(\n        interval=interval, cron=cron, rrule=rrule, schedule=schedule\n    )\n\n    job_variables = job_variables or {}\n\n    deployment = cls(\n        name=Path(name).stem,\n        flow_name=flow.name,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        tags=tags or [],\n        triggers=triggers or [],\n        parameters=parameters or {},\n        description=description,\n        version=version,\n        enforce_parameter_schema=enforce_parameter_schema,\n        work_pool_name=work_pool_name,\n        work_queue_name=work_queue_name,\n        job_variables=job_variables,\n    )\n\n    if not deployment.entrypoint:\n        no_file_location_error = (\n            \"Flows defined interactively cannot be deployed. Check out the\"\n            \" quickstart guide for help getting started:\"\n            \" https://docs.prefect.io/latest/getting-started/quickstart\"\n        )\n        ## first see if an entrypoint can be determined\n        flow_file = getattr(flow, \"__globals__\", {}).get(\"__file__\")\n        mod_name = getattr(flow, \"__module__\", None)\n        if not flow_file:\n            if not mod_name:\n                raise ValueError(no_file_location_error)\n            try:\n                module = importlib.import_module(mod_name)\n                flow_file = getattr(module, \"__file__\", None)\n            except ModuleNotFoundError as exc:\n                if \"__prefect_loader__\" in str(exc):\n                    raise ValueError(\n                        \"Cannot create a RunnerDeployment from a flow that has been\"\n                        \" loaded from an entrypoint. To deploy a flow via\"\n                        \" entrypoint, use RunnerDeployment.from_entrypoint instead.\"\n                    )\n                raise ValueError(no_file_location_error)\n            if not flow_file:\n                raise ValueError(no_file_location_error)\n\n        # set entrypoint\n        entry_path = Path(flow_file).absolute().relative_to(Path.cwd().absolute())\n        deployment.entrypoint = f\"{entry_path}:{flow.fn.__name__}\"\n\n    if not deployment._path:\n        deployment._path = \".\"\n\n    cls._set_defaults_from_flow(deployment, flow)\n\n    return deployment\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.from_storage","title":"from_storage async classmethod","text":"

    Create a RunnerDeployment from a flow located at a given entrypoint and stored in a local storage location.

    Parameters:

    Name Type Description Default entrypoint str

    The path to a file containing a flow and the name of the flow function in the format ./path/to/file.py:flow_func_name.

    required name str

    A name for the deployment

    required storage RunnerStorage

    A storage object to use for retrieving flow code. If not provided, a URL must be provided.

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the current flow. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this flow.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object of when to execute runs of this flow. Used for advanced scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that should kick of a run of this flow.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this flow.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    False work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[Dict[str, Any]]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    None Source code in prefect/deployments/runner.py
    @classmethod\n@sync_compatible\nasync def from_storage(\n    cls,\n    storage: RunnerStorage,\n    entrypoint: str,\n    name: str,\n    interval: Optional[Union[int, float, timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    work_pool_name: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[Dict[str, Any]] = None,\n):\n    \"\"\"\n    Create a RunnerDeployment from a flow located at a given entrypoint and stored in a\n    local storage location.\n\n    Args:\n        entrypoint:  The path to a file containing a flow and the name of the flow function in\n            the format `./path/to/file.py:flow_func_name`.\n        name: A name for the deployment\n        storage: A storage object to use for retrieving flow code. If not provided, a\n            URL must be provided.\n        interval: An interval on which to execute the current flow. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this flow.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        schedule: A schedule object of when to execute runs of this flow. Used for\n            advanced scheduling options like timezone.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        triggers: A list of triggers that should kick of a run of this flow.\n        parameters: A dictionary of default parameter values to pass to runs of this flow.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n    \"\"\"\n    from prefect.flows import load_flow_from_entrypoint\n\n    schedule = cls._construct_schedule(\n        interval=interval, cron=cron, rrule=rrule, schedule=schedule\n    )\n    job_variables = job_variables or {}\n\n    with tempfile.TemporaryDirectory() as tmpdir:\n        storage.set_base_path(Path(tmpdir))\n        await storage.pull_code()\n\n        full_entrypoint = str(storage.destination / entrypoint)\n        flow = await from_async.wait_for_call_in_new_thread(\n            create_call(load_flow_from_entrypoint, full_entrypoint)\n        )\n\n    deployment = cls(\n        name=Path(name).stem,\n        flow_name=flow.name,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        tags=tags or [],\n        triggers=triggers or [],\n        parameters=parameters or {},\n        description=description,\n        version=version,\n        entrypoint=entrypoint,\n        enforce_parameter_schema=enforce_parameter_schema,\n        storage=storage,\n        work_pool_name=work_pool_name,\n        work_queue_name=work_queue_name,\n        job_variables=job_variables,\n    )\n    deployment._path = str(storage.destination).replace(\n        tmpdir, \"$STORAGE_BASE_PATH\"\n    )\n\n    cls._set_defaults_from_flow(deployment, flow)\n\n    return deployment\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.validate_automation_names","title":"validate_automation_names","text":"

    Ensure that each trigger has a name for its automation if none is provided.

    Source code in prefect/deployments/runner.py
    @validator(\"triggers\", allow_reuse=True)\ndef validate_automation_names(cls, field_value, values, field, config):\n    \"\"\"Ensure that each trigger has a name for its automation if none is provided.\"\"\"\n    for i, trigger in enumerate(field_value, start=1):\n        if trigger.name is None:\n            trigger.name = f\"{values['name']}__automation_{i}\"\n\n    return field_value\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.deploy","title":"deploy async","text":"

    Deploy the provided list of deployments to dynamic infrastructure via a work pool.

    By default, calling this function will build a Docker image for the deployments, push it to a registry, and create each deployment via the Prefect API that will run the corresponding flow on the given schedule.

    If you want to use an existing image, you can pass build=False to skip building and pushing an image.

    Parameters:

    Name Type Description Default *deployments RunnerDeployment

    A list of deployments to deploy.

    () work_pool_name Optional[str]

    The name of the work pool to use for these deployments. Defaults to the value of PREFECT_DEFAULT_WORK_POOL_NAME.

    None image Optional[Union[str, DeploymentImage]]

    The name of the Docker image to build, including the registry and repository. Pass a DeploymentImage instance to customize the Dockerfile used and build arguments.

    None build bool

    Whether or not to build a new image for the flow. If False, the provided image will be used as-is and pulled at runtime.

    True push bool

    Whether or not to skip pushing the built image to a registry.

    True print_next_steps_message bool

    Whether or not to print a message with next steps after deploying the deployments.

    True

    Returns:

    Type Description List[UUID]

    A list of deployment IDs for the created/updated deployments.

    Examples:

    Deploy a group of flows to a work pool:

    from prefect import deploy, flow\n\n@flow(log_prints=True)\ndef local_flow():\n    print(\"I'm a locally defined flow!\")\n\nif __name__ == \"__main__\":\n    deploy(\n        local_flow.to_deployment(name=\"example-deploy-local-flow\"),\n        flow.from_source(\n            source=\"https://github.com/org/repo.git\",\n            entrypoint=\"flows.py:my_flow\",\n        ).to_deployment(\n            name=\"example-deploy-remote-flow\",\n        ),\n        work_pool_name=\"my-work-pool\",\n        image=\"my-registry/my-image:dev\",\n    )\n
    Source code in prefect/deployments/runner.py
    @sync_compatible\nasync def deploy(\n    *deployments: RunnerDeployment,\n    work_pool_name: Optional[str] = None,\n    image: Optional[Union[str, DeploymentImage]] = None,\n    build: bool = True,\n    push: bool = True,\n    print_next_steps_message: bool = True,\n) -> List[UUID]:\n    \"\"\"\n    Deploy the provided list of deployments to dynamic infrastructure via a\n    work pool.\n\n    By default, calling this function will build a Docker image for the deployments, push it to a\n    registry, and create each deployment via the Prefect API that will run the corresponding\n    flow on the given schedule.\n\n    If you want to use an existing image, you can pass `build=False` to skip building and pushing\n    an image.\n\n    Args:\n        *deployments: A list of deployments to deploy.\n        work_pool_name: The name of the work pool to use for these deployments. Defaults to\n            the value of `PREFECT_DEFAULT_WORK_POOL_NAME`.\n        image: The name of the Docker image to build, including the registry and\n            repository. Pass a DeploymentImage instance to customize the Dockerfile used\n            and build arguments.\n        build: Whether or not to build a new image for the flow. If False, the provided\n            image will be used as-is and pulled at runtime.\n        push: Whether or not to skip pushing the built image to a registry.\n        print_next_steps_message: Whether or not to print a message with next steps\n            after deploying the deployments.\n\n    Returns:\n        A list of deployment IDs for the created/updated deployments.\n\n    Examples:\n        Deploy a group of flows to a work pool:\n\n        ```python\n        from prefect import deploy, flow\n\n        @flow(log_prints=True)\n        def local_flow():\n            print(\"I'm a locally defined flow!\")\n\n        if __name__ == \"__main__\":\n            deploy(\n                local_flow.to_deployment(name=\"example-deploy-local-flow\"),\n                flow.from_source(\n                    source=\"https://github.com/org/repo.git\",\n                    entrypoint=\"flows.py:my_flow\",\n                ).to_deployment(\n                    name=\"example-deploy-remote-flow\",\n                ),\n                work_pool_name=\"my-work-pool\",\n                image=\"my-registry/my-image:dev\",\n            )\n        ```\n    \"\"\"\n    work_pool_name = work_pool_name or PREFECT_DEFAULT_WORK_POOL_NAME.value()\n\n    if not image and not all(d.storage for d in deployments):\n        raise ValueError(\n            \"Either an image or remote storage location must be provided when deploying\"\n            \" a deployment.\"\n        )\n\n    if not work_pool_name:\n        raise ValueError(\n            \"A work pool name must be provided when deploying a deployment. Either\"\n            \" provide a work pool name when calling `deploy` or set\"\n            \" `PREFECT_DEFAULT_WORK_POOL_NAME` in your profile.\"\n        )\n\n    if image and isinstance(image, str):\n        image_name, image_tag = parse_image_tag(image)\n        image = DeploymentImage(name=image_name, tag=image_tag)\n\n    try:\n        async with get_client() as client:\n            work_pool = await client.read_work_pool(work_pool_name)\n    except ObjectNotFound as exc:\n        raise ValueError(\n            f\"Could not find work pool {work_pool_name!r}. Please create it before\"\n            \" deploying this flow.\"\n        ) from exc\n\n    is_docker_based_work_pool = get_from_dict(\n        work_pool.base_job_template, \"variables.properties.image\", False\n    )\n    is_block_based_work_pool = get_from_dict(\n        work_pool.base_job_template, \"variables.properties.block\", False\n    )\n    # carve out an exception for block based work pools that only have a block in their base job template\n    if not is_docker_based_work_pool and not is_block_based_work_pool:\n        raise ValueError(\n            f\"Work pool {work_pool_name!r} does not support custom Docker images. \"\n            \"Please use a work pool with an `image` variable in its base job template.\"\n        )\n\n    is_managed_pool = work_pool.is_managed_pool\n    if is_managed_pool:\n        build = False\n        push = False\n\n    console = Console()\n    if image and build:\n        with Progress(\n            SpinnerColumn(),\n            TextColumn(f\"Building image {image.reference}...\"),\n            transient=True,\n            console=console,\n        ) as progress:\n            docker_build_task = progress.add_task(\"docker_build\", total=1)\n            image.build()\n\n            progress.update(docker_build_task, completed=1)\n            console.print(\n                f\"Successfully built image {image.reference!r}\", style=\"green\"\n            )\n\n    if image and build and push:\n        with Progress(\n            SpinnerColumn(),\n            TextColumn(\"Pushing image...\"),\n            transient=True,\n            console=console,\n        ) as progress:\n            docker_push_task = progress.add_task(\"docker_push\", total=1)\n\n            image.push()\n\n            progress.update(docker_push_task, completed=1)\n\n        console.print(f\"Successfully pushed image {image.reference!r}\", style=\"green\")\n\n    deployment_exceptions = []\n    deployment_ids = []\n    image_ref = image.reference if image else None\n    for deployment in track(\n        deployments,\n        description=\"Creating/updating deployments...\",\n        console=console,\n        transient=True,\n    ):\n        try:\n            deployment_ids.append(\n                await deployment.apply(image=image_ref, work_pool_name=work_pool_name)\n            )\n        except Exception as exc:\n            if len(deployments) == 1:\n                raise\n            deployment_exceptions.append({\"deployment\": deployment, \"exc\": exc})\n\n    if deployment_exceptions:\n        console.print(\n            \"Encountered errors while creating/updating deployments:\\n\",\n            style=\"orange_red1\",\n        )\n    else:\n        console.print(\"Successfully created/updated all deployments!\\n\", style=\"green\")\n\n    complete_failure = len(deployment_exceptions) == len(deployments)\n\n    table = Table(\n        title=\"Deployments\",\n        show_lines=True,\n    )\n\n    table.add_column(header=\"Name\", style=\"blue\", no_wrap=True)\n    table.add_column(header=\"Status\", style=\"blue\", no_wrap=True)\n    table.add_column(header=\"Details\", style=\"blue\")\n\n    for deployment in deployments:\n        errored_deployment = next(\n            (d for d in deployment_exceptions if d[\"deployment\"] == deployment),\n            None,\n        )\n        if errored_deployment:\n            table.add_row(\n                f\"{deployment.flow_name}/{deployment.name}\",\n                \"failed\",\n                str(errored_deployment[\"exc\"]),\n                style=\"red\",\n            )\n        else:\n            table.add_row(f\"{deployment.flow_name}/{deployment.name}\", \"applied\")\n    console.print(table)\n\n    if print_next_steps_message and not complete_failure:\n        if not work_pool.is_push_pool and not work_pool.is_managed_pool:\n            console.print(\n                \"\\nTo execute flow runs from these deployments, start a worker in a\"\n                \" separate terminal that pulls work from the\"\n                f\" {work_pool_name!r} work pool:\"\n            )\n            console.print(\n                f\"\\n\\t$ prefect worker start --pool {work_pool_name!r}\",\n                style=\"blue\",\n            )\n        console.print(\n            \"\\nTo trigger any of these deployments, use the\"\n            \" following command:\\n[blue]\\n\\t$ prefect deployment run\"\n            \" [DEPLOYMENT_NAME]\\n[/]\"\n        )\n\n        if PREFECT_UI_URL:\n            console.print(\n                \"\\nYou can also trigger your deployments via the Prefect UI:\"\n                f\" [blue]{PREFECT_UI_URL.value()}/deployments[/]\\n\"\n            )\n\n    return deployment_ids\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/steps/core/","title":"steps.core","text":"","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/steps/core/#prefect.deployments.steps.core","title":"prefect.deployments.steps.core","text":"

    Core primitives for running Prefect project steps.

    Project steps are YAML representations of Python functions along with their inputs.

    Whenever a step is run, the following actions are taken:

    • The step's inputs and block / variable references are resolved (see the projects concepts documentation for more details)
    • The step's function is imported; if it cannot be found, the requires keyword is used to install the necessary packages
    • The step's function is called with the resolved inputs
    • The step's output is returned and used to resolve inputs for subsequent steps
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/steps/core/#prefect.deployments.steps.core.StepExecutionError","title":"StepExecutionError","text":"

    Bases: Exception

    Raised when a step fails to execute.

    Source code in prefect/deployments/steps/core.py
    class StepExecutionError(Exception):\n    \"\"\"\n    Raised when a step fails to execute.\n    \"\"\"\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/steps/core/#prefect.deployments.steps.core.run_step","title":"run_step async","text":"

    Runs a step, returns the step's output.

    Steps are assumed to be in the format {\"importable.func.name\": {\"kwarg1\": \"value1\", ...}}.

    The 'id and 'requires' keywords are reserved for specific purposes and will be removed from the inputs before passing to the step function:

    This keyword is used to specify packages that should be installed before running the step.

    Source code in prefect/deployments/steps/core.py
    async def run_step(step: Dict, upstream_outputs: Optional[Dict] = None) -> Dict:\n    \"\"\"\n    Runs a step, returns the step's output.\n\n    Steps are assumed to be in the format `{\"importable.func.name\": {\"kwarg1\": \"value1\", ...}}`.\n\n    The 'id and 'requires' keywords are reserved for specific purposes and will be removed from the\n    inputs before passing to the step function:\n\n    This keyword is used to specify packages that should be installed before running the step.\n    \"\"\"\n    fqn, inputs = _get_step_fully_qualified_name_and_inputs(step)\n    upstream_outputs = upstream_outputs or {}\n\n    if len(step.keys()) > 1:\n        raise ValueError(\n            f\"Step has unexpected additional keys: {', '.join(list(step.keys())[1:])}\"\n        )\n\n    keywords = {\n        keyword: inputs.pop(keyword)\n        for keyword in RESERVED_KEYWORDS\n        if keyword in inputs\n    }\n\n    inputs = apply_values(inputs, upstream_outputs)\n    inputs = await resolve_block_document_references(inputs)\n    inputs = await resolve_variables(inputs)\n    inputs = apply_values(inputs, os.environ)\n    step_func = _get_function_for_step(fqn, requires=keywords.get(\"requires\"))\n    result = await from_async.call_soon_in_new_thread(\n        Call.new(step_func, **inputs)\n    ).aresult()\n    return result\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/steps/pull/","title":"steps.pull","text":"","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull","title":"prefect.deployments.steps.pull","text":"

    Core set of steps for specifying a Prefect project pull step.

    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.git_clone","title":"git_clone async","text":"

    Clones a git repository into the current working directory.

    Parameters:

    Name Type Description Default repository str

    the URL of the repository to clone

    required branch Optional[str]

    the branch to clone; if not provided, the default branch will be used

    None include_submodules bool

    whether to include git submodules when cloning the repository

    False access_token Optional[str]

    an access token to use for cloning the repository; if not provided the repository will be cloned using the default git credentials

    None credentials Optional[Block]

    a GitHubCredentials, GitLabCredentials, or BitBucketCredentials block can be used to specify the credentials to use for cloning the repository.

    None

    Returns:

    Name Type Description dict dict

    a dictionary containing a directory key of the new directory that was created

    Raises:

    Type Description CalledProcessError

    if the git clone command fails for any reason

    Examples:

    Clone a public repository:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/PrefectHQ/prefect.git\n

    Clone a branch of a public repository:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/PrefectHQ/prefect.git\n        branch: my-branch\n

    Clone a private repository using a GitHubCredentials block:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/org/repo.git\n        credentials: \"{{ prefect.blocks.github-credentials.my-github-credentials-block }}\"\n

    Clone a private repository using an access token:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/org/repo.git\n        access_token: \"{{ prefect.blocks.secret.github-access-token }}\" # Requires creation of a Secret block\n
    Note that you will need to create a Secret block to store the value of your git credentials. You can also store a username/password combo or token prefix (e.g. x-token-auth) in your secret block. Refer to your git providers documentation for the correct authentication schema.

    Clone a repository with submodules:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/org/repo.git\n        include_submodules: true\n

    Clone a repository with an SSH key (note that the SSH key must be added to the worker before executing flows):

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: git@github.com:org/repo.git\n

    Source code in prefect/deployments/steps/pull.py
    @sync_compatible\nasync def git_clone(\n    repository: str,\n    branch: Optional[str] = None,\n    include_submodules: bool = False,\n    access_token: Optional[str] = None,\n    credentials: Optional[Block] = None,\n) -> dict:\n    \"\"\"\n    Clones a git repository into the current working directory.\n\n    Args:\n        repository: the URL of the repository to clone\n        branch: the branch to clone; if not provided, the default branch will be used\n        include_submodules (bool): whether to include git submodules when cloning the repository\n        access_token: an access token to use for cloning the repository; if not provided\n            the repository will be cloned using the default git credentials\n        credentials: a GitHubCredentials, GitLabCredentials, or BitBucketCredentials block can be used to specify the\n            credentials to use for cloning the repository.\n\n    Returns:\n        dict: a dictionary containing a `directory` key of the new directory that was created\n\n    Raises:\n        subprocess.CalledProcessError: if the git clone command fails for any reason\n\n    Examples:\n        Clone a public repository:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/PrefectHQ/prefect.git\n        ```\n\n        Clone a branch of a public repository:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/PrefectHQ/prefect.git\n                branch: my-branch\n        ```\n\n        Clone a private repository using a GitHubCredentials block:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/org/repo.git\n                credentials: \"{{ prefect.blocks.github-credentials.my-github-credentials-block }}\"\n        ```\n\n        Clone a private repository using an access token:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/org/repo.git\n                access_token: \"{{ prefect.blocks.secret.github-access-token }}\" # Requires creation of a Secret block\n        ```\n        Note that you will need to [create a Secret block](/concepts/blocks/#using-existing-block-types) to store the\n        value of your git credentials. You can also store a username/password combo or token prefix (e.g. `x-token-auth`)\n        in your secret block. Refer to your git providers documentation for the correct authentication schema.\n\n        Clone a repository with submodules:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/org/repo.git\n                include_submodules: true\n        ```\n\n        Clone a repository with an SSH key (note that the SSH key must be added to the worker\n        before executing flows):\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: git@github.com:org/repo.git\n        ```\n    \"\"\"\n    if access_token and credentials:\n        raise ValueError(\n            \"Please provide either an access token or credentials but not both.\"\n        )\n\n    credentials = {\"access_token\": access_token} if access_token else credentials\n\n    storage = GitRepository(\n        url=repository,\n        credentials=credentials,\n        branch=branch,\n        include_submodules=include_submodules,\n    )\n\n    await storage.pull_code()\n\n    directory = str(storage.destination.relative_to(Path.cwd()))\n    deployment_logger.info(f\"Cloned repository {repository!r} into {directory!r}\")\n    return {\"directory\": directory}\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.git_clone_project","title":"git_clone_project async","text":"

    Deprecated. Use git_clone instead.

    Source code in prefect/deployments/steps/pull.py
    @deprecated_callable(start_date=\"Jun 2023\", help=\"Use 'git clone' instead.\")\n@sync_compatible\nasync def git_clone_project(\n    repository: str,\n    branch: Optional[str] = None,\n    include_submodules: bool = False,\n    access_token: Optional[str] = None,\n) -> dict:\n    \"\"\"Deprecated. Use `git_clone` instead.\"\"\"\n    return await git_clone(\n        repository=repository,\n        branch=branch,\n        include_submodules=include_submodules,\n        access_token=access_token,\n    )\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.pull_from_remote_storage","title":"pull_from_remote_storage async","text":"

    Pulls code from a remote storage location into the current working directory.

    Works with protocols supported by fsspec.

    Parameters:

    Name Type Description Default url str

    the URL of the remote storage location. Should be a valid fsspec URL. Some protocols may require an additional fsspec dependency to be installed. Refer to the fsspec docs for more details.

    required **settings Any

    any additional settings to pass the fsspec filesystem class.

    {}

    Returns:

    Name Type Description dict

    a dictionary containing a directory key of the new directory that was created

    Examples:

    Pull code from a remote storage location:

    pull:\n    - prefect.deployments.steps.pull_from_remote_storage:\n        url: s3://my-bucket/my-folder\n

    Pull code from a remote storage location with additional settings:

    pull:\n    - prefect.deployments.steps.pull_from_remote_storage:\n        url: s3://my-bucket/my-folder\n        key: {{ prefect.blocks.secret.my-aws-access-key }}}\n        secret: {{ prefect.blocks.secret.my-aws-secret-key }}}\n

    Source code in prefect/deployments/steps/pull.py
    async def pull_from_remote_storage(url: str, **settings: Any):\n    \"\"\"\n    Pulls code from a remote storage location into the current working directory.\n\n    Works with protocols supported by `fsspec`.\n\n    Args:\n        url (str): the URL of the remote storage location. Should be a valid `fsspec` URL.\n            Some protocols may require an additional `fsspec` dependency to be installed.\n            Refer to the [`fsspec` docs](https://filesystem-spec.readthedocs.io/en/latest/api.html#other-known-implementations)\n            for more details.\n        **settings (Any): any additional settings to pass the `fsspec` filesystem class.\n\n    Returns:\n        dict: a dictionary containing a `directory` key of the new directory that was created\n\n    Examples:\n        Pull code from a remote storage location:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.pull_from_remote_storage:\n                url: s3://my-bucket/my-folder\n        ```\n\n        Pull code from a remote storage location with additional settings:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.pull_from_remote_storage:\n                url: s3://my-bucket/my-folder\n                key: {{ prefect.blocks.secret.my-aws-access-key }}}\n                secret: {{ prefect.blocks.secret.my-aws-secret-key }}}\n        ```\n    \"\"\"\n    storage = RemoteStorage(url, **settings)\n\n    await storage.pull_code()\n\n    directory = str(storage.destination.relative_to(Path.cwd()))\n    deployment_logger.info(f\"Pulled code from {url!r} into {directory!r}\")\n    return {\"directory\": directory}\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.pull_with_block","title":"pull_with_block async","text":"

    Pulls code using a block.

    Parameters:

    Name Type Description Default block_document_name str

    The name of the block document to use

    required block_type_slug str

    The slug of the type of block to use

    required Source code in prefect/deployments/steps/pull.py
    async def pull_with_block(block_document_name: str, block_type_slug: str):\n    \"\"\"\n    Pulls code using a block.\n\n    Args:\n        block_document_name: The name of the block document to use\n        block_type_slug: The slug of the type of block to use\n    \"\"\"\n    full_slug = f\"{block_type_slug}/{block_document_name}\"\n    try:\n        block = await Block.load(full_slug)\n    except Exception:\n        deployment_logger.exception(\"Unable to load block '%s'\", full_slug)\n        raise\n\n    try:\n        storage = BlockStorageAdapter(block)\n    except Exception:\n        deployment_logger.exception(\n            \"Unable to create storage adapter for block '%s'\", full_slug\n        )\n        raise\n\n    await storage.pull_code()\n\n    directory = str(storage.destination.relative_to(Path.cwd()))\n    deployment_logger.info(\n        \"Pulled code using block '%s' into '%s'\", full_slug, directory\n    )\n    return {\"directory\": directory}\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.set_working_directory","title":"set_working_directory","text":"

    Sets the working directory; works with both absolute and relative paths.

    Parameters:

    Name Type Description Default directory str

    the directory to set as the working directory

    required

    Returns:

    Name Type Description dict dict

    a dictionary containing a directory key of the directory that was set

    Source code in prefect/deployments/steps/pull.py
    def set_working_directory(directory: str) -> dict:\n    \"\"\"\n    Sets the working directory; works with both absolute and relative paths.\n\n    Args:\n        directory (str): the directory to set as the working directory\n\n    Returns:\n        dict: a dictionary containing a `directory` key of the\n            directory that was set\n    \"\"\"\n    os.chdir(directory)\n    return dict(directory=directory)\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/utility/","title":"steps.utility","text":"","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/deployments/steps/utility/#prefect.deployments.steps.utility","title":"prefect.deployments.steps.utility","text":"

    Utility project steps that are useful for managing a project's deployment lifecycle.

    Steps within this module can be used within a build, push, or pull deployment action.

    Example

    Use the run_shell_script setp to retrieve the short Git commit hash of the current repository and use it as a Docker image tag:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        id: get-commit-hash\n        script: git rev-parse --short HEAD\n        stream_output: false\n    - prefect_docker.deployments.steps.build_docker_image:\n        requires: prefect-docker\n        image_name: my-image\n        image_tag: \"{{ get-commit-hash.stdout }}\"\n        dockerfile: auto\n

    ","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/deployments/steps/utility/#prefect.deployments.steps.utility.RunShellScriptResult","title":"RunShellScriptResult","text":"

    Bases: TypedDict

    The result of a run_shell_script step.

    Attributes:

    Name Type Description stdout str

    The captured standard output of the script.

    stderr str

    The captured standard error of the script.

    Source code in prefect/deployments/steps/utility.py
    class RunShellScriptResult(TypedDict):\n    \"\"\"\n    The result of a `run_shell_script` step.\n\n    Attributes:\n        stdout: The captured standard output of the script.\n        stderr: The captured standard error of the script.\n    \"\"\"\n\n    stdout: str\n    stderr: str\n
    ","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/deployments/steps/utility/#prefect.deployments.steps.utility.pip_install_requirements","title":"pip_install_requirements async","text":"

    Installs dependencies from a requirements.txt file.

    Parameters:

    Name Type Description Default requirements_file str

    The requirements.txt to use for installation.

    'requirements.txt' directory Optional[str]

    The directory the requirements.txt file is in. Defaults to the current working directory.

    None stream_output bool

    Whether to stream the output from pip install should be streamed to the console

    True

    Returns:

    Type Description

    A dictionary with the keys stdout and stderr containing the output the pip install command

    Raises:

    Type Description CalledProcessError

    if the pip install command fails for any reason

    Example
    pull:\n    - prefect.deployments.steps.git_clone:\n        id: clone-step\n        repository: https://github.com/org/repo.git\n    - prefect.deployments.steps.pip_install_requirements:\n        directory: {{ clone-step.directory }}\n        requirements_file: requirements.txt\n        stream_output: False\n
    Source code in prefect/deployments/steps/utility.py
    async def pip_install_requirements(\n    directory: Optional[str] = None,\n    requirements_file: str = \"requirements.txt\",\n    stream_output: bool = True,\n):\n    \"\"\"\n    Installs dependencies from a requirements.txt file.\n\n    Args:\n        requirements_file: The requirements.txt to use for installation.\n        directory: The directory the requirements.txt file is in. Defaults to\n            the current working directory.\n        stream_output: Whether to stream the output from pip install should be\n            streamed to the console\n\n    Returns:\n        A dictionary with the keys `stdout` and `stderr` containing the output\n            the `pip install` command\n\n    Raises:\n        subprocess.CalledProcessError: if the pip install command fails for any reason\n\n    Example:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                id: clone-step\n                repository: https://github.com/org/repo.git\n            - prefect.deployments.steps.pip_install_requirements:\n                directory: {{ clone-step.directory }}\n                requirements_file: requirements.txt\n                stream_output: False\n        ```\n    \"\"\"\n    stdout_sink = io.StringIO()\n    stderr_sink = io.StringIO()\n\n    async with open_process(\n        [get_sys_executable(), \"-m\", \"pip\", \"install\", \"-r\", requirements_file],\n        stdout=subprocess.PIPE,\n        stderr=subprocess.PIPE,\n        cwd=directory,\n    ) as process:\n        await _stream_capture_process_output(\n            process,\n            stdout_sink=stdout_sink,\n            stderr_sink=stderr_sink,\n            stream_output=stream_output,\n        )\n        await process.wait()\n\n        if process.returncode != 0:\n            raise RuntimeError(\n                f\"pip_install_requirements failed with error code {process.returncode}:\"\n                f\" {stderr_sink.getvalue()}\"\n            )\n\n    return {\n        \"stdout\": stdout_sink.getvalue().strip(),\n        \"stderr\": stderr_sink.getvalue().strip(),\n    }\n
    ","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/deployments/steps/utility/#prefect.deployments.steps.utility.run_shell_script","title":"run_shell_script async","text":"

    Runs one or more shell commands in a subprocess. Returns the standard output and standard error of the script.

    Parameters:

    Name Type Description Default script str

    The script to run

    required directory Optional[str]

    The directory to run the script in. Defaults to the current working directory.

    None env Optional[Dict[str, str]]

    A dictionary of environment variables to set for the script

    None stream_output bool

    Whether to stream the output of the script to stdout/stderr

    True expand_env_vars bool

    Whether to expand environment variables in the script before running it

    False

    Returns:

    Type Description RunShellScriptResult

    A dictionary with the keys stdout and stderr containing the output of the script

    Examples:

    Retrieve the short Git commit hash of the current repository to use as a Docker image tag:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        id: get-commit-hash\n        script: git rev-parse --short HEAD\n        stream_output: false\n    - prefect_docker.deployments.steps.build_docker_image:\n        requires: prefect-docker\n        image_name: my-image\n        image_tag: \"{{ get-commit-hash.stdout }}\"\n        dockerfile: auto\n

    Run a multi-line shell script:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        script: |\n            echo \"Hello\"\n            echo \"World\"\n

    Run a shell script with environment variables:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        script: echo \"Hello $NAME\"\n        env:\n            NAME: World\n

    Run a shell script with environment variables expanded from the current environment:

    pull:\n    - prefect.deployments.steps.run_shell_script:\n        script: |\n            echo \"User: $USER\"\n            echo \"Home Directory: $HOME\"\n        stream_output: true\n        expand_env_vars: true\n

    Run a shell script in a specific directory:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        script: echo \"Hello\"\n        directory: /path/to/directory\n

    Run a script stored in a file:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        script: \"bash path/to/script.sh\"\n

    Source code in prefect/deployments/steps/utility.py
    async def run_shell_script(\n    script: str,\n    directory: Optional[str] = None,\n    env: Optional[Dict[str, str]] = None,\n    stream_output: bool = True,\n    expand_env_vars: bool = False,\n) -> RunShellScriptResult:\n    \"\"\"\n    Runs one or more shell commands in a subprocess. Returns the standard\n    output and standard error of the script.\n\n    Args:\n        script: The script to run\n        directory: The directory to run the script in. Defaults to the current\n            working directory.\n        env: A dictionary of environment variables to set for the script\n        stream_output: Whether to stream the output of the script to\n            stdout/stderr\n        expand_env_vars: Whether to expand environment variables in the script\n            before running it\n\n    Returns:\n        A dictionary with the keys `stdout` and `stderr` containing the output\n            of the script\n\n    Examples:\n        Retrieve the short Git commit hash of the current repository to use as\n            a Docker image tag:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                id: get-commit-hash\n                script: git rev-parse --short HEAD\n                stream_output: false\n            - prefect_docker.deployments.steps.build_docker_image:\n                requires: prefect-docker\n                image_name: my-image\n                image_tag: \"{{ get-commit-hash.stdout }}\"\n                dockerfile: auto\n        ```\n\n        Run a multi-line shell script:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                script: |\n                    echo \"Hello\"\n                    echo \"World\"\n        ```\n\n        Run a shell script with environment variables:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                script: echo \"Hello $NAME\"\n                env:\n                    NAME: World\n        ```\n\n        Run a shell script with environment variables expanded\n            from the current environment:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.run_shell_script:\n                script: |\n                    echo \"User: $USER\"\n                    echo \"Home Directory: $HOME\"\n                stream_output: true\n                expand_env_vars: true\n        ```\n\n        Run a shell script in a specific directory:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                script: echo \"Hello\"\n                directory: /path/to/directory\n        ```\n\n        Run a script stored in a file:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                script: \"bash path/to/script.sh\"\n        ```\n    \"\"\"\n    current_env = os.environ.copy()\n    current_env.update(env or {})\n\n    commands = script.splitlines()\n    stdout_sink = io.StringIO()\n    stderr_sink = io.StringIO()\n\n    for command in commands:\n        if expand_env_vars:\n            # Expand environment variables in command and provided environment\n            command = string.Template(command).safe_substitute(current_env)\n        split_command = shlex.split(command, posix=sys.platform != \"win32\")\n        if not split_command:\n            continue\n        async with open_process(\n            split_command,\n            stdout=subprocess.PIPE,\n            stderr=subprocess.PIPE,\n            cwd=directory,\n            env=current_env,\n        ) as process:\n            await _stream_capture_process_output(\n                process,\n                stdout_sink=stdout_sink,\n                stderr_sink=stderr_sink,\n                stream_output=stream_output,\n            )\n\n            await process.wait()\n\n    return {\n        \"stdout\": stdout_sink.getvalue().strip(),\n        \"stderr\": stderr_sink.getvalue().strip(),\n    }\n
    ","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/input/actions/","title":"actions","text":"","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/actions/#prefect.input.actions","title":"prefect.input.actions","text":"","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/actions/#prefect.input.actions.create_flow_run_input","title":"create_flow_run_input async","text":"

    Create a new flow run input. The given value will be serialized to JSON and stored as a flow run input value.

    Parameters:

    Name Type Description Default - key (str

    the flow run input key

    required - value (Any

    the flow run input value

    required - flow_run_id (UUID

    the, optional, flow run ID. If not given will default to pulling the flow run ID from the current context.

    required Source code in prefect/input/actions.py
    @sync_compatible\n@inject_client\nasync def create_flow_run_input(\n    key: str,\n    value: Any,\n    flow_run_id: Optional[UUID] = None,\n    sender: Optional[str] = None,\n    client: \"PrefectClient\" = None,\n):\n    \"\"\"\n    Create a new flow run input. The given `value` will be serialized to JSON\n    and stored as a flow run input value.\n\n    Args:\n        - key (str): the flow run input key\n        - value (Any): the flow run input value\n        - flow_run_id (UUID): the, optional, flow run ID. If not given will\n          default to pulling the flow run ID from the current context.\n    \"\"\"\n    flow_run_id = ensure_flow_run_id(flow_run_id)\n\n    await client.create_flow_run_input(\n        flow_run_id=flow_run_id,\n        key=key,\n        sender=sender,\n        value=orjson.dumps(value).decode(),\n    )\n
    ","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/actions/#prefect.input.actions.delete_flow_run_input","title":"delete_flow_run_input async","text":"

    Delete a flow run input.

    Parameters:

    Name Type Description Default - flow_run_id (UUID

    the flow run ID

    required - key (str

    the flow run input key

    required Source code in prefect/input/actions.py
    @sync_compatible\n@inject_client\nasync def delete_flow_run_input(\n    key: str, flow_run_id: Optional[UUID] = None, client: \"PrefectClient\" = None\n):\n    \"\"\"Delete a flow run input.\n\n    Args:\n        - flow_run_id (UUID): the flow run ID\n        - key (str): the flow run input key\n    \"\"\"\n\n    flow_run_id = ensure_flow_run_id(flow_run_id)\n\n    await client.delete_flow_run_input(flow_run_id=flow_run_id, key=key)\n
    ","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/actions/#prefect.input.actions.read_flow_run_input","title":"read_flow_run_input async","text":"

    Read a flow run input.

    Parameters:

    Name Type Description Default - key (str

    the flow run input key

    required - flow_run_id (UUID

    the flow run ID

    required Source code in prefect/input/actions.py
    @sync_compatible\n@inject_client\nasync def read_flow_run_input(\n    key: str, flow_run_id: Optional[UUID] = None, client: \"PrefectClient\" = None\n) -> Any:\n    \"\"\"Read a flow run input.\n\n    Args:\n        - key (str): the flow run input key\n        - flow_run_id (UUID): the flow run ID\n    \"\"\"\n    flow_run_id = ensure_flow_run_id(flow_run_id)\n\n    try:\n        value = await client.read_flow_run_input(flow_run_id=flow_run_id, key=key)\n    except PrefectHTTPStatusError as exc:\n        if exc.response.status_code == 404:\n            return None\n        raise\n    else:\n        return orjson.loads(value)\n
    ","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/run_input/","title":"run_input","text":"","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input","title":"prefect.input.run_input","text":"","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput","title":"RunInput","text":"

    Bases: BaseModel

    Source code in prefect/input/run_input.py
    class RunInput(pydantic.BaseModel):\n    class Config:\n        extra = \"forbid\"\n\n    _metadata: RunInputMetadata = pydantic.PrivateAttr()\n\n    @property\n    def metadata(self) -> RunInputMetadata:\n        return self._metadata\n\n    @classmethod\n    def keyset_from_type(cls) -> Keyset:\n        return keyset_from_base_key(cls.__name__.lower())\n\n    @classmethod\n    @sync_compatible\n    async def save(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):\n        \"\"\"\n        Save the run input response to the given key.\n\n        Args:\n            - keyset (Keyset): the keyset to save the input for\n            - flow_run_id (UUID, optional): the flow run ID to save the input for\n        \"\"\"\n\n        if HAS_PYDANTIC_V2:\n            schema = create_v2_schema(cls.__name__, model_base=cls)\n        else:\n            schema = cls.schema(by_alias=True)\n\n        await create_flow_run_input(\n            key=keyset[\"schema\"], value=schema, flow_run_id=flow_run_id\n        )\n\n    @classmethod\n    @sync_compatible\n    async def load(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):\n        \"\"\"\n        Load the run input response from the given key.\n\n        Args:\n            - keyset (Keyset): the keyset to load the input for\n            - flow_run_id (UUID, optional): the flow run ID to load the input for\n        \"\"\"\n        flow_run_id = ensure_flow_run_id(flow_run_id)\n        value = await read_flow_run_input(keyset[\"response\"], flow_run_id=flow_run_id)\n        instance = cls(**value)\n        instance._metadata = RunInputMetadata(\n            key=keyset[\"response\"], sender=None, receiver=flow_run_id\n        )\n        return instance\n\n    @classmethod\n    def load_from_flow_run_input(cls, flow_run_input: \"FlowRunInput\"):\n        \"\"\"\n        Load the run input from a FlowRunInput object.\n\n        Args:\n            - flow_run_input (FlowRunInput): the flow run input to load the input for\n        \"\"\"\n        instance = cls(**flow_run_input.decoded_value)\n        instance._metadata = RunInputMetadata(\n            key=flow_run_input.key,\n            sender=flow_run_input.sender,\n            receiver=flow_run_input.flow_run_id,\n        )\n        return instance\n\n    @classmethod\n    def with_initial_data(cls: Type[T], **kwargs: Any) -> Type[T]:\n        \"\"\"\n        Create a new `RunInput` subclass with the given initial data as field\n        defaults.\n\n        Args:\n            - kwargs (Any): the initial data\n        \"\"\"\n        fields = {}\n        for key, value in kwargs.items():\n            fields[key] = (type(value), value)\n        return pydantic.create_model(cls.__name__, **fields, __base__=cls)\n\n    @sync_compatible\n    async def respond(\n        self,\n        run_input: \"RunInput\",\n        sender: Optional[str] = None,\n        key_prefix: Optional[str] = None,\n    ):\n        flow_run_id = None\n        if self.metadata.sender and self.metadata.sender.startswith(\"prefect.flow-run\"):\n            _, _, id = self.metadata.sender.rpartition(\".\")\n            flow_run_id = UUID(id)\n\n        if not flow_run_id:\n            raise RuntimeError(\n                \"Cannot respond to an input that was not sent by a flow run.\"\n            )\n\n        await _send_input(\n            flow_run_id=flow_run_id,\n            run_input=run_input,\n            sender=sender,\n            key_prefix=key_prefix,\n        )\n\n    @sync_compatible\n    async def send_to(\n        self,\n        flow_run_id: UUID,\n        sender: Optional[str] = None,\n        key_prefix: Optional[str] = None,\n    ):\n        await _send_input(\n            flow_run_id=flow_run_id,\n            run_input=self,\n            sender=sender,\n            key_prefix=key_prefix,\n        )\n\n    @classmethod\n    def receive(\n        cls,\n        timeout: Optional[float] = 3600,\n        poll_interval: float = 10,\n        raise_timeout_error: bool = False,\n        exclude_keys: Optional[Set[str]] = None,\n        key_prefix: Optional[str] = None,\n        flow_run_id: Optional[UUID] = None,\n    ):\n        if key_prefix is None:\n            key_prefix = f\"{cls.__name__.lower()}-auto\"\n\n        return GetInputHandler(\n            run_input_cls=cls,\n            key_prefix=key_prefix,\n            timeout=timeout,\n            poll_interval=poll_interval,\n            raise_timeout_error=raise_timeout_error,\n            exclude_keys=exclude_keys,\n            flow_run_id=flow_run_id,\n        )\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput.load","title":"load async classmethod","text":"

    Load the run input response from the given key.

    Parameters:

    Name Type Description Default - keyset (Keyset

    the keyset to load the input for

    required - flow_run_id (UUID

    the flow run ID to load the input for

    required Source code in prefect/input/run_input.py
    @classmethod\n@sync_compatible\nasync def load(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):\n    \"\"\"\n    Load the run input response from the given key.\n\n    Args:\n        - keyset (Keyset): the keyset to load the input for\n        - flow_run_id (UUID, optional): the flow run ID to load the input for\n    \"\"\"\n    flow_run_id = ensure_flow_run_id(flow_run_id)\n    value = await read_flow_run_input(keyset[\"response\"], flow_run_id=flow_run_id)\n    instance = cls(**value)\n    instance._metadata = RunInputMetadata(\n        key=keyset[\"response\"], sender=None, receiver=flow_run_id\n    )\n    return instance\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput.load_from_flow_run_input","title":"load_from_flow_run_input classmethod","text":"

    Load the run input from a FlowRunInput object.

    Parameters:

    Name Type Description Default - flow_run_input (FlowRunInput

    the flow run input to load the input for

    required Source code in prefect/input/run_input.py
    @classmethod\ndef load_from_flow_run_input(cls, flow_run_input: \"FlowRunInput\"):\n    \"\"\"\n    Load the run input from a FlowRunInput object.\n\n    Args:\n        - flow_run_input (FlowRunInput): the flow run input to load the input for\n    \"\"\"\n    instance = cls(**flow_run_input.decoded_value)\n    instance._metadata = RunInputMetadata(\n        key=flow_run_input.key,\n        sender=flow_run_input.sender,\n        receiver=flow_run_input.flow_run_id,\n    )\n    return instance\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput.save","title":"save async classmethod","text":"

    Save the run input response to the given key.

    Parameters:

    Name Type Description Default - keyset (Keyset

    the keyset to save the input for

    required - flow_run_id (UUID

    the flow run ID to save the input for

    required Source code in prefect/input/run_input.py
    @classmethod\n@sync_compatible\nasync def save(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):\n    \"\"\"\n    Save the run input response to the given key.\n\n    Args:\n        - keyset (Keyset): the keyset to save the input for\n        - flow_run_id (UUID, optional): the flow run ID to save the input for\n    \"\"\"\n\n    if HAS_PYDANTIC_V2:\n        schema = create_v2_schema(cls.__name__, model_base=cls)\n    else:\n        schema = cls.schema(by_alias=True)\n\n    await create_flow_run_input(\n        key=keyset[\"schema\"], value=schema, flow_run_id=flow_run_id\n    )\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput.with_initial_data","title":"with_initial_data classmethod","text":"

    Create a new RunInput subclass with the given initial data as field defaults.

    Parameters:

    Name Type Description Default - kwargs (Any

    the initial data

    required Source code in prefect/input/run_input.py
    @classmethod\ndef with_initial_data(cls: Type[T], **kwargs: Any) -> Type[T]:\n    \"\"\"\n    Create a new `RunInput` subclass with the given initial data as field\n    defaults.\n\n    Args:\n        - kwargs (Any): the initial data\n    \"\"\"\n    fields = {}\n    for key, value in kwargs.items():\n        fields[key] = (type(value), value)\n    return pydantic.create_model(cls.__name__, **fields, __base__=cls)\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.keyset_from_base_key","title":"keyset_from_base_key","text":"

    Get the keyset for the given base key.

    Parameters:

    Name Type Description Default - base_key (str

    the base key to get the keyset for

    required

    Returns:

    Type Description Keyset
    • Dict[str, str]: the keyset
    Source code in prefect/input/run_input.py
    def keyset_from_base_key(base_key: str) -> Keyset:\n    \"\"\"\n    Get the keyset for the given base key.\n\n    Args:\n        - base_key (str): the base key to get the keyset for\n\n    Returns:\n        - Dict[str, str]: the keyset\n    \"\"\"\n    return {\n        \"response\": f\"{base_key}-response\",\n        \"schema\": f\"{base_key}-schema\",\n    }\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.keyset_from_paused_state","title":"keyset_from_paused_state","text":"

    Get the keyset for the given Paused state.

    Parameters:

    Name Type Description Default - state (State

    the state to get the keyset for

    required Source code in prefect/input/run_input.py
    def keyset_from_paused_state(state: \"State\") -> Keyset:\n    \"\"\"\n    Get the keyset for the given Paused state.\n\n    Args:\n        - state (State): the state to get the keyset for\n    \"\"\"\n\n    if not state.is_paused():\n        raise RuntimeError(f\"{state.type.value!r} is unsupported.\")\n\n    base_key = f\"{state.name.lower()}-{str(state.state_details.pause_key)}\"\n    return keyset_from_base_key(base_key)\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/logging/configuration/","title":"configuration","text":"

    \"\"\"

    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/configuration/#prefect.logging.configuration","title":"prefect.logging.configuration","text":"","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/configuration/#prefect.logging.configuration.load_logging_config","title":"load_logging_config","text":"

    Loads logging configuration from a path allowing override from the environment

    Source code in prefect/logging/configuration.py
    def load_logging_config(path: Path) -> dict:\n    \"\"\"\n    Loads logging configuration from a path allowing override from the environment\n    \"\"\"\n    template = string.Template(path.read_text())\n    with warnings.catch_warnings():\n        warnings.filterwarnings(\"ignore\", category=DeprecationWarning)\n        config = yaml.safe_load(\n            # Substitute settings into the template in format $SETTING / ${SETTING}\n            template.substitute(\n                {\n                    setting.name: str(setting.value())\n                    for setting in SETTING_VARIABLES.values()\n                    if setting.value() is not None\n                }\n            )\n        )\n\n    # Load overrides from the environment\n    flat_config = dict_to_flatdict(config)\n\n    for key_tup, val in flat_config.items():\n        env_val = os.environ.get(\n            # Generate a valid environment variable with nesting indicated with '_'\n            to_envvar(\"PREFECT_LOGGING_\" + \"_\".join(key_tup)).upper()\n        )\n        if env_val:\n            val = env_val\n\n        # reassign the updated value\n        flat_config[key_tup] = val\n\n    return flatdict_to_dict(flat_config)\n
    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/configuration/#prefect.logging.configuration.setup_logging","title":"setup_logging","text":"

    Sets up logging.

    Returns the config used.

    Source code in prefect/logging/configuration.py
    def setup_logging(incremental: Optional[bool] = None) -> dict:\n    \"\"\"\n    Sets up logging.\n\n    Returns the config used.\n    \"\"\"\n    global PROCESS_LOGGING_CONFIG\n\n    # If the user has specified a logging path and it exists we will ignore the\n    # default entirely rather than dealing with complex merging\n    config = load_logging_config(\n        (\n            PREFECT_LOGGING_SETTINGS_PATH.value()\n            if PREFECT_LOGGING_SETTINGS_PATH.value().exists()\n            else DEFAULT_LOGGING_SETTINGS_PATH\n        )\n    )\n\n    incremental = (\n        incremental if incremental is not None else bool(PROCESS_LOGGING_CONFIG)\n    )\n\n    # Perform an incremental update if setup has already been run\n    config.setdefault(\"incremental\", incremental)\n\n    try:\n        logging.config.dictConfig(config)\n    except ValueError:\n        if incremental:\n            setup_logging(incremental=False)\n\n    # Copy configuration of the 'prefect.extra' logger to the extra loggers\n    extra_config = logging.getLogger(\"prefect.extra\")\n\n    for logger_name in PREFECT_LOGGING_EXTRA_LOGGERS.value():\n        logger = logging.getLogger(logger_name)\n        for handler in extra_config.handlers:\n            if not config[\"incremental\"]:\n                logger.addHandler(handler)\n            if logger.level == logging.NOTSET:\n                logger.setLevel(extra_config.level)\n            logger.propagate = extra_config.propagate\n\n    PROCESS_LOGGING_CONFIG = config\n\n    return config\n
    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/formatters/","title":"formatters","text":"

    \"\"\"

    ","tags":["Python API","logging","formatters"]},{"location":"api-ref/prefect/logging/formatters/#prefect.logging.formatters","title":"prefect.logging.formatters","text":"","tags":["Python API","logging","formatters"]},{"location":"api-ref/prefect/logging/formatters/#prefect.logging.formatters.JsonFormatter","title":"JsonFormatter","text":"

    Bases: Formatter

    Formats log records as a JSON string.

    The format may be specified as \"pretty\" to format the JSON with indents and newlines.

    Source code in prefect/logging/formatters.py
    class JsonFormatter(logging.Formatter):\n    \"\"\"\n    Formats log records as a JSON string.\n\n    The format may be specified as \"pretty\" to format the JSON with indents and\n    newlines.\n    \"\"\"\n\n    def __init__(self, fmt, dmft, style) -> None:  # noqa\n        super().__init__()\n\n        if fmt not in [\"pretty\", \"default\"]:\n            raise ValueError(\"Format must be either 'pretty' or 'default'.\")\n\n        self.serializer = JSONSerializer(\n            jsonlib=\"orjson\",\n            object_encoder=\"pydantic.json.pydantic_encoder\",\n            dumps_kwargs={\"option\": orjson.OPT_INDENT_2} if fmt == \"pretty\" else {},\n        )\n\n    def format(self, record: logging.LogRecord) -> str:\n        record_dict = record.__dict__.copy()\n\n        # GCP severity detection compatibility\n        record_dict.setdefault(\"severity\", record.levelname)\n\n        # replace any exception tuples returned by `sys.exc_info()`\n        # with a JSON-serializable `dict`.\n        if record.exc_info:\n            record_dict[\"exc_info\"] = format_exception_info(record.exc_info)\n\n        log_json_bytes = self.serializer.dumps(record_dict)\n\n        # JSONSerializer returns bytes; decode to string to conform to\n        # the `logging.Formatter.format` interface\n        return log_json_bytes.decode()\n
    ","tags":["Python API","logging","formatters"]},{"location":"api-ref/prefect/logging/formatters/#prefect.logging.formatters.PrefectFormatter","title":"PrefectFormatter","text":"

    Bases: Formatter

    Source code in prefect/logging/formatters.py
    class PrefectFormatter(logging.Formatter):\n    def __init__(\n        self,\n        format=None,\n        datefmt=None,\n        style=\"%\",\n        validate=True,\n        *,\n        defaults=None,\n        task_run_fmt: str = None,\n        flow_run_fmt: str = None\n    ) -> None:\n        \"\"\"\n        Implementation of the standard Python formatter with support for multiple\n        message formats.\n\n        \"\"\"\n        # See https://github.com/python/cpython/blob/c8c6113398ee9a7867fe9b08bc539cceb61e2aaa/Lib/logging/__init__.py#L546\n        # for implementation details\n\n        init_kwargs = {}\n        style_kwargs = {}\n\n        # defaults added in 3.10\n        if sys.version_info >= (3, 10):\n            init_kwargs[\"defaults\"] = defaults\n            style_kwargs[\"defaults\"] = defaults\n\n        # validate added in 3.8\n        if sys.version_info >= (3, 8):\n            init_kwargs[\"validate\"] = validate\n        else:\n            validate = False\n\n        super().__init__(format, datefmt, style, **init_kwargs)\n\n        self.flow_run_fmt = flow_run_fmt\n        self.task_run_fmt = task_run_fmt\n\n        # Retrieve the style class from the base class to avoid importing private\n        # `_STYLES` mapping\n        style_class = type(self._style)\n\n        self._flow_run_style = (\n            style_class(flow_run_fmt, **style_kwargs) if flow_run_fmt else self._style\n        )\n        self._task_run_style = (\n            style_class(task_run_fmt, **style_kwargs) if task_run_fmt else self._style\n        )\n        if validate:\n            self._flow_run_style.validate()\n            self._task_run_style.validate()\n\n    def formatMessage(self, record: logging.LogRecord):\n        if record.name == \"prefect.flow_runs\":\n            style = self._flow_run_style\n        elif record.name == \"prefect.task_runs\":\n            style = self._task_run_style\n        else:\n            style = self._style\n\n        return style.format(record)\n
    ","tags":["Python API","logging","formatters"]},{"location":"api-ref/prefect/logging/handlers/","title":"handlers","text":"

    \"\"\"

    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers","title":"prefect.logging.handlers","text":"","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler","title":"APILogHandler","text":"

    Bases: Handler

    A logging handler that sends logs to the Prefect API.

    Sends log records to the APILogWorker which manages sending batches of logs in the background.

    Source code in prefect/logging/handlers.py
    class APILogHandler(logging.Handler):\n    \"\"\"\n    A logging handler that sends logs to the Prefect API.\n\n    Sends log records to the `APILogWorker` which manages sending batches of logs in\n    the background.\n    \"\"\"\n\n    @classmethod\n    def flush(cls):\n        \"\"\"\n        Tell the `APILogWorker` to send any currently enqueued logs and block until\n        completion.\n\n        Use `aflush` from async contexts instead.\n        \"\"\"\n        loop = get_running_loop()\n        if loop:\n            if in_global_loop():  # Guard against internal misuse\n                raise RuntimeError(\n                    \"Cannot call `APILogWorker.flush` from the global event loop; it\"\n                    \" would block the event loop and cause a deadlock. Use\"\n                    \" `APILogWorker.aflush` instead.\"\n                )\n\n            # Not ideal, but this method is called by the stdlib and cannot return a\n            # coroutine so we just schedule the drain in a new thread and continue\n            from_sync.call_soon_in_new_thread(create_call(APILogWorker.drain_all))\n            return None\n        else:\n            # We set a timeout of 5s because we don't want to block forever if the worker\n            # is stuck. This can occur when the handler is being shutdown and the\n            # `logging._lock` is held but the worker is attempting to emit logs resulting\n            # in a deadlock.\n            return APILogWorker.drain_all(timeout=5)\n\n    @classmethod\n    def aflush(cls):\n        \"\"\"\n        Tell the `APILogWorker` to send any currently enqueued logs and block until\n        completion.\n\n        If called in a synchronous context, will only block up to 5s before returning.\n        \"\"\"\n\n        if not get_running_loop():\n            raise RuntimeError(\n                \"`aflush` cannot be used from a synchronous context; use `flush`\"\n                \" instead.\"\n            )\n\n        return APILogWorker.drain_all()\n\n    def emit(self, record: logging.LogRecord):\n        \"\"\"\n        Send a log to the `APILogWorker`\n        \"\"\"\n        try:\n            profile = prefect.context.get_settings_context()\n\n            if not PREFECT_LOGGING_TO_API_ENABLED.value_from(profile.settings):\n                return  # Respect the global settings toggle\n            if not getattr(record, \"send_to_api\", True):\n                return  # Do not send records that have opted out\n            if not getattr(record, \"send_to_orion\", True):\n                return  # Backwards compatibility\n\n            log = self.prepare(record)\n            APILogWorker.instance().send(log)\n\n        except Exception:\n            self.handleError(record)\n\n    def handleError(self, record: logging.LogRecord) -> None:\n        _, exc, _ = sys.exc_info()\n\n        if isinstance(exc, MissingContextError):\n            log_handling_when_missing_flow = (\n                PREFECT_LOGGING_TO_API_WHEN_MISSING_FLOW.value()\n            )\n            if log_handling_when_missing_flow == \"warn\":\n                # Warn when a logger is used outside of a run context, the stack level here\n                # gets us to the user logging call\n                warnings.warn(str(exc), stacklevel=8)\n                return\n            elif log_handling_when_missing_flow == \"ignore\":\n                return\n            else:\n                raise exc\n\n        # Display a longer traceback for other errors\n        return super().handleError(record)\n\n    def prepare(self, record: logging.LogRecord) -> Dict[str, Any]:\n        \"\"\"\n        Convert a `logging.LogRecord` to the API `LogCreate` schema and serialize.\n\n        This infers the linked flow or task run from the log record or the current\n        run context.\n\n        If a flow run id cannot be found, the log will be dropped.\n\n        Logs exceeding the maximum size will be dropped.\n        \"\"\"\n        flow_run_id = getattr(record, \"flow_run_id\", None)\n        task_run_id = getattr(record, \"task_run_id\", None)\n\n        if not flow_run_id:\n            try:\n                context = prefect.context.get_run_context()\n            except MissingContextError:\n                raise MissingContextError(\n                    f\"Logger {record.name!r} attempted to send logs to the API without\"\n                    \" a flow run id. The API log handler can only send logs within\"\n                    \" flow run contexts unless the flow run id is manually provided.\"\n                ) from None\n\n            if hasattr(context, \"flow_run\"):\n                flow_run_id = context.flow_run.id\n            elif hasattr(context, \"task_run\"):\n                flow_run_id = context.task_run.flow_run_id\n                task_run_id = task_run_id or context.task_run.id\n            else:\n                raise ValueError(\n                    \"Encountered malformed run context. Does not contain flow or task \"\n                    \"run information.\"\n                )\n\n        # Parsing to a `LogCreate` object here gives us nice parsing error messages\n        # from the standard lib `handleError` method if something goes wrong and\n        # prevents malformed logs from entering the queue\n        try:\n            is_uuid_like = isinstance(flow_run_id, uuid.UUID) or (\n                isinstance(flow_run_id, str) and uuid.UUID(flow_run_id)\n            )\n        except ValueError:\n            is_uuid_like = False\n\n        log = LogCreate(\n            flow_run_id=flow_run_id if is_uuid_like else None,\n            task_run_id=task_run_id,\n            name=record.name,\n            level=record.levelno,\n            timestamp=pendulum.from_timestamp(\n                getattr(record, \"created\", None) or time.time()\n            ),\n            message=self.format(record),\n        ).dict(json_compatible=True)\n\n        log_size = log[\"__payload_size__\"] = self._get_payload_size(log)\n        if log_size > PREFECT_LOGGING_TO_API_MAX_LOG_SIZE.value():\n            raise ValueError(\n                f\"Log of size {log_size} is greater than the max size of \"\n                f\"{PREFECT_LOGGING_TO_API_MAX_LOG_SIZE.value()}\"\n            )\n\n        return log\n\n    def _get_payload_size(self, log: Dict[str, Any]) -> int:\n        return len(json.dumps(log).encode())\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler.aflush","title":"aflush classmethod","text":"

    Tell the APILogWorker to send any currently enqueued logs and block until completion.

    If called in a synchronous context, will only block up to 5s before returning.

    Source code in prefect/logging/handlers.py
    @classmethod\ndef aflush(cls):\n    \"\"\"\n    Tell the `APILogWorker` to send any currently enqueued logs and block until\n    completion.\n\n    If called in a synchronous context, will only block up to 5s before returning.\n    \"\"\"\n\n    if not get_running_loop():\n        raise RuntimeError(\n            \"`aflush` cannot be used from a synchronous context; use `flush`\"\n            \" instead.\"\n        )\n\n    return APILogWorker.drain_all()\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler.emit","title":"emit","text":"

    Send a log to the APILogWorker

    Source code in prefect/logging/handlers.py
    def emit(self, record: logging.LogRecord):\n    \"\"\"\n    Send a log to the `APILogWorker`\n    \"\"\"\n    try:\n        profile = prefect.context.get_settings_context()\n\n        if not PREFECT_LOGGING_TO_API_ENABLED.value_from(profile.settings):\n            return  # Respect the global settings toggle\n        if not getattr(record, \"send_to_api\", True):\n            return  # Do not send records that have opted out\n        if not getattr(record, \"send_to_orion\", True):\n            return  # Backwards compatibility\n\n        log = self.prepare(record)\n        APILogWorker.instance().send(log)\n\n    except Exception:\n        self.handleError(record)\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler.flush","title":"flush classmethod","text":"

    Tell the APILogWorker to send any currently enqueued logs and block until completion.

    Use aflush from async contexts instead.

    Source code in prefect/logging/handlers.py
    @classmethod\ndef flush(cls):\n    \"\"\"\n    Tell the `APILogWorker` to send any currently enqueued logs and block until\n    completion.\n\n    Use `aflush` from async contexts instead.\n    \"\"\"\n    loop = get_running_loop()\n    if loop:\n        if in_global_loop():  # Guard against internal misuse\n            raise RuntimeError(\n                \"Cannot call `APILogWorker.flush` from the global event loop; it\"\n                \" would block the event loop and cause a deadlock. Use\"\n                \" `APILogWorker.aflush` instead.\"\n            )\n\n        # Not ideal, but this method is called by the stdlib and cannot return a\n        # coroutine so we just schedule the drain in a new thread and continue\n        from_sync.call_soon_in_new_thread(create_call(APILogWorker.drain_all))\n        return None\n    else:\n        # We set a timeout of 5s because we don't want to block forever if the worker\n        # is stuck. This can occur when the handler is being shutdown and the\n        # `logging._lock` is held but the worker is attempting to emit logs resulting\n        # in a deadlock.\n        return APILogWorker.drain_all(timeout=5)\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler.prepare","title":"prepare","text":"

    Convert a logging.LogRecord to the API LogCreate schema and serialize.

    This infers the linked flow or task run from the log record or the current run context.

    If a flow run id cannot be found, the log will be dropped.

    Logs exceeding the maximum size will be dropped.

    Source code in prefect/logging/handlers.py
    def prepare(self, record: logging.LogRecord) -> Dict[str, Any]:\n    \"\"\"\n    Convert a `logging.LogRecord` to the API `LogCreate` schema and serialize.\n\n    This infers the linked flow or task run from the log record or the current\n    run context.\n\n    If a flow run id cannot be found, the log will be dropped.\n\n    Logs exceeding the maximum size will be dropped.\n    \"\"\"\n    flow_run_id = getattr(record, \"flow_run_id\", None)\n    task_run_id = getattr(record, \"task_run_id\", None)\n\n    if not flow_run_id:\n        try:\n            context = prefect.context.get_run_context()\n        except MissingContextError:\n            raise MissingContextError(\n                f\"Logger {record.name!r} attempted to send logs to the API without\"\n                \" a flow run id. The API log handler can only send logs within\"\n                \" flow run contexts unless the flow run id is manually provided.\"\n            ) from None\n\n        if hasattr(context, \"flow_run\"):\n            flow_run_id = context.flow_run.id\n        elif hasattr(context, \"task_run\"):\n            flow_run_id = context.task_run.flow_run_id\n            task_run_id = task_run_id or context.task_run.id\n        else:\n            raise ValueError(\n                \"Encountered malformed run context. Does not contain flow or task \"\n                \"run information.\"\n            )\n\n    # Parsing to a `LogCreate` object here gives us nice parsing error messages\n    # from the standard lib `handleError` method if something goes wrong and\n    # prevents malformed logs from entering the queue\n    try:\n        is_uuid_like = isinstance(flow_run_id, uuid.UUID) or (\n            isinstance(flow_run_id, str) and uuid.UUID(flow_run_id)\n        )\n    except ValueError:\n        is_uuid_like = False\n\n    log = LogCreate(\n        flow_run_id=flow_run_id if is_uuid_like else None,\n        task_run_id=task_run_id,\n        name=record.name,\n        level=record.levelno,\n        timestamp=pendulum.from_timestamp(\n            getattr(record, \"created\", None) or time.time()\n        ),\n        message=self.format(record),\n    ).dict(json_compatible=True)\n\n    log_size = log[\"__payload_size__\"] = self._get_payload_size(log)\n    if log_size > PREFECT_LOGGING_TO_API_MAX_LOG_SIZE.value():\n        raise ValueError(\n            f\"Log of size {log_size} is greater than the max size of \"\n            f\"{PREFECT_LOGGING_TO_API_MAX_LOG_SIZE.value()}\"\n        )\n\n    return log\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.PrefectConsoleHandler","title":"PrefectConsoleHandler","text":"

    Bases: StreamHandler

    Source code in prefect/logging/handlers.py
    class PrefectConsoleHandler(logging.StreamHandler):\n    def __init__(\n        self,\n        stream=None,\n        highlighter: Highlighter = PrefectConsoleHighlighter,\n        styles: Dict[str, str] = None,\n        level: Union[int, str] = logging.NOTSET,\n    ):\n        \"\"\"\n        The default console handler for Prefect, which highlights log levels,\n        web and file URLs, flow and task (run) names, and state types in the\n        local console (terminal).\n\n        Highlighting can be toggled on/off with the PREFECT_LOGGING_COLORS setting.\n        For finer control, use logging.yml to add or remove styles, and/or\n        adjust colors.\n        \"\"\"\n        super().__init__(stream=stream)\n\n        styled_console = PREFECT_LOGGING_COLORS.value()\n        markup_console = PREFECT_LOGGING_MARKUP.value()\n        if styled_console:\n            highlighter = highlighter()\n            theme = Theme(styles, inherit=False)\n        else:\n            highlighter = NullHighlighter()\n            theme = Theme(inherit=False)\n\n        self.level = level\n        self.console = Console(\n            highlighter=highlighter,\n            theme=theme,\n            file=self.stream,\n            markup=markup_console,\n        )\n\n    def emit(self, record: logging.LogRecord):\n        try:\n            message = self.format(record)\n            self.console.print(message, soft_wrap=True)\n        except RecursionError:\n            # This was copied over from logging.StreamHandler().emit()\n            # https://bugs.python.org/issue36272\n            raise\n        except Exception:\n            self.handleError(record)\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/highlighters/","title":"highlighters","text":"

    \"\"\"

    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers","title":"prefect.logging.loggers","text":"","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.PrefectLogAdapter","title":"PrefectLogAdapter","text":"

    Bases: LoggerAdapter

    Adapter that ensures extra kwargs are passed through correctly; without this the extra fields set on the adapter would overshadow any provided on a log-by-log basis.

    See https://bugs.python.org/issue32732 \u2014 the Python team has declared that this is not a bug in the LoggingAdapter and subclassing is the intended workaround.

    Source code in prefect/logging/loggers.py
    class PrefectLogAdapter(logging.LoggerAdapter):\n    \"\"\"\n    Adapter that ensures extra kwargs are passed through correctly; without this\n    the `extra` fields set on the adapter would overshadow any provided on a\n    log-by-log basis.\n\n    See https://bugs.python.org/issue32732 \u2014 the Python team has declared that this is\n    not a bug in the LoggingAdapter and subclassing is the intended workaround.\n    \"\"\"\n\n    def process(self, msg, kwargs):\n        kwargs[\"extra\"] = {**(self.extra or {}), **(kwargs.get(\"extra\") or {})}\n\n        from prefect._internal.compatibility.deprecated import (\n            PrefectDeprecationWarning,\n            generate_deprecation_message,\n        )\n\n        if \"send_to_orion\" in kwargs[\"extra\"]:\n            warnings.warn(\n                generate_deprecation_message(\n                    'The \"send_to_orion\" option',\n                    start_date=\"May 2023\",\n                    help='Use \"send_to_api\" instead.',\n                ),\n                PrefectDeprecationWarning,\n                stacklevel=4,\n            )\n\n        return (msg, kwargs)\n\n    def getChild(\n        self, suffix: str, extra: Optional[Dict[str, str]] = None\n    ) -> \"PrefectLogAdapter\":\n        if extra is None:\n            extra = {}\n\n        return PrefectLogAdapter(\n            self.logger.getChild(suffix),\n            extra={\n                **self.extra,\n                **extra,\n            },\n        )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.disable_logger","title":"disable_logger","text":"

    Get a logger by name and disables it within the context manager. Upon exiting the context manager, the logger is returned to its original state.

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef disable_logger(name: str):\n    \"\"\"\n    Get a logger by name and disables it within the context manager.\n    Upon exiting the context manager, the logger is returned to its\n    original state.\n    \"\"\"\n    logger = logging.getLogger(name=name)\n\n    # determine if it's already disabled\n    base_state = logger.disabled\n    try:\n        # disable the logger\n        logger.disabled = True\n        yield\n    finally:\n        # return to base state\n        logger.disabled = base_state\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.disable_run_logger","title":"disable_run_logger","text":"

    Gets both prefect.flow_run and prefect.task_run and disables them within the context manager. Upon exiting the context manager, both loggers are returned to its original state.

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef disable_run_logger():\n    \"\"\"\n    Gets both `prefect.flow_run` and `prefect.task_run` and disables them\n    within the context manager. Upon exiting the context manager, both loggers\n    are returned to its original state.\n    \"\"\"\n    with disable_logger(\"prefect.flow_run\"), disable_logger(\"prefect.task_run\"):\n        yield\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.flow_run_logger","title":"flow_run_logger","text":"

    Create a flow run logger with the run's metadata attached.

    Additional keyword arguments can be provided to attach custom data to the log records.

    If the flow run context is available, see get_run_logger instead.

    Source code in prefect/logging/loggers.py
    def flow_run_logger(\n    flow_run: Union[\"FlowRun\", \"ClientFlowRun\"],\n    flow: Optional[\"Flow\"] = None,\n    **kwargs: str,\n):\n    \"\"\"\n    Create a flow run logger with the run's metadata attached.\n\n    Additional keyword arguments can be provided to attach custom data to the log\n    records.\n\n    If the flow run context is available, see `get_run_logger` instead.\n    \"\"\"\n    return PrefectLogAdapter(\n        get_logger(\"prefect.flow_runs\"),\n        extra={\n            **{\n                \"flow_run_name\": flow_run.name if flow_run else \"<unknown>\",\n                \"flow_run_id\": str(flow_run.id) if flow_run else \"<unknown>\",\n                \"flow_name\": flow.name if flow else \"<unknown>\",\n            },\n            **kwargs,\n        },\n    )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.get_logger","title":"get_logger cached","text":"

    Get a prefect logger. These loggers are intended for internal use within the prefect package.

    See get_run_logger for retrieving loggers for use within task or flow runs. By default, only run-related loggers are connected to the APILogHandler.

    Source code in prefect/logging/loggers.py
    @lru_cache()\ndef get_logger(name: str = None) -> logging.Logger:\n    \"\"\"\n    Get a `prefect` logger. These loggers are intended for internal use within the\n    `prefect` package.\n\n    See `get_run_logger` for retrieving loggers for use within task or flow runs.\n    By default, only run-related loggers are connected to the `APILogHandler`.\n    \"\"\"\n\n    parent_logger = logging.getLogger(\"prefect\")\n\n    if name:\n        # Append the name if given but allow explicit full names e.g. \"prefect.test\"\n        # should not become \"prefect.prefect.test\"\n        if not name.startswith(parent_logger.name + \".\"):\n            logger = parent_logger.getChild(name)\n        else:\n            logger = logging.getLogger(name)\n    else:\n        logger = parent_logger\n\n    return logger\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.get_run_logger","title":"get_run_logger","text":"

    Get a Prefect logger for the current task run or flow run.

    The logger will be named either prefect.task_runs or prefect.flow_runs. Contextual data about the run will be attached to the log records.

    These loggers are connected to the APILogHandler by default to send log records to the API.

    Parameters:

    Name Type Description Default context RunContext

    A specific context may be provided as an override. By default, the context is inferred from global state and this should not be needed.

    None **kwargs str

    Additional keyword arguments will be attached to the log records in addition to the run metadata

    {}

    Raises:

    Type Description RuntimeError

    If no context can be found

    Source code in prefect/logging/loggers.py
    def get_run_logger(\n    context: \"RunContext\" = None, **kwargs: str\n) -> Union[logging.Logger, logging.LoggerAdapter]:\n    \"\"\"\n    Get a Prefect logger for the current task run or flow run.\n\n    The logger will be named either `prefect.task_runs` or `prefect.flow_runs`.\n    Contextual data about the run will be attached to the log records.\n\n    These loggers are connected to the `APILogHandler` by default to send log records to\n    the API.\n\n    Arguments:\n        context: A specific context may be provided as an override. By default, the\n            context is inferred from global state and this should not be needed.\n        **kwargs: Additional keyword arguments will be attached to the log records in\n            addition to the run metadata\n\n    Raises:\n        RuntimeError: If no context can be found\n    \"\"\"\n    # Check for existing contexts\n    task_run_context = prefect.context.TaskRunContext.get()\n    flow_run_context = prefect.context.FlowRunContext.get()\n\n    # Apply the context override\n    if context:\n        if isinstance(context, prefect.context.FlowRunContext):\n            flow_run_context = context\n        elif isinstance(context, prefect.context.TaskRunContext):\n            task_run_context = context\n        else:\n            raise TypeError(\n                f\"Received unexpected type {type(context).__name__!r} for context. \"\n                \"Expected one of 'None', 'FlowRunContext', or 'TaskRunContext'.\"\n            )\n\n    # Determine if this is a task or flow run logger\n    if task_run_context:\n        logger = task_run_logger(\n            task_run=task_run_context.task_run,\n            task=task_run_context.task,\n            flow_run=flow_run_context.flow_run if flow_run_context else None,\n            flow=flow_run_context.flow if flow_run_context else None,\n            **kwargs,\n        )\n    elif flow_run_context:\n        logger = flow_run_logger(\n            flow_run=flow_run_context.flow_run, flow=flow_run_context.flow, **kwargs\n        )\n    elif (\n        get_logger(\"prefect.flow_run\").disabled\n        and get_logger(\"prefect.task_run\").disabled\n    ):\n        logger = logging.getLogger(\"null\")\n    else:\n        raise MissingContextError(\"There is no active flow or task run context.\")\n\n    return logger\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.patch_print","title":"patch_print","text":"

    Patches the Python builtin print method to use print_as_log

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef patch_print():\n    \"\"\"\n    Patches the Python builtin `print` method to use `print_as_log`\n    \"\"\"\n    import builtins\n\n    original = builtins.print\n\n    try:\n        builtins.print = print_as_log\n        yield\n    finally:\n        builtins.print = original\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.print_as_log","title":"print_as_log","text":"

    A patch for print to send printed messages to the Prefect run logger.

    If no run is active, print will behave as if it were not patched.

    If print sends data to a file other than sys.stdout or sys.stderr, it will not be forwarded to the Prefect logger either.

    Source code in prefect/logging/loggers.py
    def print_as_log(*args, **kwargs):\n    \"\"\"\n    A patch for `print` to send printed messages to the Prefect run logger.\n\n    If no run is active, `print` will behave as if it were not patched.\n\n    If `print` sends data to a file other than `sys.stdout` or `sys.stderr`, it will\n    not be forwarded to the Prefect logger either.\n    \"\"\"\n    from prefect.context import FlowRunContext, TaskRunContext\n\n    context = TaskRunContext.get() or FlowRunContext.get()\n    if (\n        not context\n        or not context.log_prints\n        or kwargs.get(\"file\") not in {None, sys.stdout, sys.stderr}\n    ):\n        return print(*args, **kwargs)\n\n    logger = get_run_logger()\n\n    # Print to an in-memory buffer; so we do not need to implement `print`\n    buffer = io.StringIO()\n    kwargs[\"file\"] = buffer\n    print(*args, **kwargs)\n\n    # Remove trailing whitespace to prevent duplicates\n    logger.info(buffer.getvalue().rstrip())\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.task_run_logger","title":"task_run_logger","text":"

    Create a task run logger with the run's metadata attached.

    Additional keyword arguments can be provided to attach custom data to the log records.

    If the task run context is available, see get_run_logger instead.

    If only the flow run context is available, it will be used for default values of flow_run and flow.

    Source code in prefect/logging/loggers.py
    def task_run_logger(\n    task_run: \"TaskRun\",\n    task: \"Task\" = None,\n    flow_run: \"FlowRun\" = None,\n    flow: \"Flow\" = None,\n    **kwargs: str,\n):\n    \"\"\"\n    Create a task run logger with the run's metadata attached.\n\n    Additional keyword arguments can be provided to attach custom data to the log\n    records.\n\n    If the task run context is available, see `get_run_logger` instead.\n\n    If only the flow run context is available, it will be used for default values\n    of `flow_run` and `flow`.\n    \"\"\"\n    if not flow_run or not flow:\n        flow_run_context = prefect.context.FlowRunContext.get()\n        if flow_run_context:\n            flow_run = flow_run or flow_run_context.flow_run\n            flow = flow or flow_run_context.flow\n\n    return PrefectLogAdapter(\n        get_logger(\"prefect.task_runs\"),\n        extra={\n            **{\n                \"task_run_id\": str(task_run.id),\n                \"flow_run_id\": str(task_run.flow_run_id),\n                \"task_run_name\": task_run.name,\n                \"task_name\": task.name if task else \"<unknown>\",\n                \"flow_run_name\": flow_run.name if flow_run else \"<unknown>\",\n                \"flow_name\": flow.name if flow else \"<unknown>\",\n            },\n            **kwargs,\n        },\n    )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/","title":"loggers","text":"

    \"\"\"

    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers","title":"prefect.logging.loggers","text":"","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.PrefectLogAdapter","title":"PrefectLogAdapter","text":"

    Bases: LoggerAdapter

    Adapter that ensures extra kwargs are passed through correctly; without this the extra fields set on the adapter would overshadow any provided on a log-by-log basis.

    See https://bugs.python.org/issue32732 \u2014 the Python team has declared that this is not a bug in the LoggingAdapter and subclassing is the intended workaround.

    Source code in prefect/logging/loggers.py
    class PrefectLogAdapter(logging.LoggerAdapter):\n    \"\"\"\n    Adapter that ensures extra kwargs are passed through correctly; without this\n    the `extra` fields set on the adapter would overshadow any provided on a\n    log-by-log basis.\n\n    See https://bugs.python.org/issue32732 \u2014 the Python team has declared that this is\n    not a bug in the LoggingAdapter and subclassing is the intended workaround.\n    \"\"\"\n\n    def process(self, msg, kwargs):\n        kwargs[\"extra\"] = {**(self.extra or {}), **(kwargs.get(\"extra\") or {})}\n\n        from prefect._internal.compatibility.deprecated import (\n            PrefectDeprecationWarning,\n            generate_deprecation_message,\n        )\n\n        if \"send_to_orion\" in kwargs[\"extra\"]:\n            warnings.warn(\n                generate_deprecation_message(\n                    'The \"send_to_orion\" option',\n                    start_date=\"May 2023\",\n                    help='Use \"send_to_api\" instead.',\n                ),\n                PrefectDeprecationWarning,\n                stacklevel=4,\n            )\n\n        return (msg, kwargs)\n\n    def getChild(\n        self, suffix: str, extra: Optional[Dict[str, str]] = None\n    ) -> \"PrefectLogAdapter\":\n        if extra is None:\n            extra = {}\n\n        return PrefectLogAdapter(\n            self.logger.getChild(suffix),\n            extra={\n                **self.extra,\n                **extra,\n            },\n        )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.disable_logger","title":"disable_logger","text":"

    Get a logger by name and disables it within the context manager. Upon exiting the context manager, the logger is returned to its original state.

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef disable_logger(name: str):\n    \"\"\"\n    Get a logger by name and disables it within the context manager.\n    Upon exiting the context manager, the logger is returned to its\n    original state.\n    \"\"\"\n    logger = logging.getLogger(name=name)\n\n    # determine if it's already disabled\n    base_state = logger.disabled\n    try:\n        # disable the logger\n        logger.disabled = True\n        yield\n    finally:\n        # return to base state\n        logger.disabled = base_state\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.disable_run_logger","title":"disable_run_logger","text":"

    Gets both prefect.flow_run and prefect.task_run and disables them within the context manager. Upon exiting the context manager, both loggers are returned to its original state.

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef disable_run_logger():\n    \"\"\"\n    Gets both `prefect.flow_run` and `prefect.task_run` and disables them\n    within the context manager. Upon exiting the context manager, both loggers\n    are returned to its original state.\n    \"\"\"\n    with disable_logger(\"prefect.flow_run\"), disable_logger(\"prefect.task_run\"):\n        yield\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.flow_run_logger","title":"flow_run_logger","text":"

    Create a flow run logger with the run's metadata attached.

    Additional keyword arguments can be provided to attach custom data to the log records.

    If the flow run context is available, see get_run_logger instead.

    Source code in prefect/logging/loggers.py
    def flow_run_logger(\n    flow_run: Union[\"FlowRun\", \"ClientFlowRun\"],\n    flow: Optional[\"Flow\"] = None,\n    **kwargs: str,\n):\n    \"\"\"\n    Create a flow run logger with the run's metadata attached.\n\n    Additional keyword arguments can be provided to attach custom data to the log\n    records.\n\n    If the flow run context is available, see `get_run_logger` instead.\n    \"\"\"\n    return PrefectLogAdapter(\n        get_logger(\"prefect.flow_runs\"),\n        extra={\n            **{\n                \"flow_run_name\": flow_run.name if flow_run else \"<unknown>\",\n                \"flow_run_id\": str(flow_run.id) if flow_run else \"<unknown>\",\n                \"flow_name\": flow.name if flow else \"<unknown>\",\n            },\n            **kwargs,\n        },\n    )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.get_logger","title":"get_logger cached","text":"

    Get a prefect logger. These loggers are intended for internal use within the prefect package.

    See get_run_logger for retrieving loggers for use within task or flow runs. By default, only run-related loggers are connected to the APILogHandler.

    Source code in prefect/logging/loggers.py
    @lru_cache()\ndef get_logger(name: str = None) -> logging.Logger:\n    \"\"\"\n    Get a `prefect` logger. These loggers are intended for internal use within the\n    `prefect` package.\n\n    See `get_run_logger` for retrieving loggers for use within task or flow runs.\n    By default, only run-related loggers are connected to the `APILogHandler`.\n    \"\"\"\n\n    parent_logger = logging.getLogger(\"prefect\")\n\n    if name:\n        # Append the name if given but allow explicit full names e.g. \"prefect.test\"\n        # should not become \"prefect.prefect.test\"\n        if not name.startswith(parent_logger.name + \".\"):\n            logger = parent_logger.getChild(name)\n        else:\n            logger = logging.getLogger(name)\n    else:\n        logger = parent_logger\n\n    return logger\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.get_run_logger","title":"get_run_logger","text":"

    Get a Prefect logger for the current task run or flow run.

    The logger will be named either prefect.task_runs or prefect.flow_runs. Contextual data about the run will be attached to the log records.

    These loggers are connected to the APILogHandler by default to send log records to the API.

    Parameters:

    Name Type Description Default context RunContext

    A specific context may be provided as an override. By default, the context is inferred from global state and this should not be needed.

    None **kwargs str

    Additional keyword arguments will be attached to the log records in addition to the run metadata

    {}

    Raises:

    Type Description RuntimeError

    If no context can be found

    Source code in prefect/logging/loggers.py
    def get_run_logger(\n    context: \"RunContext\" = None, **kwargs: str\n) -> Union[logging.Logger, logging.LoggerAdapter]:\n    \"\"\"\n    Get a Prefect logger for the current task run or flow run.\n\n    The logger will be named either `prefect.task_runs` or `prefect.flow_runs`.\n    Contextual data about the run will be attached to the log records.\n\n    These loggers are connected to the `APILogHandler` by default to send log records to\n    the API.\n\n    Arguments:\n        context: A specific context may be provided as an override. By default, the\n            context is inferred from global state and this should not be needed.\n        **kwargs: Additional keyword arguments will be attached to the log records in\n            addition to the run metadata\n\n    Raises:\n        RuntimeError: If no context can be found\n    \"\"\"\n    # Check for existing contexts\n    task_run_context = prefect.context.TaskRunContext.get()\n    flow_run_context = prefect.context.FlowRunContext.get()\n\n    # Apply the context override\n    if context:\n        if isinstance(context, prefect.context.FlowRunContext):\n            flow_run_context = context\n        elif isinstance(context, prefect.context.TaskRunContext):\n            task_run_context = context\n        else:\n            raise TypeError(\n                f\"Received unexpected type {type(context).__name__!r} for context. \"\n                \"Expected one of 'None', 'FlowRunContext', or 'TaskRunContext'.\"\n            )\n\n    # Determine if this is a task or flow run logger\n    if task_run_context:\n        logger = task_run_logger(\n            task_run=task_run_context.task_run,\n            task=task_run_context.task,\n            flow_run=flow_run_context.flow_run if flow_run_context else None,\n            flow=flow_run_context.flow if flow_run_context else None,\n            **kwargs,\n        )\n    elif flow_run_context:\n        logger = flow_run_logger(\n            flow_run=flow_run_context.flow_run, flow=flow_run_context.flow, **kwargs\n        )\n    elif (\n        get_logger(\"prefect.flow_run\").disabled\n        and get_logger(\"prefect.task_run\").disabled\n    ):\n        logger = logging.getLogger(\"null\")\n    else:\n        raise MissingContextError(\"There is no active flow or task run context.\")\n\n    return logger\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.patch_print","title":"patch_print","text":"

    Patches the Python builtin print method to use print_as_log

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef patch_print():\n    \"\"\"\n    Patches the Python builtin `print` method to use `print_as_log`\n    \"\"\"\n    import builtins\n\n    original = builtins.print\n\n    try:\n        builtins.print = print_as_log\n        yield\n    finally:\n        builtins.print = original\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.print_as_log","title":"print_as_log","text":"

    A patch for print to send printed messages to the Prefect run logger.

    If no run is active, print will behave as if it were not patched.

    If print sends data to a file other than sys.stdout or sys.stderr, it will not be forwarded to the Prefect logger either.

    Source code in prefect/logging/loggers.py
    def print_as_log(*args, **kwargs):\n    \"\"\"\n    A patch for `print` to send printed messages to the Prefect run logger.\n\n    If no run is active, `print` will behave as if it were not patched.\n\n    If `print` sends data to a file other than `sys.stdout` or `sys.stderr`, it will\n    not be forwarded to the Prefect logger either.\n    \"\"\"\n    from prefect.context import FlowRunContext, TaskRunContext\n\n    context = TaskRunContext.get() or FlowRunContext.get()\n    if (\n        not context\n        or not context.log_prints\n        or kwargs.get(\"file\") not in {None, sys.stdout, sys.stderr}\n    ):\n        return print(*args, **kwargs)\n\n    logger = get_run_logger()\n\n    # Print to an in-memory buffer; so we do not need to implement `print`\n    buffer = io.StringIO()\n    kwargs[\"file\"] = buffer\n    print(*args, **kwargs)\n\n    # Remove trailing whitespace to prevent duplicates\n    logger.info(buffer.getvalue().rstrip())\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.task_run_logger","title":"task_run_logger","text":"

    Create a task run logger with the run's metadata attached.

    Additional keyword arguments can be provided to attach custom data to the log records.

    If the task run context is available, see get_run_logger instead.

    If only the flow run context is available, it will be used for default values of flow_run and flow.

    Source code in prefect/logging/loggers.py
    def task_run_logger(\n    task_run: \"TaskRun\",\n    task: \"Task\" = None,\n    flow_run: \"FlowRun\" = None,\n    flow: \"Flow\" = None,\n    **kwargs: str,\n):\n    \"\"\"\n    Create a task run logger with the run's metadata attached.\n\n    Additional keyword arguments can be provided to attach custom data to the log\n    records.\n\n    If the task run context is available, see `get_run_logger` instead.\n\n    If only the flow run context is available, it will be used for default values\n    of `flow_run` and `flow`.\n    \"\"\"\n    if not flow_run or not flow:\n        flow_run_context = prefect.context.FlowRunContext.get()\n        if flow_run_context:\n            flow_run = flow_run or flow_run_context.flow_run\n            flow = flow or flow_run_context.flow\n\n    return PrefectLogAdapter(\n        get_logger(\"prefect.task_runs\"),\n        extra={\n            **{\n                \"task_run_id\": str(task_run.id),\n                \"flow_run_id\": str(task_run.flow_run_id),\n                \"task_run_name\": task_run.name,\n                \"task_name\": task.name if task else \"<unknown>\",\n                \"flow_run_name\": flow_run.name if flow_run else \"<unknown>\",\n                \"flow_name\": flow.name if flow else \"<unknown>\",\n            },\n            **kwargs,\n        },\n    )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/packaging/base/","title":"base","text":"","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base","title":"prefect.packaging.base","text":"","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.PackageManifest","title":"PackageManifest","text":"

    Bases: BaseModel, ABC

    Describes a package.

    Source code in prefect/packaging/base.py
    @add_type_dispatch\nclass PackageManifest(BaseModel, abc.ABC):\n    \"\"\"\n    Describes a package.\n    \"\"\"\n\n    type: str\n    flow_name: str\n    flow_parameter_schema: ParameterSchema\n\n    @abc.abstractmethod\n    async def unpackage(self) -> Flow:\n        \"\"\"\n        Retrieve a flow from the package.\n        \"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.PackageManifest.unpackage","title":"unpackage abstractmethod async","text":"

    Retrieve a flow from the package.

    Source code in prefect/packaging/base.py
    @abc.abstractmethod\nasync def unpackage(self) -> Flow:\n    \"\"\"\n    Retrieve a flow from the package.\n    \"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Packager","title":"Packager","text":"

    Bases: BaseModel, ABC

    Creates a package for a flow.

    A package contains the flow and is typically stored outside of Prefect. To facilitate interaction with the package, a manifest is returned that describes how to access and use the package.

    Source code in prefect/packaging/base.py
    @add_type_dispatch\nclass Packager(BaseModel, abc.ABC):\n    \"\"\"\n    Creates a package for a flow.\n\n    A package contains the flow and is typically stored outside of Prefect. To\n    facilitate interaction with the package, a manifest is returned that describes how\n    to access and use the package.\n    \"\"\"\n\n    type: str\n\n    def base_manifest(self, flow: Flow) -> PartialModel[PackageManifest]:\n        manifest_cls = lookup_type(PackageManifest, self.type)\n        return PartialModel(\n            manifest_cls,\n            type=self.type,\n            flow_name=flow.name,\n            flow_parameter_schema=parameter_schema(flow.fn),\n        )\n\n    @abc.abstractmethod\n    async def package(self, flow: Flow) -> \"PackageManifest\":\n        \"\"\"\n        Package a flow and return a manifest describing the created package.\n        \"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Packager.package","title":"package abstractmethod async","text":"

    Package a flow and return a manifest describing the created package.

    Source code in prefect/packaging/base.py
    @abc.abstractmethod\nasync def package(self, flow: Flow) -> \"PackageManifest\":\n    \"\"\"\n    Package a flow and return a manifest describing the created package.\n    \"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Serializer","title":"Serializer","text":"

    Bases: BaseModel, Generic[D], ABC

    A serializer that can encode objects of type 'D' into bytes.

    Source code in prefect/packaging/base.py
    @add_type_dispatch\nclass Serializer(BaseModel, Generic[D], abc.ABC):\n    \"\"\"\n    A serializer that can encode objects of type 'D' into bytes.\n    \"\"\"\n\n    type: str\n\n    def dumps(self, obj: D) -> bytes:\n        \"\"\"Encode the object into a blob of bytes.\"\"\"\n\n    def loads(self, blob: bytes) -> D:\n        \"\"\"Decode the blob of bytes into an object.\"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Serializer.dumps","title":"dumps","text":"

    Encode the object into a blob of bytes.

    Source code in prefect/packaging/base.py
    def dumps(self, obj: D) -> bytes:\n    \"\"\"Encode the object into a blob of bytes.\"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Serializer.loads","title":"loads","text":"

    Decode the blob of bytes into an object.

    Source code in prefect/packaging/base.py
    def loads(self, blob: bytes) -> D:\n    \"\"\"Decode the blob of bytes into an object.\"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/docker/","title":"docker","text":"","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/docker/#prefect.packaging.docker","title":"prefect.packaging.docker","text":"","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/docker/#prefect.packaging.docker.DockerPackageManifest","title":"DockerPackageManifest","text":"

    Bases: PackageManifest

    Represents a flow packaged in a Docker image

    Source code in prefect/packaging/docker.py
    class DockerPackageManifest(PackageManifest):\n    \"\"\"\n    Represents a flow packaged in a Docker image\n    \"\"\"\n\n    type: Literal[\"docker\"] = \"docker\"\n\n    image: str\n    image_flow_location: str\n\n    async def unpackage(self) -> Flow:\n        return load_flow_from_script(self.image_flow_location, self.flow_name)\n
    ","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/docker/#prefect.packaging.docker.DockerPackager","title":"DockerPackager","text":"

    Bases: Packager

    This packager builds a Docker image containing the flow and the runtime environment necessary to run the flow. The resulting image is optionally pushed to a container registry, given by registry_url.

    Source code in prefect/packaging/docker.py
    class DockerPackager(Packager):\n    \"\"\"\n    This packager builds a Docker image containing the flow and the runtime environment\n    necessary to run the flow.  The resulting image is optionally pushed to a container\n    registry, given by `registry_url`.\n    \"\"\"\n\n    type: Literal[\"docker\"] = \"docker\"\n\n    base_image: Optional[str] = None\n    python_environment: Optional[Union[PythonEnvironment, CondaEnvironment]] = None\n    dockerfile: Optional[Path] = None\n    platform: Optional[str] = (None,)\n    image_flow_location: str = \"/flow.py\"\n    registry_url: Optional[AnyHttpUrl] = None\n\n    @root_validator\n    def set_default_base_image(cls, values):\n        if not values.get(\"base_image\") and not values.get(\"dockerfile\"):\n            values[\"base_image\"] = get_prefect_image_name(\n                flavor=(\n                    \"conda\"\n                    if isinstance(values.get(\"python_environment\"), CondaEnvironment)\n                    else None\n                )\n            )\n        return values\n\n    @root_validator\n    def base_image_and_dockerfile_exclusive(cls, values: Mapping[str, Any]):\n        if values.get(\"base_image\") and values.get(\"dockerfile\"):\n            raise ValueError(\n                \"Either `base_image` or `dockerfile` should be provided, but not both\"\n            )\n        return values\n\n    @root_validator\n    def default_python_environment(cls, values: Mapping[str, Any]):\n        if values.get(\"base_image\") and not values.get(\"python_environment\"):\n            values[\"python_environment\"] = PythonEnvironment.from_environment()\n        return values\n\n    @validator(\"registry_url\", pre=True)\n    def ensure_registry_url_is_prefixed(cls, value):\n        if isinstance(value, str):\n            if \"://\" not in value:\n                return \"https://\" + value\n        return value\n\n    async def package(self, flow: Flow) -> DockerPackageManifest:\n        \"\"\"\n        Package a flow as a Docker image and, optionally, push it to a registry\n        \"\"\"\n        image_reference = await self._build_image(flow)\n\n        if self.registry_url:\n            image_name = f\"{slugify(flow.name)}\"\n            image_reference = await run_sync_in_worker_thread(\n                push_image, image_reference, self.registry_url, image_name\n            )\n\n        return self.base_manifest(flow).finalize(\n            image=image_reference, image_flow_location=self.image_flow_location\n        )\n\n    async def _build_image(self, flow: Flow) -> str:\n        if self.dockerfile:\n            return await self._build_from_dockerfile()\n        return await self._build_from_base_image(flow)\n\n    async def _build_from_dockerfile(self) -> str:\n        context = self.dockerfile.resolve().parent\n        dockerfile = self.dockerfile.relative_to(context)\n        return await run_sync_in_worker_thread(\n            build_image,\n            platform=self.platform,\n            context=context,\n            dockerfile=str(dockerfile),\n        )\n\n    async def _build_from_base_image(self, flow: Flow) -> str:\n        with ImageBuilder(\n            base_image=self.base_image, platform=self.platform\n        ) as builder:\n            for command in self.python_environment.install_commands():\n                builder.add_line(to_run_command(command))\n\n            source_info = json.loads(SourceSerializer().dumps(flow))\n\n            builder.write_text(source_info[\"source\"], self.image_flow_location)\n\n            return await run_sync_in_worker_thread(\n                builder.build, stream_progress_to=sys.stdout\n            )\n
    ","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/docker/#prefect.packaging.docker.DockerPackager.package","title":"package async","text":"

    Package a flow as a Docker image and, optionally, push it to a registry

    Source code in prefect/packaging/docker.py
    async def package(self, flow: Flow) -> DockerPackageManifest:\n    \"\"\"\n    Package a flow as a Docker image and, optionally, push it to a registry\n    \"\"\"\n    image_reference = await self._build_image(flow)\n\n    if self.registry_url:\n        image_name = f\"{slugify(flow.name)}\"\n        image_reference = await run_sync_in_worker_thread(\n            push_image, image_reference, self.registry_url, image_name\n        )\n\n    return self.base_manifest(flow).finalize(\n        image=image_reference, image_flow_location=self.image_flow_location\n    )\n
    ","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/file/","title":"file","text":"","tags":["Python API","packaging","file"]},{"location":"api-ref/prefect/packaging/file/#prefect.packaging.file","title":"prefect.packaging.file","text":"","tags":["Python API","packaging","file"]},{"location":"api-ref/prefect/packaging/file/#prefect.packaging.file.FilePackager","title":"FilePackager","text":"

    Bases: Packager

    This packager stores the flow as a single file.

    By default, the file is the source code of the module the flow is defined in. Alternative serialization modes are available in prefect.packaging.serializers.

    Source code in prefect/packaging/file.py
    class FilePackager(Packager):\n    \"\"\"\n    This packager stores the flow as a single file.\n\n    By default, the file is the source code of the module the flow is defined in.\n    Alternative serialization modes are available in `prefect.packaging.serializers`.\n    \"\"\"\n\n    type: Literal[\"file\"] = \"file\"\n    serializer: Serializer = Field(default_factory=SourceSerializer)\n    filesystem: WritableFileSystem = Field(\n        default_factory=lambda: LocalFileSystem(\n            basepath=PREFECT_HOME.value() / \"storage\"\n        )\n    )\n\n    @inject_client\n    async def package(self, flow: Flow, client: \"PrefectClient\") -> FilePackageManifest:\n        content = self.serializer.dumps(flow)\n        key = stable_hash(content)\n\n        await self.filesystem.write_path(key, content)\n\n        filesystem_id = (\n            self.filesystem._block_document_id\n            or await self.filesystem._save(is_anonymous=True)\n        )\n\n        return self.base_manifest(flow).finalize(\n            serializer=self.serializer,\n            filesystem_id=filesystem_id,\n            key=key,\n        )\n
    ","tags":["Python API","packaging","file"]},{"location":"api-ref/prefect/packaging/serializers/","title":"serializers","text":"","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers","title":"prefect.packaging.serializers","text":"","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.ImportSerializer","title":"ImportSerializer","text":"

    Bases: Serializer

    Serializes objects by storing their importable path.

    Source code in prefect/packaging/serializers.py
    class ImportSerializer(Serializer):\n    \"\"\"\n    Serializes objects by storing their importable path.\n    \"\"\"\n\n    type: Literal[\"import\"] = \"import\"\n\n    def dumps(self, obj: Any) -> bytes:\n        return to_qualified_name(obj).encode()\n\n    def loads(self, blob: bytes) -> Any:\n        return from_qualified_name(blob.decode())\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.PickleSerializer","title":"PickleSerializer","text":"

    Bases: Serializer

    Serializes objects using the pickle protocol.

    If using cloudpickle, you may specify a list of 'pickle_modules'. These modules will be serialized by value instead of by reference, which means they do not have to be installed in the runtime location. This is especially useful for serializing objects that rely on local packages.

    Wraps pickles in base64 for safe transmission.

    Source code in prefect/packaging/serializers.py
    class PickleSerializer(Serializer):\n    \"\"\"\n    Serializes objects using the pickle protocol.\n\n    If using cloudpickle, you may specify a list of 'pickle_modules'. These modules will\n    be serialized by value instead of by reference, which means they do not have to be\n    installed in the runtime location. This is especially useful for serializing objects\n    that rely on local packages.\n\n    Wraps pickles in base64 for safe transmission.\n    \"\"\"\n\n    type: Literal[\"pickle\"] = \"pickle\"\n\n    picklelib: str = \"cloudpickle\"\n    picklelib_version: str = None\n    pickle_modules: List[str] = pydantic.Field(default_factory=list)\n\n    @pydantic.validator(\"picklelib\")\n    def check_picklelib(cls, value):\n        \"\"\"\n        Check that the given pickle library is importable and has dumps/loads methods.\n        \"\"\"\n        try:\n            pickler = from_qualified_name(value)\n        except (ImportError, AttributeError) as exc:\n            raise ValueError(\n                f\"Failed to import requested pickle library: {value!r}.\"\n            ) from exc\n\n        if not callable(getattr(pickler, \"dumps\", None)):\n            raise ValueError(\n                f\"Pickle library at {value!r} does not have a 'dumps' method.\"\n            )\n\n        if not callable(getattr(pickler, \"loads\", None)):\n            raise ValueError(\n                f\"Pickle library at {value!r} does not have a 'loads' method.\"\n            )\n\n        return value\n\n    @pydantic.root_validator\n    def check_picklelib_version(cls, values):\n        \"\"\"\n        Infers a default value for `picklelib_version` if null or ensures it matches\n        the version retrieved from the `pickelib`.\n        \"\"\"\n        picklelib = values.get(\"picklelib\")\n        picklelib_version = values.get(\"picklelib_version\")\n\n        if not picklelib:\n            raise ValueError(\"Unable to check version of unrecognized picklelib module\")\n\n        pickler = from_qualified_name(picklelib)\n        pickler_version = getattr(pickler, \"__version__\", None)\n\n        if not picklelib_version:\n            values[\"picklelib_version\"] = pickler_version\n        elif picklelib_version != pickler_version:\n            warnings.warn(\n                (\n                    f\"Mismatched {picklelib!r} versions. Found {pickler_version} in the\"\n                    f\" environment but {picklelib_version} was requested. This may\"\n                    \" cause the serializer to fail.\"\n                ),\n                RuntimeWarning,\n                stacklevel=3,\n            )\n\n        return values\n\n    @pydantic.root_validator\n    def check_picklelib_and_modules(cls, values):\n        \"\"\"\n        Prevents modules from being specified if picklelib is not cloudpickle\n        \"\"\"\n        if values.get(\"picklelib\") != \"cloudpickle\" and values.get(\"pickle_modules\"):\n            raise ValueError(\n                \"`pickle_modules` cannot be used without 'cloudpickle'. Got\"\n                f\" {values.get('picklelib')!r}.\"\n            )\n        return values\n\n    def dumps(self, obj: Any) -> bytes:\n        pickler = from_qualified_name(self.picklelib)\n\n        for module in self.pickle_modules:\n            pickler.register_pickle_by_value(from_qualified_name(module))\n\n        blob = pickler.dumps(obj)\n\n        for module in self.pickle_modules:\n            # Restore the pickler settings\n            pickler.unregister_pickle_by_value(from_qualified_name(module))\n\n        return base64.encodebytes(blob)\n\n    def loads(self, blob: bytes) -> Any:\n        pickler = from_qualified_name(self.picklelib)\n        return pickler.loads(base64.decodebytes(blob))\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.PickleSerializer.check_picklelib","title":"check_picklelib","text":"

    Check that the given pickle library is importable and has dumps/loads methods.

    Source code in prefect/packaging/serializers.py
    @pydantic.validator(\"picklelib\")\ndef check_picklelib(cls, value):\n    \"\"\"\n    Check that the given pickle library is importable and has dumps/loads methods.\n    \"\"\"\n    try:\n        pickler = from_qualified_name(value)\n    except (ImportError, AttributeError) as exc:\n        raise ValueError(\n            f\"Failed to import requested pickle library: {value!r}.\"\n        ) from exc\n\n    if not callable(getattr(pickler, \"dumps\", None)):\n        raise ValueError(\n            f\"Pickle library at {value!r} does not have a 'dumps' method.\"\n        )\n\n    if not callable(getattr(pickler, \"loads\", None)):\n        raise ValueError(\n            f\"Pickle library at {value!r} does not have a 'loads' method.\"\n        )\n\n    return value\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.PickleSerializer.check_picklelib_and_modules","title":"check_picklelib_and_modules","text":"

    Prevents modules from being specified if picklelib is not cloudpickle

    Source code in prefect/packaging/serializers.py
    @pydantic.root_validator\ndef check_picklelib_and_modules(cls, values):\n    \"\"\"\n    Prevents modules from being specified if picklelib is not cloudpickle\n    \"\"\"\n    if values.get(\"picklelib\") != \"cloudpickle\" and values.get(\"pickle_modules\"):\n        raise ValueError(\n            \"`pickle_modules` cannot be used without 'cloudpickle'. Got\"\n            f\" {values.get('picklelib')!r}.\"\n        )\n    return values\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.PickleSerializer.check_picklelib_version","title":"check_picklelib_version","text":"

    Infers a default value for picklelib_version if null or ensures it matches the version retrieved from the pickelib.

    Source code in prefect/packaging/serializers.py
    @pydantic.root_validator\ndef check_picklelib_version(cls, values):\n    \"\"\"\n    Infers a default value for `picklelib_version` if null or ensures it matches\n    the version retrieved from the `pickelib`.\n    \"\"\"\n    picklelib = values.get(\"picklelib\")\n    picklelib_version = values.get(\"picklelib_version\")\n\n    if not picklelib:\n        raise ValueError(\"Unable to check version of unrecognized picklelib module\")\n\n    pickler = from_qualified_name(picklelib)\n    pickler_version = getattr(pickler, \"__version__\", None)\n\n    if not picklelib_version:\n        values[\"picklelib_version\"] = pickler_version\n    elif picklelib_version != pickler_version:\n        warnings.warn(\n            (\n                f\"Mismatched {picklelib!r} versions. Found {pickler_version} in the\"\n                f\" environment but {picklelib_version} was requested. This may\"\n                \" cause the serializer to fail.\"\n            ),\n            RuntimeWarning,\n            stacklevel=3,\n        )\n\n    return values\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.SourceSerializer","title":"SourceSerializer","text":"

    Bases: Serializer

    Serializes objects by retrieving the source code of the module they are defined in.

    Creates a JSON blob with keys

    Deserialization requires the code to run with exec.

    Source code in prefect/packaging/serializers.py
    class SourceSerializer(Serializer):\n    \"\"\"\n    Serializes objects by retrieving the source code of the module they are defined in.\n\n    Creates a JSON blob with keys:\n        source: The source code\n        file_name: The name of the file the source was in\n        symbol_name: The name of the object to extract from the source code\n\n    Deserialization requires the code to run with `exec`.\n    \"\"\"\n\n    type: Literal[\"source\"] = \"source\"\n\n    def dumps(self, obj: Any) -> bytes:\n        module = inspect.getmodule(obj)\n\n        if module is None:\n            raise ValueError(f\"Cannot determine source module for object: {obj!r}.\")\n\n        if not getattr(module, \"__file__\", None):\n            raise ValueError(\n                f\"Found module {module!r} without source code file while serializing \"\n                f\"object: {obj!r}.\"\n            )\n\n        source = inspect.getsource(module)\n\n        return json.dumps(\n            {\n                \"source\": source,\n                \"file_name\": os.path.basename(module.__file__),\n                \"symbol_name\": obj.__name__,\n            }\n        ).encode()\n\n    def loads(self, blob: bytes) -> Any:\n        document = json.loads(blob)\n        if not isinstance(document, dict) or set(document.keys()) != {\n            \"source\",\n            \"file_name\",\n            \"symbol_name\",\n        }:\n            raise ValueError(\n                \"Invalid serialized data. \"\n                \"Expected dictionary with keys 'source', 'file_name', and \"\n                \"'symbol_name'. \"\n                f\"Got: {document}\"\n            )\n\n        with TemporaryDirectory() as tmpdir:\n            temp_script = Path(tmpdir) / document[\"file_name\"]\n            temp_script.write_text(document[\"source\"])\n            module = load_script_as_module(str(temp_script))\n\n        return getattr(module, document[\"symbol_name\"])\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/runner/runner/","title":"runner","text":"","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner","title":"prefect.runner.runner","text":"

    Runners are responsible for managing the execution of deployments created and managed by either flow.serve or the serve utility.

    Example
    import time\nfrom prefect import flow, serve\n\n\n@flow\ndef slow_flow(sleep: int = 60):\n    \"Sleepy flow - sleeps the provided amount of time (in seconds).\"\n    time.sleep(sleep)\n\n\n@flow\ndef fast_flow():\n    \"Fastest flow this side of the Mississippi.\"\n    return\n\n\nif __name__ == \"__main__\":\n    slow_deploy = slow_flow.to_deployment(name=\"sleeper\", interval=45)\n    fast_deploy = fast_flow.to_deployment(name=\"fast\")\n\n    # serve generates a Runner instance\n    serve(slow_deploy, fast_deploy)\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner","title":"Runner","text":"Source code in prefect/runner/runner.py
    class Runner:\n    def __init__(\n        self,\n        name: Optional[str] = None,\n        query_seconds: Optional[float] = None,\n        prefetch_seconds: float = 10,\n        limit: Optional[int] = None,\n        pause_on_shutdown: bool = True,\n        webserver: bool = False,\n    ):\n        \"\"\"\n        Responsible for managing the execution of remotely initiated flow runs.\n\n        Args:\n            name: The name of the runner. If not provided, a random one\n                will be generated. If provided, it cannot contain '/' or '%'.\n            query_seconds: The number of seconds to wait between querying for\n                scheduled flow runs; defaults to `PREFECT_RUNNER_POLL_FREQUENCY`\n            prefetch_seconds: The number of seconds to prefetch flow runs for.\n            limit: The maximum number of flow runs this runner should be running at\n            pause_on_shutdown: A boolean for whether or not to automatically pause\n                deployment schedules on shutdown; defaults to `True`\n            webserver: a boolean flag for whether to start a webserver for this runner\n\n        Examples:\n            Set up a Runner to manage the execute of scheduled flow runs for two flows:\n                ```python\n                from prefect import flow, Runner\n\n                @flow\n                def hello_flow(name):\n                    print(f\"hello {name}\")\n\n                @flow\n                def goodbye_flow(name):\n                    print(f\"goodbye {name}\")\n\n                if __name__ == \"__main__\"\n                    runner = Runner(name=\"my-runner\")\n\n                    # Will be runnable via the API\n                    runner.add_flow(hello_flow)\n\n                    # Run on a cron schedule\n                    runner.add_flow(goodbye_flow, schedule={\"cron\": \"0 * * * *\"})\n\n                    runner.start()\n                ```\n        \"\"\"\n        if name and (\"/\" in name or \"%\" in name):\n            raise ValueError(\"Runner name cannot contain '/' or '%'\")\n        self.name = Path(name).stem if name is not None else f\"runner-{uuid4()}\"\n        self._logger = get_logger(\"runner\")\n\n        self.started = False\n        self.stopping = False\n        self.pause_on_shutdown = pause_on_shutdown\n        self.limit = limit or PREFECT_RUNNER_PROCESS_LIMIT.value()\n        self.webserver = webserver\n\n        self.query_seconds = query_seconds or PREFECT_RUNNER_POLL_FREQUENCY.value()\n        self._prefetch_seconds = prefetch_seconds\n\n        self._runs_task_group: anyio.abc.TaskGroup = anyio.create_task_group()\n        self._loops_task_group: anyio.abc.TaskGroup = anyio.create_task_group()\n\n        self._limiter: Optional[anyio.CapacityLimiter] = anyio.CapacityLimiter(\n            self.limit\n        )\n        self._client = get_client()\n        self._submitting_flow_run_ids = set()\n        self._cancelling_flow_run_ids = set()\n        self._scheduled_task_scopes = set()\n        self._deployment_ids: Set[UUID] = set()\n        self._flow_run_process_map = dict()\n\n        self._tmp_dir: Path = (\n            Path(tempfile.gettempdir()) / \"runner_storage\" / str(uuid4())\n        )\n        self._storage_objs: List[RunnerStorage] = []\n        self._deployment_storage_map: Dict[UUID, RunnerStorage] = {}\n\n    @sync_compatible\n    async def add_deployment(\n        self,\n        deployment: RunnerDeployment,\n    ) -> UUID:\n        \"\"\"\n        Registers the deployment with the Prefect API and will monitor for work once\n        the runner is started.\n\n        Args:\n            deployment: A deployment for the runner to register.\n        \"\"\"\n        deployment_id = await deployment.apply()\n        storage = deployment.storage\n        if storage is not None:\n            storage = await self._add_storage(storage)\n            self._deployment_storage_map[deployment_id] = storage\n        self._deployment_ids.add(deployment_id)\n\n        return deployment_id\n\n    @sync_compatible\n    async def add_flow(\n        self,\n        flow: Flow,\n        name: str = None,\n        interval: Optional[Union[int, float, datetime.timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n    ) -> UUID:\n        \"\"\"\n        Provides a flow to the runner to be run based on the provided configuration.\n\n        Will create a deployment for the provided flow and register the deployment\n        with the runner.\n\n        Args:\n            flow: A flow for the runner to run.\n            name: The name to give the created deployment. Will default to the name\n                of the runner.\n            interval: An interval on which to execute the current flow. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this flow.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            triggers: A list of triggers that should kick of a run of this flow.\n            parameters: A dictionary of default parameter values to pass to runs of this flow.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n        \"\"\"\n        api = PREFECT_API_URL.value()\n        if any([interval, cron, rrule]) and not api:\n            self._logger.warning(\n                \"Cannot schedule flows on an ephemeral server; run `prefect server\"\n                \" start` to start the scheduler.\"\n            )\n        name = self.name if name is None else name\n\n        deployment = await flow.to_deployment(\n            name=name,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            triggers=triggers,\n            parameters=parameters,\n            description=description,\n            tags=tags,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n        )\n        return await self.add_deployment(deployment)\n\n    @sync_compatible\n    async def _add_storage(self, storage: RunnerStorage) -> RunnerStorage:\n        \"\"\"\n        Adds a storage object to the runner. The storage object will be used to pull\n        code to the runner's working directory before the runner starts.\n\n        Args:\n            storage: The storage object to add to the runner.\n        Returns:\n            The updated storage object that was added to the runner.\n        \"\"\"\n        if storage not in self._storage_objs:\n            storage_copy = deepcopy(storage)\n            storage_copy.set_base_path(self._tmp_dir)\n\n            self._logger.debug(\n                f\"Adding storage {storage_copy!r} to runner at\"\n                f\" {str(storage_copy.destination)!r}\"\n            )\n            self._storage_objs.append(storage_copy)\n\n            return storage_copy\n        else:\n            return next(s for s in self._storage_objs if s == storage)\n\n    def handle_sigterm(self, signum, frame):\n        \"\"\"\n        Gracefully shuts down the runner when a SIGTERM is received.\n        \"\"\"\n        self._logger.info(\"SIGTERM received, initiating graceful shutdown...\")\n        from_sync.call_in_loop_thread(create_call(self.stop))\n\n        sys.exit(0)\n\n    @sync_compatible\n    async def start(\n        self, run_once: bool = False, webserver: Optional[bool] = None\n    ) -> None:\n        \"\"\"\n        Starts a runner.\n\n        The runner will begin monitoring for and executing any scheduled work for all added flows.\n\n        Args:\n            run_once: If True, the runner will through one query loop and then exit.\n            webserver: a boolean for whether to start a webserver for this runner. If provided,\n                overrides the default on the runner\n\n        Examples:\n            Initialize a Runner, add two flows, and serve them by starting the Runner:\n\n            ```python\n            from prefect import flow, Runner\n\n            @flow\n            def hello_flow(name):\n                print(f\"hello {name}\")\n\n            @flow\n            def goodbye_flow(name):\n                print(f\"goodbye {name}\")\n\n            if __name__ == \"__main__\"\n                runner = Runner(name=\"my-runner\")\n\n                # Will be runnable via the API\n                runner.add_flow(hello_flow)\n\n                # Run on a cron schedule\n                runner.add_flow(goodbye_flow, schedule={\"cron\": \"0 * * * *\"})\n\n                runner.start()\n            ```\n        \"\"\"\n        _register_signal(signal.SIGTERM, self.handle_sigterm)\n\n        webserver = webserver if webserver is not None else self.webserver\n\n        if webserver:\n            # we'll start the ASGI server in a separate thread so that\n            # uvicorn does not block the main thread\n            server_thread = threading.Thread(\n                name=\"runner-server-thread\",\n                target=partial(\n                    start_webserver,\n                    runner=self,\n                ),\n                daemon=True,\n            )\n            server_thread.start()\n\n        async with self as runner:\n            async with self._loops_task_group as tg:\n                for storage in self._storage_objs:\n                    if storage.pull_interval:\n                        tg.start_soon(\n                            partial(\n                                critical_service_loop,\n                                workload=storage.pull_code,\n                                interval=storage.pull_interval,\n                                run_once=run_once,\n                                jitter_range=0.3,\n                            )\n                        )\n                    else:\n                        tg.start_soon(storage.pull_code)\n                tg.start_soon(\n                    partial(\n                        critical_service_loop,\n                        workload=runner._get_and_submit_flow_runs,\n                        interval=self.query_seconds,\n                        run_once=run_once,\n                        jitter_range=0.3,\n                    )\n                )\n                tg.start_soon(\n                    partial(\n                        critical_service_loop,\n                        workload=runner._check_for_cancelled_flow_runs,\n                        interval=self.query_seconds * 2,\n                        run_once=run_once,\n                        jitter_range=0.3,\n                    )\n                )\n\n    async def cancel_all(self):\n        runs_to_cancel = []\n\n        # done to avoid dictionary size changing during iteration\n        for flow_run_id, info in self._flow_run_process_map.items():\n            runs_to_cancel.append(info[\"flow_run\"])\n        if runs_to_cancel:\n            for run in runs_to_cancel:\n                try:\n                    await self._cancel_run(run, state_msg=\"Runner is shutting down.\")\n                except Exception:\n                    self._logger.exception(\n                        f\"Exception encountered while cancelling {run.id}\",\n                        exc_info=True,\n                    )\n\n    @sync_compatible\n    async def stop(self):\n        \"\"\"Stops the runner's polling cycle.\"\"\"\n        if not self.started:\n            raise RuntimeError(\n                \"Runner has not yet started. Please start the runner by calling\"\n                \" .start()\"\n            )\n\n        self.started = False\n        self.stopping = True\n        await self.cancel_all()\n        try:\n            self._loops_task_group.cancel_scope.cancel()\n        except Exception:\n            self._logger.exception(\n                \"Exception encountered while shutting down\", exc_info=True\n            )\n\n    async def execute_flow_run(self, flow_run_id: UUID):\n        \"\"\"\n        Executes a single flow run with the given ID.\n\n        Execution will wait to monitor for cancellation requests. Exits once\n        the flow run process has exited.\n        \"\"\"\n        self.pause_on_shutdown = False\n        async with self:\n            if not self._acquire_limit_slot(flow_run_id):\n                return\n\n            async with anyio.create_task_group() as tg:\n                with anyio.CancelScope():\n                    self._submitting_flow_run_ids.add(flow_run_id)\n                    flow_run = await self._client.read_flow_run(flow_run_id)\n\n                    pid = await self._runs_task_group.start(\n                        self._submit_run_and_capture_errors, flow_run\n                    )\n\n                    self._flow_run_process_map[flow_run.id] = dict(\n                        pid=pid, flow_run=flow_run\n                    )\n\n                    # We want this loop to stop when the flow run process exits\n                    # so we'll check if the flow run process is still alive on\n                    # each iteration and cancel the task group if it is not.\n                    workload = partial(\n                        self._check_for_cancelled_flow_runs,\n                        should_stop=lambda: not self._flow_run_process_map,\n                        on_stop=tg.cancel_scope.cancel,\n                    )\n\n                    tg.start_soon(\n                        partial(\n                            critical_service_loop,\n                            workload=workload,\n                            interval=self.query_seconds,\n                            jitter_range=0.3,\n                        )\n                    )\n\n    def _get_flow_run_logger(self, flow_run: \"FlowRun\") -> PrefectLogAdapter:\n        return flow_run_logger(flow_run=flow_run).getChild(\n            \"runner\",\n            extra={\n                \"runner_name\": self.name,\n            },\n        )\n\n    async def _run_process(\n        self,\n        flow_run: \"FlowRun\",\n        task_status: Optional[anyio.abc.TaskStatus] = None,\n    ):\n        \"\"\"\n        Runs the given flow run in a subprocess.\n\n        Args:\n            flow_run: Flow run to execute via process. The ID of this flow run\n                is stored in the PREFECT__FLOW_RUN_ID environment variable to\n                allow the engine to retrieve the corresponding flow's code and\n                begin execution.\n            task_status: anyio task status used to send a message to the caller\n                than the flow run process has started.\n        \"\"\"\n        command = f\"{shlex.quote(sys.executable)} -m prefect.engine\"\n\n        flow_run_logger = self._get_flow_run_logger(flow_run)\n\n        # We must add creationflags to a dict so it is only passed as a function\n        # parameter on Windows, because the presence of creationflags causes\n        # errors on Unix even if set to None\n        kwargs: Dict[str, object] = {}\n        if sys.platform == \"win32\":\n            kwargs[\"creationflags\"] = subprocess.CREATE_NEW_PROCESS_GROUP\n\n        _use_threaded_child_watcher()\n        flow_run_logger.info(\"Opening process...\")\n\n        env = get_current_settings().to_environment_variables(exclude_unset=True)\n        env.update(\n            {\n                \"PREFECT__FLOW_RUN_ID\": str(flow_run.id),\n                \"PREFECT__STORAGE_BASE_PATH\": str(self._tmp_dir),\n                \"PREFECT__ENABLE_CANCELLATION_AND_CRASHED_HOOKS\": \"false\",\n            }\n        )\n        env.update(**os.environ)  # is this really necessary??\n\n        storage = self._deployment_storage_map.get(flow_run.deployment_id)\n        if storage and storage.pull_interval:\n            # perform an adhoc pull of code before running the flow if an\n            # adhoc pull hasn't been performed in the last pull_interval\n            # TODO: Explore integrating this behavior with global concurrency.\n            last_adhoc_pull = getattr(storage, \"last_adhoc_pull\", None)\n            if (\n                last_adhoc_pull is None\n                or last_adhoc_pull\n                < datetime.datetime.now()\n                - datetime.timedelta(seconds=storage.pull_interval)\n            ):\n                self._logger.debug(\n                    \"Performing adhoc pull of code for flow run %s with storage %r\",\n                    flow_run.id,\n                    storage,\n                )\n                await storage.pull_code()\n                setattr(storage, \"last_adhoc_pull\", datetime.datetime.now())\n\n        process = await run_process(\n            shlex.split(command),\n            stream_output=True,\n            task_status=task_status,\n            env=env,\n            **kwargs,\n            cwd=storage.destination if storage else None,\n        )\n\n        # Use the pid for display if no name was given\n\n        if process.returncode:\n            help_message = None\n            level = logging.ERROR\n            if process.returncode == -9:\n                level = logging.INFO\n                help_message = (\n                    \"This indicates that the process exited due to a SIGKILL signal. \"\n                    \"Typically, this is either caused by manual cancellation or \"\n                    \"high memory usage causing the operating system to \"\n                    \"terminate the process.\"\n                )\n            if process.returncode == -15:\n                level = logging.INFO\n                help_message = (\n                    \"This indicates that the process exited due to a SIGTERM signal. \"\n                    \"Typically, this is caused by manual cancellation.\"\n                )\n            elif process.returncode == 247:\n                help_message = (\n                    \"This indicates that the process was terminated due to high \"\n                    \"memory usage.\"\n                )\n            elif (\n                sys.platform == \"win32\" and process.returncode == STATUS_CONTROL_C_EXIT\n            ):\n                level = logging.INFO\n                help_message = (\n                    \"Process was terminated due to a Ctrl+C or Ctrl+Break signal. \"\n                    \"Typically, this is caused by manual cancellation.\"\n                )\n\n            flow_run_logger.log(\n                level,\n                f\"Process for flow run {flow_run.name!r} exited with status code:\"\n                f\" {process.returncode}\"\n                + (f\"; {help_message}\" if help_message else \"\"),\n            )\n        else:\n            flow_run_logger.info(\n                f\"Process for flow run {flow_run.name!r} exited cleanly.\"\n            )\n\n        return process.returncode\n\n    async def _kill_process(\n        self,\n        pid: int,\n        grace_seconds: int = 30,\n    ):\n        \"\"\"\n        Kills a given flow run process.\n\n        Args:\n            pid: ID of the process to kill\n            grace_seconds: Number of seconds to wait for the process to end.\n        \"\"\"\n        # In a non-windows environment first send a SIGTERM, then, after\n        # `grace_seconds` seconds have passed subsequent send SIGKILL. In\n        # Windows we use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        if sys.platform == \"win32\":\n            try:\n                os.kill(pid, signal.CTRL_BREAK_EVENT)\n            except (ProcessLookupError, WindowsError):\n                raise RuntimeError(\n                    f\"Unable to kill process {pid!r}: The process was not found.\"\n                )\n        else:\n            try:\n                os.kill(pid, signal.SIGTERM)\n            except ProcessLookupError:\n                raise RuntimeError(\n                    f\"Unable to kill process {pid!r}: The process was not found.\"\n                )\n\n            # Throttle how often we check if the process is still alive to keep\n            # from making too many system calls in a short period of time.\n            check_interval = max(grace_seconds / 10, 1)\n\n            with anyio.move_on_after(grace_seconds):\n                while True:\n                    await anyio.sleep(check_interval)\n\n                    # Detect if the process is still alive. If not do an early\n                    # return as the process respected the SIGTERM from above.\n                    try:\n                        os.kill(pid, 0)\n                    except ProcessLookupError:\n                        return\n\n            try:\n                os.kill(pid, signal.SIGKILL)\n            except OSError:\n                # We shouldn't ever end up here, but it's possible that the\n                # process ended right after the check above.\n                return\n\n    async def _pause_schedules(self):\n        \"\"\"\n        Pauses all deployment schedules.\n        \"\"\"\n        self._logger.info(\"Pausing schedules for all deployments...\")\n        for deployment_id in self._deployment_ids:\n            self._logger.debug(f\"Pausing schedule for deployment '{deployment_id}'\")\n            await self._client.update_schedule(deployment_id, active=False)\n        self._logger.info(\"All deployment schedules have been paused!\")\n\n    async def _get_and_submit_flow_runs(self):\n        if self.stopping:\n            return\n        runs_response = await self._get_scheduled_flow_runs()\n        self.last_polled = pendulum.now(\"UTC\")\n        return await self._submit_scheduled_flow_runs(flow_run_response=runs_response)\n\n    async def _check_for_cancelled_flow_runs(\n        self, should_stop: Callable = lambda: False, on_stop: Callable = lambda: None\n    ):\n        \"\"\"\n        Checks for flow runs with CANCELLING a cancelling state and attempts to\n        cancel them.\n\n        Args:\n            should_stop: A callable that returns a boolean indicating whether or not\n                the runner should stop checking for cancelled flow runs.\n            on_stop: A callable that is called when the runner should stop checking\n                for cancelled flow runs.\n        \"\"\"\n        if self.stopping:\n            return\n        if not self.started:\n            raise RuntimeError(\n                \"Runner is not set up. Please make sure you are running this runner \"\n                \"as an async context manager.\"\n            )\n\n        if should_stop():\n            self._logger.debug(\n                \"Runner has no active flow runs or deployments. Sending message to loop\"\n                \" service that no further cancellation checks are needed.\"\n            )\n            on_stop()\n\n        self._logger.debug(\"Checking for cancelled flow runs...\")\n\n        named_cancelling_flow_runs = await self._client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLED]),\n                    name=FlowRunFilterStateName(any_=[\"Cancelling\"]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(\n                    any_=list(\n                        self._flow_run_process_map.keys()\n                        - self._cancelling_flow_run_ids\n                    )\n                ),\n            ),\n        )\n\n        typed_cancelling_flow_runs = await self._client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLING]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(\n                    any_=list(\n                        self._flow_run_process_map.keys()\n                        - self._cancelling_flow_run_ids\n                    )\n                ),\n            ),\n        )\n\n        cancelling_flow_runs = named_cancelling_flow_runs + typed_cancelling_flow_runs\n\n        if cancelling_flow_runs:\n            self._logger.info(\n                f\"Found {len(cancelling_flow_runs)} flow runs awaiting cancellation.\"\n            )\n\n        for flow_run in cancelling_flow_runs:\n            self._cancelling_flow_run_ids.add(flow_run.id)\n            self._runs_task_group.start_soon(self._cancel_run, flow_run)\n\n        return cancelling_flow_runs\n\n    async def _cancel_run(self, flow_run: \"FlowRun\", state_msg: Optional[str] = None):\n        run_logger = self._get_flow_run_logger(flow_run)\n\n        pid = self._flow_run_process_map.get(flow_run.id, {}).get(\"pid\")\n        if not pid:\n            await self._run_on_cancellation_hooks(flow_run, flow_run.state)\n            await self._mark_flow_run_as_cancelled(\n                flow_run,\n                state_updates={\n                    \"message\": (\n                        \"Could not find process ID for flow run\"\n                        \" and cancellation cannot be guaranteed.\"\n                    )\n                },\n            )\n            return\n\n        try:\n            await self._kill_process(pid)\n        except RuntimeError as exc:\n            self._logger.warning(f\"{exc} Marking flow run as cancelled.\")\n            await self._run_on_cancellation_hooks(flow_run, flow_run.state)\n            await self._mark_flow_run_as_cancelled(flow_run)\n        except Exception:\n            run_logger.exception(\n                \"Encountered exception while killing process for flow run \"\n                f\"'{flow_run.id}'. Flow run may not be cancelled.\"\n            )\n            # We will try again on generic exceptions\n            self._cancelling_flow_run_ids.remove(flow_run.id)\n        else:\n            await self._run_on_cancellation_hooks(flow_run, flow_run.state)\n            await self._mark_flow_run_as_cancelled(\n                flow_run,\n                state_updates={\n                    \"message\": state_msg or \"Flow run was cancelled successfully.\"\n                },\n            )\n            run_logger.info(f\"Cancelled flow run '{flow_run.name}'!\")\n\n    async def _get_scheduled_flow_runs(\n        self,\n    ) -> List[\"FlowRun\"]:\n        \"\"\"\n        Retrieve scheduled flow runs for this runner.\n        \"\"\"\n        scheduled_before = pendulum.now(\"utc\").add(seconds=int(self._prefetch_seconds))\n        self._logger.debug(\n            f\"Querying for flow runs scheduled before {scheduled_before}\"\n        )\n\n        scheduled_flow_runs = (\n            await self._client.get_scheduled_flow_runs_for_deployments(\n                deployment_ids=list(self._deployment_ids),\n                scheduled_before=scheduled_before,\n            )\n        )\n        self._logger.debug(f\"Discovered {len(scheduled_flow_runs)} scheduled_flow_runs\")\n        return scheduled_flow_runs\n\n    def _acquire_limit_slot(self, flow_run_id: str) -> bool:\n        \"\"\"\n        Enforces flow run limit set on runner.\n\n        Returns:\n            - bool: True if a slot was acquired, False otherwise.\n        \"\"\"\n        try:\n            if self._limiter:\n                self._limiter.acquire_on_behalf_of_nowait(flow_run_id)\n                self._logger.debug(\"Limit slot acquired for flow run '%s'\", flow_run_id)\n            return True\n        except RuntimeError as exc:\n            if (\n                \"this borrower is already holding one of this CapacityLimiter's tokens\"\n                in str(exc)\n            ):\n                self._logger.warning(\n                    f\"Duplicate submission of flow run '{flow_run_id}' detected. Runner\"\n                    \" will not re-submit flow run.\"\n                )\n                return False\n            else:\n                raise\n        except anyio.WouldBlock:\n            self._logger.info(\n                f\"Flow run limit reached; {self._limiter.borrowed_tokens} flow runs\"\n                \" in progress. You can control this limit by adjusting the\"\n                \" PREFECT_RUNNER_PROCESS_LIMIT setting.\"\n            )\n            return False\n\n    def _release_limit_slot(self, flow_run_id: str) -> None:\n        \"\"\"\n        Frees up a slot taken by the given flow run id.\n        \"\"\"\n        if self._limiter:\n            self._limiter.release_on_behalf_of(flow_run_id)\n            self._logger.debug(\"Limit slot released for flow run '%s'\", flow_run_id)\n\n    async def _submit_scheduled_flow_runs(\n        self, flow_run_response: List[\"FlowRun\"]\n    ) -> List[\"FlowRun\"]:\n        \"\"\"\n        Takes a list of FlowRuns and submits the referenced flow runs\n        for execution by the runner.\n        \"\"\"\n        submittable_flow_runs = flow_run_response\n        submittable_flow_runs.sort(key=lambda run: run.next_scheduled_start_time)\n        for flow_run in submittable_flow_runs:\n            if flow_run.id in self._submitting_flow_run_ids:\n                continue\n\n            if self._acquire_limit_slot(flow_run.id):\n                run_logger = self._get_flow_run_logger(flow_run)\n                run_logger.info(\n                    f\"Runner '{self.name}' submitting flow run '{flow_run.id}'\"\n                )\n                self._submitting_flow_run_ids.add(flow_run.id)\n                self._runs_task_group.start_soon(\n                    self._submit_run,\n                    flow_run,\n                )\n            else:\n                break\n\n        return list(\n            filter(\n                lambda run: run.id in self._submitting_flow_run_ids,\n                submittable_flow_runs,\n            )\n        )\n\n    async def _submit_run(self, flow_run: \"FlowRun\") -> None:\n        \"\"\"\n        Submits a given flow run for execution by the runner.\n        \"\"\"\n        run_logger = self._get_flow_run_logger(flow_run)\n\n        ready_to_submit = await self._propose_pending_state(flow_run)\n\n        if ready_to_submit:\n            readiness_result = await self._runs_task_group.start(\n                self._submit_run_and_capture_errors, flow_run\n            )\n\n            if readiness_result and not isinstance(readiness_result, Exception):\n                self._flow_run_process_map[flow_run.id] = dict(\n                    pid=readiness_result, flow_run=flow_run\n                )\n\n            run_logger.info(f\"Completed submission of flow run '{flow_run.id}'\")\n        else:\n            # If the run is not ready to submit, release the concurrency slot\n            self._release_limit_slot(flow_run.id)\n\n        self._submitting_flow_run_ids.remove(flow_run.id)\n\n    async def _submit_run_and_capture_errors(\n        self, flow_run: \"FlowRun\", task_status: anyio.abc.TaskStatus = None\n    ) -> Union[Optional[int], Exception]:\n        run_logger = self._get_flow_run_logger(flow_run)\n\n        try:\n            status_code = await self._run_process(\n                flow_run=flow_run,\n                task_status=task_status,\n            )\n        except Exception as exc:\n            if not task_status._future.done():\n                # This flow run was being submitted and did not start successfully\n                run_logger.exception(\n                    f\"Failed to start process for flow run '{flow_run.id}'.\"\n                )\n                # Mark the task as started to prevent agent crash\n                task_status.started(exc)\n                await self._propose_crashed_state(\n                    flow_run, \"Flow run process could not be started\"\n                )\n            else:\n                run_logger.exception(\n                    f\"An error occurred while monitoring flow run '{flow_run.id}'. \"\n                    \"The flow run will not be marked as failed, but an issue may have \"\n                    \"occurred.\"\n                )\n            return exc\n        finally:\n            self._release_limit_slot(flow_run.id)\n            self._flow_run_process_map.pop(flow_run.id, None)\n\n        if status_code != 0:\n            await self._propose_crashed_state(\n                flow_run,\n                f\"Flow run process exited with non-zero status code {status_code}.\",\n            )\n\n        api_flow_run = await self._client.read_flow_run(flow_run_id=flow_run.id)\n        terminal_state = api_flow_run.state\n        if terminal_state.is_crashed():\n            await self._run_on_crashed_hooks(flow_run=flow_run, state=terminal_state)\n\n        return status_code\n\n    async def _propose_pending_state(self, flow_run: \"FlowRun\") -> bool:\n        run_logger = self._get_flow_run_logger(flow_run)\n        state = flow_run.state\n        try:\n            state = await propose_state(\n                self._client, Pending(), flow_run_id=flow_run.id\n            )\n        except Abort as exc:\n            run_logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}'. \"\n                    f\"Server sent an abort signal: {exc}\"\n                ),\n            )\n            return False\n        except Exception:\n            run_logger.exception(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n            )\n            return False\n\n        if not state.is_pending():\n            run_logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}': \"\n                    f\"Server returned a non-pending state {state.type.value!r}\"\n                ),\n            )\n            return False\n\n        return True\n\n    async def _propose_failed_state(self, flow_run: \"FlowRun\", exc: Exception) -> None:\n        run_logger = self._get_flow_run_logger(flow_run)\n        try:\n            await propose_state(\n                self._client,\n                await exception_to_failed_state(message=\"Submission failed.\", exc=exc),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # We've already failed, no need to note the abort but we don't want it to\n            # raise in the agent process\n            pass\n        except Exception:\n            run_logger.error(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n                exc_info=True,\n            )\n\n    async def _propose_crashed_state(self, flow_run: \"FlowRun\", message: str) -> None:\n        run_logger = self._get_flow_run_logger(flow_run)\n        try:\n            state = await propose_state(\n                self._client,\n                Crashed(message=message),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # Flow run already marked as failed\n            pass\n        except Exception:\n            run_logger.exception(f\"Failed to update state of flow run '{flow_run.id}'\")\n        else:\n            if state.is_crashed():\n                run_logger.info(\n                    f\"Reported flow run '{flow_run.id}' as crashed: {message}\"\n                )\n\n    async def _mark_flow_run_as_cancelled(\n        self, flow_run: \"FlowRun\", state_updates: Optional[dict] = None\n    ) -> None:\n        state_updates = state_updates or {}\n        state_updates.setdefault(\"name\", \"Cancelled\")\n        state_updates.setdefault(\"type\", StateType.CANCELLED)\n        state = flow_run.state.copy(update=state_updates)\n\n        await self._client.set_flow_run_state(flow_run.id, state, force=True)\n\n        # Do not remove the flow run from the cancelling set immediately because\n        # the API caches responses for the `read_flow_runs` and we do not want to\n        # duplicate cancellations.\n        await self._schedule_task(\n            60 * 10, self._cancelling_flow_run_ids.remove, flow_run.id\n        )\n\n    async def _schedule_task(self, __in_seconds: int, fn, *args, **kwargs):\n        \"\"\"\n        Schedule a background task to start after some time.\n\n        These tasks will be run immediately when the runner exits instead of waiting.\n\n        The function may be async or sync. Async functions will be awaited.\n        \"\"\"\n\n        async def wrapper(task_status):\n            # If we are shutting down, do not sleep; otherwise sleep until the scheduled\n            # time or shutdown\n            if self.started:\n                with anyio.CancelScope() as scope:\n                    self._scheduled_task_scopes.add(scope)\n                    task_status.started()\n                    await anyio.sleep(__in_seconds)\n\n                self._scheduled_task_scopes.remove(scope)\n            else:\n                task_status.started()\n\n            result = fn(*args, **kwargs)\n            if inspect.iscoroutine(result):\n                await result\n\n        await self._runs_task_group.start(wrapper)\n\n    async def _run_on_cancellation_hooks(\n        self,\n        flow_run: \"FlowRun\",\n        state: State,\n    ) -> None:\n        \"\"\"\n        Run the hooks for a flow.\n        \"\"\"\n        if state.is_cancelling():\n            flow = await load_flow_from_flow_run(\n                flow_run, client=self._client, storage_base_path=str(self._tmp_dir)\n            )\n            hooks = flow.on_cancellation or []\n\n            await _run_hooks(hooks, flow_run, flow, state)\n\n    async def _run_on_crashed_hooks(\n        self,\n        flow_run: \"FlowRun\",\n        state: State,\n    ) -> None:\n        \"\"\"\n        Run the hooks for a flow.\n        \"\"\"\n        if state.is_crashed():\n            flow = await load_flow_from_flow_run(\n                flow_run, client=self._client, storage_base_path=str(self._tmp_dir)\n            )\n            hooks = flow.on_crashed or []\n\n            await _run_hooks(hooks, flow_run, flow, state)\n\n    async def __aenter__(self):\n        self._logger.debug(\"Starting runner...\")\n        self._client = get_client()\n        self._tmp_dir.mkdir(parents=True)\n        await self._client.__aenter__()\n        await self._runs_task_group.__aenter__()\n\n        self.started = True\n        return self\n\n    async def __aexit__(self, *exc_info):\n        self._logger.debug(\"Stopping runner...\")\n        if self.pause_on_shutdown:\n            await self._pause_schedules()\n        self.started = False\n        for scope in self._scheduled_task_scopes:\n            scope.cancel()\n        if self._runs_task_group:\n            await self._runs_task_group.__aexit__(*exc_info)\n        if self._client:\n            await self._client.__aexit__(*exc_info)\n        shutil.rmtree(str(self._tmp_dir))\n\n    def __repr__(self):\n        return f\"Runner(name={self.name!r})\"\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.add_deployment","title":"add_deployment async","text":"

    Registers the deployment with the Prefect API and will monitor for work once the runner is started.

    Parameters:

    Name Type Description Default deployment RunnerDeployment

    A deployment for the runner to register.

    required Source code in prefect/runner/runner.py
    @sync_compatible\nasync def add_deployment(\n    self,\n    deployment: RunnerDeployment,\n) -> UUID:\n    \"\"\"\n    Registers the deployment with the Prefect API and will monitor for work once\n    the runner is started.\n\n    Args:\n        deployment: A deployment for the runner to register.\n    \"\"\"\n    deployment_id = await deployment.apply()\n    storage = deployment.storage\n    if storage is not None:\n        storage = await self._add_storage(storage)\n        self._deployment_storage_map[deployment_id] = storage\n    self._deployment_ids.add(deployment_id)\n\n    return deployment_id\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.add_flow","title":"add_flow async","text":"

    Provides a flow to the runner to be run based on the provided configuration.

    Will create a deployment for the provided flow and register the deployment with the runner.

    Parameters:

    Name Type Description Default flow Flow

    A flow for the runner to run.

    required name str

    The name to give the created deployment. Will default to the name of the runner.

    None interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the current flow. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this flow.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object of when to execute runs of this flow. Used for advanced scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that should kick of a run of this flow.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this flow.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None Source code in prefect/runner/runner.py
    @sync_compatible\nasync def add_flow(\n    self,\n    flow: Flow,\n    name: str = None,\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n) -> UUID:\n    \"\"\"\n    Provides a flow to the runner to be run based on the provided configuration.\n\n    Will create a deployment for the provided flow and register the deployment\n    with the runner.\n\n    Args:\n        flow: A flow for the runner to run.\n        name: The name to give the created deployment. Will default to the name\n            of the runner.\n        interval: An interval on which to execute the current flow. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this flow.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        schedule: A schedule object of when to execute runs of this flow. Used for\n            advanced scheduling options like timezone.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        triggers: A list of triggers that should kick of a run of this flow.\n        parameters: A dictionary of default parameter values to pass to runs of this flow.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n    \"\"\"\n    api = PREFECT_API_URL.value()\n    if any([interval, cron, rrule]) and not api:\n        self._logger.warning(\n            \"Cannot schedule flows on an ephemeral server; run `prefect server\"\n            \" start` to start the scheduler.\"\n        )\n    name = self.name if name is None else name\n\n    deployment = await flow.to_deployment(\n        name=name,\n        interval=interval,\n        cron=cron,\n        rrule=rrule,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        triggers=triggers,\n        parameters=parameters,\n        description=description,\n        tags=tags,\n        version=version,\n        enforce_parameter_schema=enforce_parameter_schema,\n    )\n    return await self.add_deployment(deployment)\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.execute_flow_run","title":"execute_flow_run async","text":"

    Executes a single flow run with the given ID.

    Execution will wait to monitor for cancellation requests. Exits once the flow run process has exited.

    Source code in prefect/runner/runner.py
    async def execute_flow_run(self, flow_run_id: UUID):\n    \"\"\"\n    Executes a single flow run with the given ID.\n\n    Execution will wait to monitor for cancellation requests. Exits once\n    the flow run process has exited.\n    \"\"\"\n    self.pause_on_shutdown = False\n    async with self:\n        if not self._acquire_limit_slot(flow_run_id):\n            return\n\n        async with anyio.create_task_group() as tg:\n            with anyio.CancelScope():\n                self._submitting_flow_run_ids.add(flow_run_id)\n                flow_run = await self._client.read_flow_run(flow_run_id)\n\n                pid = await self._runs_task_group.start(\n                    self._submit_run_and_capture_errors, flow_run\n                )\n\n                self._flow_run_process_map[flow_run.id] = dict(\n                    pid=pid, flow_run=flow_run\n                )\n\n                # We want this loop to stop when the flow run process exits\n                # so we'll check if the flow run process is still alive on\n                # each iteration and cancel the task group if it is not.\n                workload = partial(\n                    self._check_for_cancelled_flow_runs,\n                    should_stop=lambda: not self._flow_run_process_map,\n                    on_stop=tg.cancel_scope.cancel,\n                )\n\n                tg.start_soon(\n                    partial(\n                        critical_service_loop,\n                        workload=workload,\n                        interval=self.query_seconds,\n                        jitter_range=0.3,\n                    )\n                )\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.handle_sigterm","title":"handle_sigterm","text":"

    Gracefully shuts down the runner when a SIGTERM is received.

    Source code in prefect/runner/runner.py
    def handle_sigterm(self, signum, frame):\n    \"\"\"\n    Gracefully shuts down the runner when a SIGTERM is received.\n    \"\"\"\n    self._logger.info(\"SIGTERM received, initiating graceful shutdown...\")\n    from_sync.call_in_loop_thread(create_call(self.stop))\n\n    sys.exit(0)\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.start","title":"start async","text":"

    Starts a runner.

    The runner will begin monitoring for and executing any scheduled work for all added flows.

    Parameters:

    Name Type Description Default run_once bool

    If True, the runner will through one query loop and then exit.

    False webserver Optional[bool]

    a boolean for whether to start a webserver for this runner. If provided, overrides the default on the runner

    None

    Examples:

    Initialize a Runner, add two flows, and serve them by starting the Runner:

    from prefect import flow, Runner\n\n@flow\ndef hello_flow(name):\n    print(f\"hello {name}\")\n\n@flow\ndef goodbye_flow(name):\n    print(f\"goodbye {name}\")\n\nif __name__ == \"__main__\"\n    runner = Runner(name=\"my-runner\")\n\n    # Will be runnable via the API\n    runner.add_flow(hello_flow)\n\n    # Run on a cron schedule\n    runner.add_flow(goodbye_flow, schedule={\"cron\": \"0 * * * *\"})\n\n    runner.start()\n
    Source code in prefect/runner/runner.py
    @sync_compatible\nasync def start(\n    self, run_once: bool = False, webserver: Optional[bool] = None\n) -> None:\n    \"\"\"\n    Starts a runner.\n\n    The runner will begin monitoring for and executing any scheduled work for all added flows.\n\n    Args:\n        run_once: If True, the runner will through one query loop and then exit.\n        webserver: a boolean for whether to start a webserver for this runner. If provided,\n            overrides the default on the runner\n\n    Examples:\n        Initialize a Runner, add two flows, and serve them by starting the Runner:\n\n        ```python\n        from prefect import flow, Runner\n\n        @flow\n        def hello_flow(name):\n            print(f\"hello {name}\")\n\n        @flow\n        def goodbye_flow(name):\n            print(f\"goodbye {name}\")\n\n        if __name__ == \"__main__\"\n            runner = Runner(name=\"my-runner\")\n\n            # Will be runnable via the API\n            runner.add_flow(hello_flow)\n\n            # Run on a cron schedule\n            runner.add_flow(goodbye_flow, schedule={\"cron\": \"0 * * * *\"})\n\n            runner.start()\n        ```\n    \"\"\"\n    _register_signal(signal.SIGTERM, self.handle_sigterm)\n\n    webserver = webserver if webserver is not None else self.webserver\n\n    if webserver:\n        # we'll start the ASGI server in a separate thread so that\n        # uvicorn does not block the main thread\n        server_thread = threading.Thread(\n            name=\"runner-server-thread\",\n            target=partial(\n                start_webserver,\n                runner=self,\n            ),\n            daemon=True,\n        )\n        server_thread.start()\n\n    async with self as runner:\n        async with self._loops_task_group as tg:\n            for storage in self._storage_objs:\n                if storage.pull_interval:\n                    tg.start_soon(\n                        partial(\n                            critical_service_loop,\n                            workload=storage.pull_code,\n                            interval=storage.pull_interval,\n                            run_once=run_once,\n                            jitter_range=0.3,\n                        )\n                    )\n                else:\n                    tg.start_soon(storage.pull_code)\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=runner._get_and_submit_flow_runs,\n                    interval=self.query_seconds,\n                    run_once=run_once,\n                    jitter_range=0.3,\n                )\n            )\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=runner._check_for_cancelled_flow_runs,\n                    interval=self.query_seconds * 2,\n                    run_once=run_once,\n                    jitter_range=0.3,\n                )\n            )\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.stop","title":"stop async","text":"

    Stops the runner's polling cycle.

    Source code in prefect/runner/runner.py
    @sync_compatible\nasync def stop(self):\n    \"\"\"Stops the runner's polling cycle.\"\"\"\n    if not self.started:\n        raise RuntimeError(\n            \"Runner has not yet started. Please start the runner by calling\"\n            \" .start()\"\n        )\n\n    self.started = False\n    self.stopping = True\n    await self.cancel_all()\n    try:\n        self._loops_task_group.cancel_scope.cancel()\n    except Exception:\n        self._logger.exception(\n            \"Exception encountered while shutting down\", exc_info=True\n        )\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.serve","title":"serve async","text":"

    Serve the provided list of deployments.

    Parameters:

    Name Type Description Default *args RunnerDeployment

    A list of deployments to serve.

    () pause_on_shutdown bool

    A boolean for whether or not to automatically pause deployment schedules on shutdown.

    True **kwargs

    Additional keyword arguments to pass to the runner.

    {}

    Examples:

    Prepare two deployments and serve them:

    import datetime\n\nfrom prefect import flow, serve\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\n@flow\ndef my_other_flow(name):\n    print(f\"goodbye {name}\")\n\nif __name__ == \"__main__\":\n    # Run once a day\n    hello_deploy = my_flow.to_deployment(\n        \"hello\", tags=[\"dev\"], interval=datetime.timedelta(days=1)\n    )\n\n    # Run every Sunday at 4:00 AM\n    bye_deploy = my_other_flow.to_deployment(\n        \"goodbye\", tags=[\"dev\"], cron=\"0 4 * * sun\"\n    )\n\n    serve(hello_deploy, bye_deploy)\n
    Source code in prefect/runner/runner.py
    @sync_compatible\nasync def serve(\n    *args: RunnerDeployment,\n    pause_on_shutdown: bool = True,\n    print_starting_message: bool = True,\n    **kwargs,\n):\n    \"\"\"\n    Serve the provided list of deployments.\n\n    Args:\n        *args: A list of deployments to serve.\n        pause_on_shutdown: A boolean for whether or not to automatically pause\n            deployment schedules on shutdown.\n        **kwargs: Additional keyword arguments to pass to the runner.\n\n    Examples:\n        Prepare two deployments and serve them:\n\n        ```python\n        import datetime\n\n        from prefect import flow, serve\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        @flow\n        def my_other_flow(name):\n            print(f\"goodbye {name}\")\n\n        if __name__ == \"__main__\":\n            # Run once a day\n            hello_deploy = my_flow.to_deployment(\n                \"hello\", tags=[\"dev\"], interval=datetime.timedelta(days=1)\n            )\n\n            # Run every Sunday at 4:00 AM\n            bye_deploy = my_other_flow.to_deployment(\n                \"goodbye\", tags=[\"dev\"], cron=\"0 4 * * sun\"\n            )\n\n            serve(hello_deploy, bye_deploy)\n        ```\n    \"\"\"\n    runner = Runner(pause_on_shutdown=pause_on_shutdown, **kwargs)\n    for deployment in args:\n        await runner.add_deployment(deployment)\n\n    if print_starting_message:\n        help_message_top = (\n            \"[green]Your deployments are being served and polling for\"\n            \" scheduled runs!\\n[/]\"\n        )\n\n        table = Table(title=\"Deployments\", show_header=False)\n\n        table.add_column(style=\"blue\", no_wrap=True)\n\n        for deployment in args:\n            table.add_row(f\"{deployment.flow_name}/{deployment.name}\")\n\n        help_message_bottom = (\n            \"\\nTo trigger any of these deployments, use the\"\n            \" following command:\\n[blue]\\n\\t$ prefect deployment run\"\n            \" [DEPLOYMENT_NAME]\\n[/]\"\n        )\n        if PREFECT_UI_URL:\n            help_message_bottom += (\n                \"\\nYou can also trigger your deployments via the Prefect UI:\"\n                f\" [blue]{PREFECT_UI_URL.value()}/deployments[/]\\n\"\n            )\n\n        console = Console()\n        console.print(Panel(Group(help_message_top, table, help_message_bottom)))\n\n    await runner.start()\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/server/","title":"server","text":"","tags":["Python API","server"]},{"location":"api-ref/prefect/runner/server/#prefect.runner.server","title":"prefect.runner.server","text":"","tags":["Python API","server"]},{"location":"api-ref/prefect/runner/server/#prefect.runner.server.build_server","title":"build_server async","text":"

    Build a FastAPI server for a runner.

    Parameters:

    Name Type Description Default runner Runner

    the runner this server interacts with and monitors

    required log_level str

    the log level to use for the server

    required Source code in prefect/runner/server.py
    @sync_compatible\nasync def build_server(runner: \"Runner\") -> FastAPI:\n    \"\"\"\n    Build a FastAPI server for a runner.\n\n    Args:\n        runner (Runner): the runner this server interacts with and monitors\n        log_level (str): the log level to use for the server\n    \"\"\"\n    webserver = FastAPI()\n    router = APIRouter()\n\n    router.add_api_route(\n        \"/health\", perform_health_check(runner=runner), methods=[\"GET\"]\n    )\n    router.add_api_route(\"/run_count\", run_count(runner=runner), methods=[\"GET\"])\n    router.add_api_route(\"/shutdown\", shutdown(runner=runner), methods=[\"POST\"])\n    webserver.include_router(router)\n\n    if PREFECT_EXPERIMENTAL_ENABLE_EXTRA_RUNNER_ENDPOINTS.value():\n        deployments_router, deployment_schemas = await get_deployment_router(runner)\n        webserver.include_router(deployments_router)\n\n        def customize_openapi():\n            if webserver.openapi_schema:\n                return webserver.openapi_schema\n\n            openapi_schema = inject_schemas_into_openapi(webserver, deployment_schemas)\n            webserver.openapi_schema = openapi_schema\n            return webserver.openapi_schema\n\n        webserver.openapi = customize_openapi\n\n    return webserver\n
    ","tags":["Python API","server"]},{"location":"api-ref/prefect/runner/server/#prefect.runner.server.start_webserver","title":"start_webserver","text":"

    Run a FastAPI server for a runner.

    Parameters:

    Name Type Description Default runner Runner

    the runner this server interacts with and monitors

    required log_level str

    the log level to use for the server

    None Source code in prefect/runner/server.py
    def start_webserver(runner: \"Runner\", log_level: Optional[str] = None) -> None:\n    \"\"\"\n    Run a FastAPI server for a runner.\n\n    Args:\n        runner (Runner): the runner this server interacts with and monitors\n        log_level (str): the log level to use for the server\n    \"\"\"\n    host = PREFECT_RUNNER_SERVER_HOST.value()\n    port = PREFECT_RUNNER_SERVER_PORT.value()\n    log_level = log_level or PREFECT_RUNNER_SERVER_LOG_LEVEL.value()\n    webserver = build_server(runner)\n    uvicorn.run(webserver, host=host, port=port, log_level=log_level)\n
    ","tags":["Python API","server"]},{"location":"api-ref/prefect/runner/storage/","title":"storage","text":"","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage","title":"prefect.runner.storage","text":"","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.BlockStorageAdapter","title":"BlockStorageAdapter","text":"

    A storage adapter for a storage block object to allow it to be used as a runner storage object.

    Source code in prefect/runner/storage.py
    class BlockStorageAdapter:\n    \"\"\"\n    A storage adapter for a storage block object to allow it to be used as a\n    runner storage object.\n    \"\"\"\n\n    def __init__(\n        self,\n        block: Union[ReadableDeploymentStorage, WritableDeploymentStorage],\n        pull_interval: Optional[int] = 60,\n    ):\n        self._block = block\n        self._pull_interval = pull_interval\n        self._storage_base_path = Path.cwd()\n        if not isinstance(block, Block):\n            raise TypeError(\n                f\"Expected a block object. Received a {type(block).__name__!r} object.\"\n            )\n        if not hasattr(block, \"get_directory\"):\n            raise ValueError(\"Provided block must have a `get_directory` method.\")\n\n        self._name = (\n            f\"{block.get_block_type_slug()}-{block._block_document_name}\"\n            if block._block_document_name\n            else str(uuid4())\n        )\n\n    def set_base_path(self, path: Path):\n        self._storage_base_path = path\n\n    @property\n    def pull_interval(self) -> Optional[int]:\n        return self._pull_interval\n\n    @property\n    def destination(self) -> Path:\n        return self._storage_base_path / self._name\n\n    async def pull_code(self):\n        if not self.destination.exists():\n            self.destination.mkdir(parents=True, exist_ok=True)\n        await self._block.get_directory(local_path=str(self.destination))\n\n    def to_pull_step(self) -> dict:\n        # Give blocks the change to implement their own pull step\n        if hasattr(self._block, \"get_pull_step\"):\n            return self._block.get_pull_step()\n        else:\n            if not self._block._block_document_name:\n                raise BlockNotSavedError(\n                    \"Block must be saved with `.save()` before it can be converted to a\"\n                    \" pull step.\"\n                )\n            return {\n                \"prefect.deployments.steps.pull_with_block\": {\n                    \"block_type_slug\": self._block.get_block_type_slug(),\n                    \"block_document_name\": self._block._block_document_name,\n                }\n            }\n\n    def __eq__(self, __value) -> bool:\n        if isinstance(__value, BlockStorageAdapter):\n            return self._block == __value._block\n        return False\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.GitRepository","title":"GitRepository","text":"

    Pulls the contents of a git repository to the local filesystem.

    Parameters:

    Name Type Description Default url str

    The URL of the git repository to pull from

    required credentials Union[GitCredentials, Block, None]

    A dictionary of credentials to use when pulling from the repository. If a username is provided, an access token must also be provided.

    None name Optional[str]

    The name of the repository. If not provided, the name will be inferred from the repository URL.

    None branch Optional[str]

    The branch to pull from. Defaults to \"main\".

    None pull_interval Optional[int]

    The interval in seconds at which to pull contents from remote storage to local storage. If None, remote storage will perform a one-time sync.

    60

    Examples:

    Pull the contents of a private git repository to the local filesystem:

    from prefect.runner.storage import GitRepository\n\nstorage = GitRepository(\n    url=\"https://github.com/org/repo.git\",\n    credentials={\"username\": \"oauth2\", \"access_token\": \"my-access-token\"},\n)\n\nawait storage.pull_code()\n
    Source code in prefect/runner/storage.py
    class GitRepository:\n    \"\"\"\n    Pulls the contents of a git repository to the local filesystem.\n\n    Parameters:\n        url: The URL of the git repository to pull from\n        credentials: A dictionary of credentials to use when pulling from the\n            repository. If a username is provided, an access token must also be\n            provided.\n        name: The name of the repository. If not provided, the name will be\n            inferred from the repository URL.\n        branch: The branch to pull from. Defaults to \"main\".\n        pull_interval: The interval in seconds at which to pull contents from\n            remote storage to local storage. If None, remote storage will perform\n            a one-time sync.\n\n    Examples:\n        Pull the contents of a private git repository to the local filesystem:\n\n        ```python\n        from prefect.runner.storage import GitRepository\n\n        storage = GitRepository(\n            url=\"https://github.com/org/repo.git\",\n            credentials={\"username\": \"oauth2\", \"access_token\": \"my-access-token\"},\n        )\n\n        await storage.pull_code()\n        ```\n    \"\"\"\n\n    def __init__(\n        self,\n        url: str,\n        credentials: Union[GitCredentials, Block, None] = None,\n        name: Optional[str] = None,\n        branch: Optional[str] = None,\n        include_submodules: bool = False,\n        pull_interval: Optional[int] = 60,\n    ):\n        if credentials is None:\n            credentials = {}\n\n        if (\n            isinstance(credentials, dict)\n            and credentials.get(\"username\")\n            and not (credentials.get(\"access_token\") or credentials.get(\"password\"))\n        ):\n            raise ValueError(\n                \"If a username is provided, an access token or password must also be\"\n                \" provided.\"\n            )\n        self._url = url\n        self._branch = branch\n        self._credentials = credentials\n        self._include_submodules = include_submodules\n        repo_name = urlparse(url).path.split(\"/\")[-1].replace(\".git\", \"\")\n        default_name = f\"{repo_name}-{branch}\" if branch else repo_name\n        self._name = name or default_name\n        self._logger = get_logger(f\"runner.storage.git-repository.{self._name}\")\n        self._storage_base_path = Path.cwd()\n        self._pull_interval = pull_interval\n\n    @property\n    def destination(self) -> Path:\n        return self._storage_base_path / self._name\n\n    def set_base_path(self, path: Path):\n        self._storage_base_path = path\n\n    @property\n    def pull_interval(self) -> Optional[int]:\n        return self._pull_interval\n\n    @property\n    def _repository_url_with_credentials(self) -> str:\n        if not self._credentials:\n            return self._url\n\n        url_components = urlparse(self._url)\n\n        credentials = (\n            self._credentials.dict()\n            if isinstance(self._credentials, Block)\n            else deepcopy(self._credentials)\n        )\n\n        for k, v in credentials.items():\n            if isinstance(v, Secret):\n                credentials[k] = v.get()\n            elif isinstance(v, SecretStr):\n                credentials[k] = v.get_secret_value()\n\n        formatted_credentials = _format_token_from_credentials(\n            urlparse(self._url).netloc, credentials\n        )\n        if url_components.scheme == \"https\" and formatted_credentials is not None:\n            updated_components = url_components._replace(\n                netloc=f\"{formatted_credentials}@{url_components.netloc}\"\n            )\n            repository_url = urlunparse(updated_components)\n        else:\n            repository_url = self._url\n\n        return repository_url\n\n    async def pull_code(self):\n        \"\"\"\n        Pulls the contents of the configured repository to the local filesystem.\n        \"\"\"\n        self._logger.debug(\n            \"Pulling contents from repository '%s' to '%s'...\",\n            self._name,\n            self.destination,\n        )\n\n        git_dir = self.destination / \".git\"\n\n        if git_dir.exists():\n            # Check if the existing repository matches the configured repository\n            result = await run_process(\n                [\"git\", \"config\", \"--get\", \"remote.origin.url\"],\n                cwd=str(self.destination),\n            )\n            existing_repo_url = None\n            if result.stdout is not None:\n                existing_repo_url = _strip_auth_from_url(result.stdout.decode().strip())\n\n            if existing_repo_url != self._url:\n                raise ValueError(\n                    f\"The existing repository at {str(self.destination)} \"\n                    f\"does not match the configured repository {self._url}\"\n                )\n\n            self._logger.debug(\"Pulling latest changes from origin/%s\", self._branch)\n            # Update the existing repository\n            cmd = [\"git\", \"pull\", \"origin\"]\n            if self._branch:\n                cmd += [self._branch]\n            if self._include_submodules:\n                cmd += [\"--recurse-submodules\"]\n            cmd += [\"--depth\", \"1\"]\n            try:\n                await run_process(cmd, cwd=self.destination)\n                self._logger.debug(\"Successfully pulled latest changes\")\n            except subprocess.CalledProcessError as exc:\n                self._logger.error(\n                    f\"Failed to pull latest changes with exit code {exc}\"\n                )\n                shutil.rmtree(self.destination)\n                await self._clone_repo()\n\n        else:\n            await self._clone_repo()\n\n    async def _clone_repo(self):\n        \"\"\"\n        Clones the repository into the local destination.\n        \"\"\"\n        self._logger.debug(\"Cloning repository %s\", self._url)\n\n        repository_url = self._repository_url_with_credentials\n\n        cmd = [\n            \"git\",\n            \"clone\",\n            repository_url,\n        ]\n        if self._branch:\n            cmd += [\"--branch\", self._branch]\n        if self._include_submodules:\n            cmd += [\"--recurse-submodules\"]\n\n        # Limit git history and set path to clone to\n        cmd += [\"--depth\", \"1\", str(self.destination)]\n\n        try:\n            await run_process(cmd)\n        except subprocess.CalledProcessError as exc:\n            # Hide the command used to avoid leaking the access token\n            exc_chain = None if self._credentials else exc\n            raise RuntimeError(\n                f\"Failed to clone repository {self._url!r} with exit code\"\n                f\" {exc.returncode}.\"\n            ) from exc_chain\n\n    def __eq__(self, __value) -> bool:\n        if isinstance(__value, GitRepository):\n            return (\n                self._url == __value._url\n                and self._branch == __value._branch\n                and self._name == __value._name\n            )\n        return False\n\n    def __repr__(self) -> str:\n        return (\n            f\"GitRepository(name={self._name!r} repository={self._url!r},\"\n            f\" branch={self._branch!r})\"\n        )\n\n    def to_pull_step(self) -> Dict:\n        pull_step = {\n            \"prefect.deployments.steps.git_clone\": {\n                \"repository\": self._url,\n                \"branch\": self._branch,\n            }\n        }\n        if isinstance(self._credentials, Block):\n            pull_step[\"prefect.deployments.steps.git_clone\"][\n                \"credentials\"\n            ] = f\"{{{{ {self._credentials.get_block_placeholder()} }}}}\"\n        elif isinstance(self._credentials, dict):\n            if isinstance(self._credentials.get(\"access_token\"), Secret):\n                pull_step[\"prefect.deployments.steps.git_clone\"][\"credentials\"] = {\n                    **self._credentials,\n                    \"access_token\": (\n                        \"{{\"\n                        f\" {self._credentials['access_token'].get_block_placeholder()} }}}}\"\n                    ),\n                }\n            elif self._credentials.get(\"access_token\") is not None:\n                raise ValueError(\n                    \"Please save your access token as a Secret block before converting\"\n                    \" this storage object to a pull step.\"\n                )\n\n        return pull_step\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.GitRepository.pull_code","title":"pull_code async","text":"

    Pulls the contents of the configured repository to the local filesystem.

    Source code in prefect/runner/storage.py
    async def pull_code(self):\n    \"\"\"\n    Pulls the contents of the configured repository to the local filesystem.\n    \"\"\"\n    self._logger.debug(\n        \"Pulling contents from repository '%s' to '%s'...\",\n        self._name,\n        self.destination,\n    )\n\n    git_dir = self.destination / \".git\"\n\n    if git_dir.exists():\n        # Check if the existing repository matches the configured repository\n        result = await run_process(\n            [\"git\", \"config\", \"--get\", \"remote.origin.url\"],\n            cwd=str(self.destination),\n        )\n        existing_repo_url = None\n        if result.stdout is not None:\n            existing_repo_url = _strip_auth_from_url(result.stdout.decode().strip())\n\n        if existing_repo_url != self._url:\n            raise ValueError(\n                f\"The existing repository at {str(self.destination)} \"\n                f\"does not match the configured repository {self._url}\"\n            )\n\n        self._logger.debug(\"Pulling latest changes from origin/%s\", self._branch)\n        # Update the existing repository\n        cmd = [\"git\", \"pull\", \"origin\"]\n        if self._branch:\n            cmd += [self._branch]\n        if self._include_submodules:\n            cmd += [\"--recurse-submodules\"]\n        cmd += [\"--depth\", \"1\"]\n        try:\n            await run_process(cmd, cwd=self.destination)\n            self._logger.debug(\"Successfully pulled latest changes\")\n        except subprocess.CalledProcessError as exc:\n            self._logger.error(\n                f\"Failed to pull latest changes with exit code {exc}\"\n            )\n            shutil.rmtree(self.destination)\n            await self._clone_repo()\n\n    else:\n        await self._clone_repo()\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage","title":"RemoteStorage","text":"

    Pulls the contents of a remote storage location to the local filesystem.

    Parameters:

    Name Type Description Default url str

    The URL of the remote storage location to pull from. Supports fsspec URLs. Some protocols may require an additional fsspec dependency to be installed. Refer to the fsspec docs for more details.

    required pull_interval Optional[int]

    The interval in seconds at which to pull contents from remote storage to local storage. If None, remote storage will perform a one-time sync.

    60 **settings Any

    Any additional settings to pass the fsspec filesystem class.

    {}

    Examples:

    Pull the contents of a remote storage location to the local filesystem:

    from prefect.runner.storage import RemoteStorage\n\nstorage = RemoteStorage(url=\"s3://my-bucket/my-folder\")\n\nawait storage.pull_code()\n

    Pull the contents of a remote storage location to the local filesystem with additional settings:

    from prefect.runner.storage import RemoteStorage\nfrom prefect.blocks.system import Secret\n\nstorage = RemoteStorage(\n    url=\"s3://my-bucket/my-folder\",\n    # Use Secret blocks to keep credentials out of your code\n    key=Secret.load(\"my-aws-access-key\"),\n    secret=Secret.load(\"my-aws-secret-key\"),\n)\n\nawait storage.pull_code()\n
    Source code in prefect/runner/storage.py
    class RemoteStorage:\n    \"\"\"\n    Pulls the contents of a remote storage location to the local filesystem.\n\n    Parameters:\n        url: The URL of the remote storage location to pull from. Supports\n            `fsspec` URLs. Some protocols may require an additional `fsspec`\n            dependency to be installed. Refer to the\n            [`fsspec` docs](https://filesystem-spec.readthedocs.io/en/latest/api.html#other-known-implementations)\n            for more details.\n        pull_interval: The interval in seconds at which to pull contents from\n            remote storage to local storage. If None, remote storage will perform\n            a one-time sync.\n        **settings: Any additional settings to pass the `fsspec` filesystem class.\n\n    Examples:\n        Pull the contents of a remote storage location to the local filesystem:\n\n        ```python\n        from prefect.runner.storage import RemoteStorage\n\n        storage = RemoteStorage(url=\"s3://my-bucket/my-folder\")\n\n        await storage.pull_code()\n        ```\n\n        Pull the contents of a remote storage location to the local filesystem\n        with additional settings:\n\n        ```python\n        from prefect.runner.storage import RemoteStorage\n        from prefect.blocks.system import Secret\n\n        storage = RemoteStorage(\n            url=\"s3://my-bucket/my-folder\",\n            # Use Secret blocks to keep credentials out of your code\n            key=Secret.load(\"my-aws-access-key\"),\n            secret=Secret.load(\"my-aws-secret-key\"),\n        )\n\n        await storage.pull_code()\n        ```\n    \"\"\"\n\n    def __init__(\n        self,\n        url: str,\n        pull_interval: Optional[int] = 60,\n        **settings: Any,\n    ):\n        self._url = url\n        self._settings = settings\n        self._logger = get_logger(\"runner.storage.remote-storage\")\n        self._storage_base_path = Path.cwd()\n        self._pull_interval = pull_interval\n\n    @staticmethod\n    def _get_required_package_for_scheme(scheme: str) -> Optional[str]:\n        # attempt to discover the package name for the given scheme\n        # from fsspec's registry\n        known_implementation = fsspec.registry.get(scheme)\n        if known_implementation:\n            return known_implementation.__module__.split(\".\")[0]\n        # if we don't know the implementation, try to guess it for some\n        # common schemes\n        elif scheme == \"s3\":\n            return \"s3fs\"\n        elif scheme == \"gs\" or scheme == \"gcs\":\n            return \"gcsfs\"\n        elif scheme == \"abfs\" or scheme == \"az\":\n            return \"adlfs\"\n        else:\n            return None\n\n    @property\n    def _filesystem(self) -> fsspec.AbstractFileSystem:\n        scheme, _, _, _, _ = urlsplit(self._url)\n\n        def replace_blocks_with_values(obj: Any) -> Any:\n            if isinstance(obj, Block):\n                if hasattr(obj, \"get\"):\n                    return obj.get()\n                if hasattr(obj, \"value\"):\n                    return obj.value\n                else:\n                    return obj.dict()\n            return obj\n\n        settings_with_block_values = visit_collection(\n            self._settings, replace_blocks_with_values, return_data=True\n        )\n\n        return fsspec.filesystem(scheme, **settings_with_block_values)\n\n    def set_base_path(self, path: Path):\n        self._storage_base_path = path\n\n    @property\n    def pull_interval(self) -> Optional[int]:\n        \"\"\"\n        The interval at which contents from remote storage should be pulled to\n        local storage. If None, remote storage will perform a one-time sync.\n        \"\"\"\n        return self._pull_interval\n\n    @property\n    def destination(self) -> Path:\n        \"\"\"\n        The local file path to pull contents from remote storage to.\n        \"\"\"\n        return self._storage_base_path / self._remote_path\n\n    @property\n    def _remote_path(self) -> Path:\n        \"\"\"\n        The remote file path to pull contents from remote storage to.\n        \"\"\"\n        _, netloc, urlpath, _, _ = urlsplit(self._url)\n        return Path(netloc) / Path(urlpath.lstrip(\"/\"))\n\n    async def pull_code(self):\n        \"\"\"\n        Pulls contents from remote storage to the local filesystem.\n        \"\"\"\n        self._logger.debug(\n            \"Pulling contents from remote storage '%s' to '%s'...\",\n            self._url,\n            self.destination,\n        )\n\n        if not self.destination.exists():\n            self.destination.mkdir(parents=True, exist_ok=True)\n\n        remote_path = str(self._remote_path) + \"/\"\n\n        try:\n            await from_async.wait_for_call_in_new_thread(\n                create_call(\n                    self._filesystem.get,\n                    remote_path,\n                    str(self.destination),\n                    recursive=True,\n                )\n            )\n        except Exception as exc:\n            raise RuntimeError(\n                f\"Failed to pull contents from remote storage {self._url!r} to\"\n                f\" {self.destination!r}\"\n            ) from exc\n\n    def to_pull_step(self) -> dict:\n        \"\"\"\n        Returns a dictionary representation of the storage object that can be\n        used as a deployment pull step.\n        \"\"\"\n\n        def replace_block_with_placeholder(obj: Any) -> Any:\n            if isinstance(obj, Block):\n                return f\"{{{{ {obj.get_block_placeholder()} }}}}\"\n            return obj\n\n        settings_with_placeholders = visit_collection(\n            self._settings, replace_block_with_placeholder, return_data=True\n        )\n        required_package = self._get_required_package_for_scheme(\n            urlparse(self._url).scheme\n        )\n        step = {\n            \"prefect.deployments.steps.pull_from_remote_storage\": {\n                \"url\": self._url,\n                **settings_with_placeholders,\n            }\n        }\n        if required_package:\n            step[\"prefect.deployments.steps.pull_from_remote_storage\"][\n                \"requires\"\n            ] = required_package\n        return step\n\n    def __eq__(self, __value) -> bool:\n        \"\"\"\n        Equality check for runner storage objects.\n        \"\"\"\n        if isinstance(__value, RemoteStorage):\n            return self._url == __value._url and self._settings == __value._settings\n        return False\n\n    def __repr__(self) -> str:\n        return f\"RemoteStorage(url={self._url!r})\"\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage.destination","title":"destination: Path property","text":"

    The local file path to pull contents from remote storage to.

    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage.pull_interval","title":"pull_interval: Optional[int] property","text":"

    The interval at which contents from remote storage should be pulled to local storage. If None, remote storage will perform a one-time sync.

    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage.pull_code","title":"pull_code async","text":"

    Pulls contents from remote storage to the local filesystem.

    Source code in prefect/runner/storage.py
    async def pull_code(self):\n    \"\"\"\n    Pulls contents from remote storage to the local filesystem.\n    \"\"\"\n    self._logger.debug(\n        \"Pulling contents from remote storage '%s' to '%s'...\",\n        self._url,\n        self.destination,\n    )\n\n    if not self.destination.exists():\n        self.destination.mkdir(parents=True, exist_ok=True)\n\n    remote_path = str(self._remote_path) + \"/\"\n\n    try:\n        await from_async.wait_for_call_in_new_thread(\n            create_call(\n                self._filesystem.get,\n                remote_path,\n                str(self.destination),\n                recursive=True,\n            )\n        )\n    except Exception as exc:\n        raise RuntimeError(\n            f\"Failed to pull contents from remote storage {self._url!r} to\"\n            f\" {self.destination!r}\"\n        ) from exc\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage.to_pull_step","title":"to_pull_step","text":"

    Returns a dictionary representation of the storage object that can be used as a deployment pull step.

    Source code in prefect/runner/storage.py
    def to_pull_step(self) -> dict:\n    \"\"\"\n    Returns a dictionary representation of the storage object that can be\n    used as a deployment pull step.\n    \"\"\"\n\n    def replace_block_with_placeholder(obj: Any) -> Any:\n        if isinstance(obj, Block):\n            return f\"{{{{ {obj.get_block_placeholder()} }}}}\"\n        return obj\n\n    settings_with_placeholders = visit_collection(\n        self._settings, replace_block_with_placeholder, return_data=True\n    )\n    required_package = self._get_required_package_for_scheme(\n        urlparse(self._url).scheme\n    )\n    step = {\n        \"prefect.deployments.steps.pull_from_remote_storage\": {\n            \"url\": self._url,\n            **settings_with_placeholders,\n        }\n    }\n    if required_package:\n        step[\"prefect.deployments.steps.pull_from_remote_storage\"][\n            \"requires\"\n        ] = required_package\n    return step\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage","title":"RunnerStorage","text":"

    Bases: Protocol

    A storage interface for a runner to use to retrieve remotely stored flow code.

    Source code in prefect/runner/storage.py
    @runtime_checkable\nclass RunnerStorage(Protocol):\n    \"\"\"\n    A storage interface for a runner to use to retrieve\n    remotely stored flow code.\n    \"\"\"\n\n    def set_base_path(self, path: Path):\n        \"\"\"\n        Sets the base path to use when pulling contents from remote storage to\n        local storage.\n        \"\"\"\n        ...\n\n    @property\n    def pull_interval(self) -> Optional[int]:\n        \"\"\"\n        The interval at which contents from remote storage should be pulled to\n        local storage. If None, remote storage will perform a one-time sync.\n        \"\"\"\n        ...\n\n    @property\n    def destination(self) -> Path:\n        \"\"\"\n        The local file path to pull contents from remote storage to.\n        \"\"\"\n        ...\n\n    async def pull_code(self):\n        \"\"\"\n        Pulls contents from remote storage to the local filesystem.\n        \"\"\"\n        ...\n\n    def to_pull_step(self) -> dict:\n        \"\"\"\n        Returns a dictionary representation of the storage object that can be\n        used as a deployment pull step.\n        \"\"\"\n        ...\n\n    def __eq__(self, __value) -> bool:\n        \"\"\"\n        Equality check for runner storage objects.\n        \"\"\"\n        ...\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.destination","title":"destination: Path property","text":"

    The local file path to pull contents from remote storage to.

    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.pull_interval","title":"pull_interval: Optional[int] property","text":"

    The interval at which contents from remote storage should be pulled to local storage. If None, remote storage will perform a one-time sync.

    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.pull_code","title":"pull_code async","text":"

    Pulls contents from remote storage to the local filesystem.

    Source code in prefect/runner/storage.py
    async def pull_code(self):\n    \"\"\"\n    Pulls contents from remote storage to the local filesystem.\n    \"\"\"\n    ...\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.set_base_path","title":"set_base_path","text":"

    Sets the base path to use when pulling contents from remote storage to local storage.

    Source code in prefect/runner/storage.py
    def set_base_path(self, path: Path):\n    \"\"\"\n    Sets the base path to use when pulling contents from remote storage to\n    local storage.\n    \"\"\"\n    ...\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.to_pull_step","title":"to_pull_step","text":"

    Returns a dictionary representation of the storage object that can be used as a deployment pull step.

    Source code in prefect/runner/storage.py
    def to_pull_step(self) -> dict:\n    \"\"\"\n    Returns a dictionary representation of the storage object that can be\n    used as a deployment pull step.\n    \"\"\"\n    ...\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.create_storage_from_url","title":"create_storage_from_url","text":"

    Creates a storage object from a URL.

    Parameters:

    Name Type Description Default url str

    The URL to create a storage object from. Supports git and fsspec URLs.

    required pull_interval Optional[int]

    The interval at which to pull contents from remote storage to local storage

    60

    Returns:

    Name Type Description RunnerStorage RunnerStorage

    A runner storage compatible object

    Source code in prefect/runner/storage.py
    def create_storage_from_url(\n    url: str, pull_interval: Optional[int] = 60\n) -> RunnerStorage:\n    \"\"\"\n    Creates a storage object from a URL.\n\n    Args:\n        url: The URL to create a storage object from. Supports git and `fsspec`\n            URLs.\n        pull_interval: The interval at which to pull contents from remote storage to\n            local storage\n\n    Returns:\n        RunnerStorage: A runner storage compatible object\n    \"\"\"\n    parsed_url = urlparse(url)\n    if parsed_url.scheme == \"git\" or parsed_url.path.endswith(\".git\"):\n        return GitRepository(url=url, pull_interval=pull_interval)\n    else:\n        return RemoteStorage(url=url, pull_interval=pull_interval)\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/utils/","title":"utils","text":"","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils","title":"prefect.runner.utils","text":"","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils.inject_schemas_into_openapi","title":"inject_schemas_into_openapi","text":"

    Augments the webserver's OpenAPI schema with additional schemas from deployments.

    Parameters:

    Name Type Description Default webserver FastAPI

    The FastAPI instance representing the webserver.

    required deployment_schemas Dict[str, Any]

    A dictionary of deployment schemas to integrate.

    required

    Returns:

    Type Description Dict[str, Any]

    The augmented OpenAPI schema dictionary.

    Source code in prefect/runner/utils.py
    def inject_schemas_into_openapi(\n    webserver: FastAPI, deployment_schemas: Dict[str, Any]\n) -> Dict[str, Any]:\n    \"\"\"\n    Augments the webserver's OpenAPI schema with additional schemas from deployments.\n\n    Args:\n        webserver: The FastAPI instance representing the webserver.\n        deployment_schemas: A dictionary of deployment schemas to integrate.\n\n    Returns:\n        The augmented OpenAPI schema dictionary.\n    \"\"\"\n    openapi_schema = get_openapi(\n        title=\"FastAPI Prefect Runner\", version=PREFECT_VERSION, routes=webserver.routes\n    )\n\n    augmented_schema = merge_definitions(deployment_schemas, openapi_schema)\n    return update_refs_to_components(augmented_schema)\n
    ","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils.merge_definitions","title":"merge_definitions","text":"

    Integrates definitions from deployment schemas into the OpenAPI components.

    Parameters:

    Name Type Description Default deployment_schemas Dict[str, Any]

    A dictionary of deployment-specific schemas.

    required openapi_schema Dict[str, Any]

    The base OpenAPI schema to update.

    required Source code in prefect/runner/utils.py
    def merge_definitions(\n    deployment_schemas: Dict[str, Any], openapi_schema: Dict[str, Any]\n) -> Dict[str, Any]:\n    \"\"\"\n    Integrates definitions from deployment schemas into the OpenAPI components.\n\n    Args:\n        deployment_schemas: A dictionary of deployment-specific schemas.\n        openapi_schema: The base OpenAPI schema to update.\n    \"\"\"\n    openapi_schema_copy = deepcopy(openapi_schema)\n    components = openapi_schema_copy.setdefault(\"components\", {}).setdefault(\n        \"schemas\", {}\n    )\n    for definitions in deployment_schemas.values():\n        if \"definitions\" in definitions:\n            for def_name, def_schema in definitions[\"definitions\"].items():\n                def_schema_copy = deepcopy(def_schema)\n                update_refs_in_schema(def_schema_copy, \"#/components/schemas/\")\n                components[def_name] = def_schema_copy\n    return openapi_schema_copy\n
    ","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils.update_refs_in_schema","title":"update_refs_in_schema","text":"

    Recursively replaces $ref with a new reference base in a schema item.

    Parameters:

    Name Type Description Default schema_item Any

    A schema or part of a schema to update references in.

    required new_ref str

    The new base string to replace in $ref values.

    required Source code in prefect/runner/utils.py
    def update_refs_in_schema(schema_item: Any, new_ref: str) -> None:\n    \"\"\"\n    Recursively replaces `$ref` with a new reference base in a schema item.\n\n    Args:\n        schema_item: A schema or part of a schema to update references in.\n        new_ref: The new base string to replace in `$ref` values.\n    \"\"\"\n    if isinstance(schema_item, dict):\n        if \"$ref\" in schema_item:\n            schema_item[\"$ref\"] = schema_item[\"$ref\"].replace(\"#/definitions/\", new_ref)\n        for value in schema_item.values():\n            update_refs_in_schema(value, new_ref)\n    elif isinstance(schema_item, list):\n        for item in schema_item:\n            update_refs_in_schema(item, new_ref)\n
    ","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils.update_refs_to_components","title":"update_refs_to_components","text":"

    Updates all $ref fields in the OpenAPI schema to reference the components section.

    Parameters:

    Name Type Description Default openapi_schema Dict[str, Any]

    The OpenAPI schema to modify $ref fields in.

    required Source code in prefect/runner/utils.py
    def update_refs_to_components(openapi_schema: Dict[str, Any]) -> Dict[str, Any]:\n    \"\"\"\n    Updates all `$ref` fields in the OpenAPI schema to reference the components section.\n\n    Args:\n        openapi_schema: The OpenAPI schema to modify `$ref` fields in.\n    \"\"\"\n    for path_item in openapi_schema.get(\"paths\", {}).values():\n        for operation in path_item.values():\n            schema = (\n                operation.get(\"requestBody\", {})\n                .get(\"content\", {})\n                .get(\"application/json\", {})\n                .get(\"schema\", {})\n            )\n            update_refs_in_schema(schema, \"#/components/schemas/\")\n\n    for definition in openapi_schema.get(\"definitions\", {}).values():\n        update_refs_in_schema(definition, \"#/components/schemas/\")\n\n    return openapi_schema\n
    ","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runtime/deployment/","title":"deployment","text":"","tags":["Python API","deployment context","context"]},{"location":"api-ref/prefect/runtime/deployment/#prefect.runtime.deployment","title":"prefect.runtime.deployment","text":"

    Access attributes of the current deployment run dynamically.

    Note that if a deployment is not currently being run, all attributes will return empty values.

    You can mock the runtime attributes for testing purposes by setting environment variables prefixed with PREFECT__RUNTIME__DEPLOYMENT.

    Example usage
    from prefect.runtime import deployment\n\ndef get_task_runner():\n    task_runner_config = deployment.parameters.get(\"runner_config\", \"default config here\")\n    return DummyTaskRunner(task_runner_specs=task_runner_config)\n
    Available attributes
    • id: the deployment's unique ID
    • name: the deployment's name
    • version: the deployment's version
    • flow_run_id: the current flow run ID for this deployment
    • parameters: the parameters that were passed to this run; note that these do not necessarily include default values set on the flow function, only the parameter values set on the deployment object or those directly provided via API for this run
    ","tags":["Python API","deployment context","context"]},{"location":"api-ref/prefect/runtime/flow_run/","title":"flow_run","text":"","tags":["Python API","flow run context","context"]},{"location":"api-ref/prefect/runtime/flow_run/#prefect.runtime.flow_run","title":"prefect.runtime.flow_run","text":"

    Access attributes of the current flow run dynamically.

    Note that if a flow run cannot be discovered, all attributes will return empty values.

    You can mock the runtime attributes for testing purposes by setting environment variables prefixed with PREFECT__RUNTIME__FLOW_RUN.

    Available attributes
    • id: the flow run's unique ID
    • tags: the flow run's set of tags
    • scheduled_start_time: the flow run's expected scheduled start time; defaults to now if not present
    • name: the name of the flow run
    • flow_name: the name of the flow
    • parameters: the parameters that were passed to this run; note that these do not necessarily include default values set on the flow function, only the parameter values explicitly passed for the run
    • parent_flow_run_id: the ID of the flow run that triggered this run, if any
    • parent_deployment_id: the ID of the deployment that triggered this run, if any
    • run_count: the number of times this flow run has been run
    ","tags":["Python API","flow run context","context"]},{"location":"api-ref/prefect/runtime/task_run/","title":"task_run","text":"","tags":["Python API","task run context","context","task run"]},{"location":"api-ref/prefect/runtime/task_run/#prefect.runtime.task_run","title":"prefect.runtime.task_run","text":"

    Access attributes of the current task run dynamically.

    Note that if a task run cannot be discovered, all attributes will return empty values.

    You can mock the runtime attributes for testing purposes by setting environment variables prefixed with PREFECT__RUNTIME__TASK_RUN.

    Available attributes
    • id: the task run's unique ID
    • name: the name of the task run
    • tags: the task run's set of tags
    • parameters: the parameters the task was called with
    • run_count: the number of times this task run has been run
    • task_name: the name of the task
    ","tags":["Python API","task run context","context","task run"]},{"location":"api-ref/prefect/utilities/annotations/","title":"annotations","text":"","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations","title":"prefect.utilities.annotations","text":"","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.BaseAnnotation","title":"BaseAnnotation","text":"

    Bases: namedtuple('BaseAnnotation', field_names='value'), ABC, Generic[T]

    Base class for Prefect annotation types.

    Inherits from namedtuple for unpacking support in another tools.

    Source code in prefect/utilities/annotations.py
    class BaseAnnotation(\n    namedtuple(\"BaseAnnotation\", field_names=\"value\"), ABC, Generic[T]\n):\n    \"\"\"\n    Base class for Prefect annotation types.\n\n    Inherits from `namedtuple` for unpacking support in another tools.\n    \"\"\"\n\n    def unwrap(self) -> T:\n        if sys.version_info < (3, 8):\n            # cannot simply return self.value due to recursion error in Python 3.7\n            # also _asdict does not follow convention; it's not an internal method\n            # https://stackoverflow.com/a/26180604\n            return self._asdict()[\"value\"]\n        else:\n            return self.value\n\n    def rewrap(self, value: T) -> \"BaseAnnotation[T]\":\n        return type(self)(value)\n\n    def __eq__(self, other: object) -> bool:\n        if not type(self) == type(other):\n            return False\n        return self.unwrap() == other.unwrap()\n\n    def __repr__(self) -> str:\n        return f\"{type(self).__name__}({self.value!r})\"\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.NotSet","title":"NotSet","text":"

    Singleton to distinguish None from a value that is not provided by the user.

    Source code in prefect/utilities/annotations.py
    class NotSet:\n    \"\"\"\n    Singleton to distinguish `None` from a value that is not provided by the user.\n    \"\"\"\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.allow_failure","title":"allow_failure","text":"

    Bases: BaseAnnotation[T]

    Wrapper for states or futures.

    Indicates that the upstream run for this input can be failed.

    Generally, Prefect will not allow a downstream run to start if any of its inputs are failed. This annotation allows you to opt into receiving a failed input downstream.

    If the input is from a failed run, the attached exception will be passed to your function.

    Source code in prefect/utilities/annotations.py
    class allow_failure(BaseAnnotation[T]):\n    \"\"\"\n    Wrapper for states or futures.\n\n    Indicates that the upstream run for this input can be failed.\n\n    Generally, Prefect will not allow a downstream run to start if any of its inputs\n    are failed. This annotation allows you to opt into receiving a failed input\n    downstream.\n\n    If the input is from a failed run, the attached exception will be passed to your\n    function.\n    \"\"\"\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.quote","title":"quote","text":"

    Bases: BaseAnnotation[T]

    Simple wrapper to mark an expression as a different type so it will not be coerced by Prefect. For example, if you want to return a state from a flow without having the flow assume that state.

    quote will also instruct prefect to ignore introspection of the wrapped object when passed as flow or task parameter. Parameter introspection can be a significant performance hit when the object is a large collection, e.g. a large dictionary or DataFrame, and each element needs to be visited. This will disable task dependency tracking for the wrapped object, but likely will increase performance.

    @task\ndef my_task(df):\n    ...\n\n@flow\ndef my_flow():\n    my_task(quote(df))\n
    Source code in prefect/utilities/annotations.py
    class quote(BaseAnnotation[T]):\n    \"\"\"\n    Simple wrapper to mark an expression as a different type so it will not be coerced\n    by Prefect. For example, if you want to return a state from a flow without having\n    the flow assume that state.\n\n    quote will also instruct prefect to ignore introspection of the wrapped object\n    when passed as flow or task parameter. Parameter introspection can be a\n    significant performance hit when the object is a large collection,\n    e.g. a large dictionary or DataFrame, and each element needs to be visited. This\n    will disable task dependency tracking for the wrapped object, but likely will\n    increase performance.\n\n    ```\n    @task\n    def my_task(df):\n        ...\n\n    @flow\n    def my_flow():\n        my_task(quote(df))\n    ```\n    \"\"\"\n\n    def unquote(self) -> T:\n        return self.unwrap()\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.unmapped","title":"unmapped","text":"

    Bases: BaseAnnotation[T]

    Wrapper for iterables.

    Indicates that this input should be sent as-is to all runs created during a mapping operation instead of being split.

    Source code in prefect/utilities/annotations.py
    class unmapped(BaseAnnotation[T]):\n    \"\"\"\n    Wrapper for iterables.\n\n    Indicates that this input should be sent as-is to all runs created during a mapping\n    operation instead of being split.\n    \"\"\"\n\n    def __getitem__(self, _) -> T:\n        # Internally, this acts as an infinite array where all items are the same value\n        return self.unwrap()\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/asyncutils/","title":"asyncutils","text":"","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils","title":"prefect.utilities.asyncutils","text":"

    Utilities for interoperability with async functions and workers from various contexts.

    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.GatherIncomplete","title":"GatherIncomplete","text":"

    Bases: RuntimeError

    Used to indicate retrieving gather results before completion

    Source code in prefect/utilities/asyncutils.py
    class GatherIncomplete(RuntimeError):\n    \"\"\"Used to indicate retrieving gather results before completion\"\"\"\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.GatherTaskGroup","title":"GatherTaskGroup","text":"

    Bases: TaskGroup

    A task group that gathers results.

    AnyIO does not include support gather. This class extends the TaskGroup interface to allow simple gathering.

    See https://github.com/agronholm/anyio/issues/100

    This class should be instantiated with create_gather_task_group.

    Source code in prefect/utilities/asyncutils.py
    class GatherTaskGroup(anyio.abc.TaskGroup):\n    \"\"\"\n    A task group that gathers results.\n\n    AnyIO does not include support `gather`. This class extends the `TaskGroup`\n    interface to allow simple gathering.\n\n    See https://github.com/agronholm/anyio/issues/100\n\n    This class should be instantiated with `create_gather_task_group`.\n    \"\"\"\n\n    def __init__(self, task_group: anyio.abc.TaskGroup):\n        self._results: Dict[UUID, Any] = {}\n        # The concrete task group implementation to use\n        self._task_group: anyio.abc.TaskGroup = task_group\n\n    async def _run_and_store(self, key, fn, args):\n        self._results[key] = await fn(*args)\n\n    def start_soon(self, fn, *args) -> UUID:\n        key = uuid4()\n        # Put a placeholder in-case the result is retrieved earlier\n        self._results[key] = GatherIncomplete\n        self._task_group.start_soon(self._run_and_store, key, fn, args)\n        return key\n\n    async def start(self, fn, *args):\n        \"\"\"\n        Since `start` returns the result of `task_status.started()` but here we must\n        return the key instead, we just won't support this method for now.\n        \"\"\"\n        raise RuntimeError(\"`GatherTaskGroup` does not support `start`.\")\n\n    def get_result(self, key: UUID) -> Any:\n        result = self._results[key]\n        if result is GatherIncomplete:\n            raise GatherIncomplete(\n                \"Task is not complete. \"\n                \"Results should not be retrieved until the task group exits.\"\n            )\n        return result\n\n    async def __aenter__(self):\n        await self._task_group.__aenter__()\n        return self\n\n    async def __aexit__(self, *tb):\n        try:\n            retval = await self._task_group.__aexit__(*tb)\n            return retval\n        finally:\n            del self._task_group\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.GatherTaskGroup.start","title":"start async","text":"

    Since start returns the result of task_status.started() but here we must return the key instead, we just won't support this method for now.

    Source code in prefect/utilities/asyncutils.py
    async def start(self, fn, *args):\n    \"\"\"\n    Since `start` returns the result of `task_status.started()` but here we must\n    return the key instead, we just won't support this method for now.\n    \"\"\"\n    raise RuntimeError(\"`GatherTaskGroup` does not support `start`.\")\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.add_event_loop_shutdown_callback","title":"add_event_loop_shutdown_callback async","text":"

    Adds a callback to the given callable on event loop closure. The callable must be a coroutine function. It will be awaited when the current event loop is shutting down.

    Requires use of asyncio.run() which waits for async generator shutdown by default or explicit call of asyncio.shutdown_asyncgens(). If the application is entered with asyncio.run_until_complete() and the user calls asyncio.close() without the generator shutdown call, this will not trigger callbacks.

    asyncio does not provided any other way to clean up a resource when the event loop is about to close.

    Source code in prefect/utilities/asyncutils.py
    async def add_event_loop_shutdown_callback(coroutine_fn: Callable[[], Awaitable]):\n    \"\"\"\n    Adds a callback to the given callable on event loop closure. The callable must be\n    a coroutine function. It will be awaited when the current event loop is shutting\n    down.\n\n    Requires use of `asyncio.run()` which waits for async generator shutdown by\n    default or explicit call of `asyncio.shutdown_asyncgens()`. If the application\n    is entered with `asyncio.run_until_complete()` and the user calls\n    `asyncio.close()` without the generator shutdown call, this will not trigger\n    callbacks.\n\n    asyncio does not provided _any_ other way to clean up a resource when the event\n    loop is about to close.\n    \"\"\"\n\n    async def on_shutdown(key):\n        # It appears that EVENT_LOOP_GC_REFS is somehow being garbage collected early.\n        # We hold a reference to it so as to preserve it, at least for the lifetime of\n        # this coroutine. See the issue below for the initial report/discussion:\n        # https://github.com/PrefectHQ/prefect/issues/7709#issuecomment-1560021109\n        _ = EVENT_LOOP_GC_REFS\n        try:\n            yield\n        except GeneratorExit:\n            await coroutine_fn()\n            # Remove self from the garbage collection set\n            EVENT_LOOP_GC_REFS.pop(key)\n\n    # Create the iterator and store it in a global variable so it is not garbage\n    # collected. If the iterator is garbage collected before the event loop closes, the\n    # callback will not run. Since this function does not know the scope of the event\n    # loop that is calling it, a reference with global scope is necessary to ensure\n    # garbage collection does not occur until after event loop closure.\n    key = id(on_shutdown)\n    EVENT_LOOP_GC_REFS[key] = on_shutdown(key)\n\n    # Begin iterating so it will be cleaned up as an incomplete generator\n    await EVENT_LOOP_GC_REFS[key].__anext__()\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.create_gather_task_group","title":"create_gather_task_group","text":"

    Create a new task group that gathers results

    Source code in prefect/utilities/asyncutils.py
    def create_gather_task_group() -> GatherTaskGroup:\n    \"\"\"Create a new task group that gathers results\"\"\"\n    # This function matches the AnyIO API which uses callables since the concrete\n    # task group class depends on the async library being used and cannot be\n    # determined until runtime\n    return GatherTaskGroup(anyio.create_task_group())\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.gather","title":"gather async","text":"

    Run calls concurrently and gather their results.

    Unlike asyncio.gather this expects to receive callables not coroutines. This matches anyio semantics.

    Source code in prefect/utilities/asyncutils.py
    async def gather(*calls: Callable[[], Coroutine[Any, Any, T]]) -> List[T]:\n    \"\"\"\n    Run calls concurrently and gather their results.\n\n    Unlike `asyncio.gather` this expects to receive _callables_ not _coroutines_.\n    This matches `anyio` semantics.\n    \"\"\"\n    keys = []\n    async with create_gather_task_group() as tg:\n        for call in calls:\n            keys.append(tg.start_soon(call))\n    return [tg.get_result(key) for key in keys]\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.is_async_fn","title":"is_async_fn","text":"

    Returns True if a function returns a coroutine.

    See https://github.com/microsoft/pyright/issues/2142 for an example use

    Source code in prefect/utilities/asyncutils.py
    def is_async_fn(\n    func: Union[Callable[P, R], Callable[P, Awaitable[R]]]\n) -> TypeGuard[Callable[P, Awaitable[R]]]:\n    \"\"\"\n    Returns `True` if a function returns a coroutine.\n\n    See https://github.com/microsoft/pyright/issues/2142 for an example use\n    \"\"\"\n    while hasattr(func, \"__wrapped__\"):\n        func = func.__wrapped__\n\n    return inspect.iscoroutinefunction(func)\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.is_async_gen_fn","title":"is_async_gen_fn","text":"

    Returns True if a function is an async generator.

    Source code in prefect/utilities/asyncutils.py
    def is_async_gen_fn(func):\n    \"\"\"\n    Returns `True` if a function is an async generator.\n    \"\"\"\n    while hasattr(func, \"__wrapped__\"):\n        func = func.__wrapped__\n\n    return inspect.isasyncgenfunction(func)\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.raise_async_exception_in_thread","title":"raise_async_exception_in_thread","text":"

    Raise an exception in a thread asynchronously.

    This will not interrupt long-running system calls like sleep or wait.

    Source code in prefect/utilities/asyncutils.py
    def raise_async_exception_in_thread(thread: Thread, exc_type: Type[BaseException]):\n    \"\"\"\n    Raise an exception in a thread asynchronously.\n\n    This will not interrupt long-running system calls like `sleep` or `wait`.\n    \"\"\"\n    ret = ctypes.pythonapi.PyThreadState_SetAsyncExc(\n        ctypes.c_long(thread.ident), ctypes.py_object(exc_type)\n    )\n    if ret == 0:\n        raise ValueError(\"Thread not found.\")\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.run_async_from_worker_thread","title":"run_async_from_worker_thread","text":"

    Runs an async function in the main thread's event loop, blocking the worker thread until completion

    Source code in prefect/utilities/asyncutils.py
    def run_async_from_worker_thread(\n    __fn: Callable[..., Awaitable[T]], *args: Any, **kwargs: Any\n) -> T:\n    \"\"\"\n    Runs an async function in the main thread's event loop, blocking the worker\n    thread until completion\n    \"\"\"\n    call = partial(__fn, *args, **kwargs)\n    return anyio.from_thread.run(call)\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.run_sync_in_interruptible_worker_thread","title":"run_sync_in_interruptible_worker_thread async","text":"

    Runs a sync function in a new interruptible worker thread so that the main thread's event loop is not blocked

    Unlike the anyio function, this performs best-effort cancellation of the thread using the C API. Cancellation will not interrupt system calls like sleep.

    Source code in prefect/utilities/asyncutils.py
    async def run_sync_in_interruptible_worker_thread(\n    __fn: Callable[..., T], *args: Any, **kwargs: Any\n) -> T:\n    \"\"\"\n    Runs a sync function in a new interruptible worker thread so that the main\n    thread's event loop is not blocked\n\n    Unlike the anyio function, this performs best-effort cancellation of the\n    thread using the C API. Cancellation will not interrupt system calls like\n    `sleep`.\n    \"\"\"\n\n    class NotSet:\n        pass\n\n    thread: Thread = None\n    result = NotSet\n    event = asyncio.Event()\n    loop = asyncio.get_running_loop()\n\n    def capture_worker_thread_and_result():\n        # Captures the worker thread that AnyIO is using to execute the function so\n        # the main thread can perform actions on it\n        nonlocal thread, result\n        try:\n            thread = threading.current_thread()\n            result = __fn(*args, **kwargs)\n        except BaseException as exc:\n            result = exc\n            raise\n        finally:\n            loop.call_soon_threadsafe(event.set)\n\n    async def send_interrupt_to_thread():\n        # This task waits until the result is returned from the thread, if cancellation\n        # occurs during that time, we will raise the exception in the thread as well\n        try:\n            await event.wait()\n        except anyio.get_cancelled_exc_class():\n            # NOTE: We could send a SIGINT here which allow us to interrupt system\n            # calls but the interrupt bubbles from the child thread into the main thread\n            # and there is not a clear way to prevent it.\n            raise_async_exception_in_thread(thread, anyio.get_cancelled_exc_class())\n            raise\n\n    async with anyio.create_task_group() as tg:\n        tg.start_soon(send_interrupt_to_thread)\n        tg.start_soon(\n            partial(\n                anyio.to_thread.run_sync,\n                capture_worker_thread_and_result,\n                cancellable=True,\n                limiter=get_thread_limiter(),\n            )\n        )\n\n    assert result is not NotSet\n    return result\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.run_sync_in_worker_thread","title":"run_sync_in_worker_thread async","text":"

    Runs a sync function in a new worker thread so that the main thread's event loop is not blocked

    Unlike the anyio function, this defaults to a cancellable thread and does not allow passing arguments to the anyio function so users can pass kwargs to their function.

    Note that cancellation of threads will not result in interrupted computation, the thread may continue running \u2014 the outcome will just be ignored.

    Source code in prefect/utilities/asyncutils.py
    async def run_sync_in_worker_thread(\n    __fn: Callable[..., T], *args: Any, **kwargs: Any\n) -> T:\n    \"\"\"\n    Runs a sync function in a new worker thread so that the main thread's event loop\n    is not blocked\n\n    Unlike the anyio function, this defaults to a cancellable thread and does not allow\n    passing arguments to the anyio function so users can pass kwargs to their function.\n\n    Note that cancellation of threads will not result in interrupted computation, the\n    thread may continue running \u2014 the outcome will just be ignored.\n    \"\"\"\n    call = partial(__fn, *args, **kwargs)\n    return await anyio.to_thread.run_sync(\n        call, cancellable=True, limiter=get_thread_limiter()\n    )\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.sync","title":"sync","text":"

    Call an async function from a synchronous context. Block until completion.

    If in an asynchronous context, we will run the code in a separate loop instead of failing but a warning will be displayed since this is not recommended.

    Source code in prefect/utilities/asyncutils.py
    def sync(__async_fn: Callable[P, Awaitable[T]], *args: P.args, **kwargs: P.kwargs) -> T:\n    \"\"\"\n    Call an async function from a synchronous context. Block until completion.\n\n    If in an asynchronous context, we will run the code in a separate loop instead of\n    failing but a warning will be displayed since this is not recommended.\n    \"\"\"\n    if in_async_main_thread():\n        warnings.warn(\n            \"`sync` called from an asynchronous context; \"\n            \"you should `await` the async function directly instead.\"\n        )\n        with anyio.start_blocking_portal() as portal:\n            return portal.call(partial(__async_fn, *args, **kwargs))\n    elif in_async_worker_thread():\n        # In a sync context but we can access the event loop thread; send the async\n        # call to the parent\n        return run_async_from_worker_thread(__async_fn, *args, **kwargs)\n    else:\n        # In a sync context and there is no event loop; just create an event loop\n        # to run the async code then tear it down\n        return run_async_in_new_loop(__async_fn, *args, **kwargs)\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.sync_compatible","title":"sync_compatible","text":"

    Converts an async function into a dual async and sync function.

    When the returned function is called, we will attempt to determine the best way to enter the async function.

    • If in a thread with a running event loop, we will return the coroutine for the caller to await. This is normal async behavior.
    • If in a blocking worker thread with access to an event loop in another thread, we will submit the async method to the event loop.
    • If we cannot find an event loop, we will create a new one and run the async method then tear down the loop.
    Source code in prefect/utilities/asyncutils.py
    def sync_compatible(async_fn: T) -> T:\n    \"\"\"\n    Converts an async function into a dual async and sync function.\n\n    When the returned function is called, we will attempt to determine the best way\n    to enter the async function.\n\n    - If in a thread with a running event loop, we will return the coroutine for the\n        caller to await. This is normal async behavior.\n    - If in a blocking worker thread with access to an event loop in another thread, we\n        will submit the async method to the event loop.\n    - If we cannot find an event loop, we will create a new one and run the async method\n        then tear down the loop.\n    \"\"\"\n\n    @wraps(async_fn)\n    def coroutine_wrapper(*args, **kwargs):\n        from prefect._internal.concurrency.api import create_call, from_sync\n        from prefect._internal.concurrency.calls import get_current_call, logger\n        from prefect._internal.concurrency.event_loop import get_running_loop\n        from prefect._internal.concurrency.threads import get_global_loop\n\n        global_thread_portal = get_global_loop()\n        current_thread = threading.current_thread()\n        current_call = get_current_call()\n        current_loop = get_running_loop()\n\n        if current_thread.ident == global_thread_portal.thread.ident:\n            logger.debug(f\"{async_fn} --> return coroutine for internal await\")\n            # In the prefect async context; return the coro for us to await\n            return async_fn(*args, **kwargs)\n        elif in_async_main_thread() and (\n            not current_call or is_async_fn(current_call.fn)\n        ):\n            # In the main async context; return the coro for them to await\n            logger.debug(f\"{async_fn} --> return coroutine for user await\")\n            return async_fn(*args, **kwargs)\n        elif in_async_worker_thread():\n            # In a sync context but we can access the event loop thread; send the async\n            # call to the parent\n            return run_async_from_worker_thread(async_fn, *args, **kwargs)\n        elif current_loop is not None:\n            logger.debug(f\"{async_fn} --> run async in global loop portal\")\n            # An event loop is already present but we are in a sync context, run the\n            # call in Prefect's event loop thread\n            return from_sync.call_soon_in_loop_thread(\n                create_call(async_fn, *args, **kwargs)\n            ).result()\n        else:\n            logger.debug(f\"{async_fn} --> run async in new loop\")\n            # Run in a new event loop, but use a `Call` for nested context detection\n            call = create_call(async_fn, *args, **kwargs)\n            return call()\n\n    # TODO: This is breaking type hints on the callable... mypy is behind the curve\n    #       on argument annotations. We can still fix this for editors though.\n    if is_async_fn(async_fn):\n        wrapper = coroutine_wrapper\n    elif is_async_gen_fn(async_fn):\n        raise ValueError(\"Async generators cannot yet be marked as `sync_compatible`\")\n    else:\n        raise TypeError(\"The decorated function must be async.\")\n\n    wrapper.aio = async_fn\n    return wrapper\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/callables/","title":"callables","text":"","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables","title":"prefect.utilities.callables","text":"

    Utilities for working with Python callables.

    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.ParameterSchema","title":"ParameterSchema","text":"

    Bases: BaseModel

    Simple data model corresponding to an OpenAPI Schema.

    Source code in prefect/utilities/callables.py
    class ParameterSchema(pydantic.BaseModel):\n    \"\"\"Simple data model corresponding to an OpenAPI `Schema`.\"\"\"\n\n    title: Literal[\"Parameters\"] = \"Parameters\"\n    type: Literal[\"object\"] = \"object\"\n    properties: Dict[str, Any] = pydantic.Field(default_factory=dict)\n    required: List[str] = None\n    definitions: Dict[str, Any] = None\n\n    def dict(self, *args, **kwargs):\n        \"\"\"Exclude `None` fields by default to comply with\n        the OpenAPI spec.\n        \"\"\"\n        kwargs.setdefault(\"exclude_none\", True)\n        return super().dict(*args, **kwargs)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.ParameterSchema.dict","title":"dict","text":"

    Exclude None fields by default to comply with the OpenAPI spec.

    Source code in prefect/utilities/callables.py
    def dict(self, *args, **kwargs):\n    \"\"\"Exclude `None` fields by default to comply with\n    the OpenAPI spec.\n    \"\"\"\n    kwargs.setdefault(\"exclude_none\", True)\n    return super().dict(*args, **kwargs)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.call_with_parameters","title":"call_with_parameters","text":"

    Call a function with parameters extracted with get_call_parameters

    The function must have an identical signature to the original function or this will fail. If you need to send to a function with a different signature, extract the args/kwargs using parameters_to_positional_and_keyword directly

    Source code in prefect/utilities/callables.py
    def call_with_parameters(fn: Callable, parameters: Dict[str, Any]):\n    \"\"\"\n    Call a function with parameters extracted with `get_call_parameters`\n\n    The function _must_ have an identical signature to the original function or this\n    will fail. If you need to send to a function with a different signature, extract\n    the args/kwargs using `parameters_to_positional_and_keyword` directly\n    \"\"\"\n    args, kwargs = parameters_to_args_kwargs(fn, parameters)\n    return fn(*args, **kwargs)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.cloudpickle_wrapped_call","title":"cloudpickle_wrapped_call","text":"

    Serializes a function call using cloudpickle then returns a callable which will execute that call and return a cloudpickle serialized return value

    This is particularly useful for sending calls to libraries that only use the Python built-in pickler (e.g. anyio.to_process and multiprocessing) but may require a wider range of pickling support.

    Source code in prefect/utilities/callables.py
    def cloudpickle_wrapped_call(\n    __fn: Callable, *args: Any, **kwargs: Any\n) -> Callable[[], bytes]:\n    \"\"\"\n    Serializes a function call using cloudpickle then returns a callable which will\n    execute that call and return a cloudpickle serialized return value\n\n    This is particularly useful for sending calls to libraries that only use the Python\n    built-in pickler (e.g. `anyio.to_process` and `multiprocessing`) but may require\n    a wider range of pickling support.\n    \"\"\"\n    payload = cloudpickle.dumps((__fn, args, kwargs))\n    return partial(_run_serialized_call, payload)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.collapse_variadic_parameters","title":"collapse_variadic_parameters","text":"

    Given a parameter dictionary, move any parameters stored not present in the signature into the variadic keyword argument.

    Example:

    ```python\ndef foo(a, b, **kwargs):\n    pass\n\nparameters = {\"a\": 1, \"b\": 2, \"c\": 3, \"d\": 4}\ncollapse_variadic_parameters(foo, parameters)\n# {\"a\": 1, \"b\": 2, \"kwargs\": {\"c\": 3, \"d\": 4}}\n```\n
    Source code in prefect/utilities/callables.py
    def collapse_variadic_parameters(\n    fn: Callable, parameters: Dict[str, Any]\n) -> Dict[str, Any]:\n    \"\"\"\n    Given a parameter dictionary, move any parameters stored not present in the\n    signature into the variadic keyword argument.\n\n    Example:\n\n        ```python\n        def foo(a, b, **kwargs):\n            pass\n\n        parameters = {\"a\": 1, \"b\": 2, \"c\": 3, \"d\": 4}\n        collapse_variadic_parameters(foo, parameters)\n        # {\"a\": 1, \"b\": 2, \"kwargs\": {\"c\": 3, \"d\": 4}}\n        ```\n    \"\"\"\n    signature_parameters = inspect.signature(fn).parameters\n    variadic_key = None\n    for key, parameter in signature_parameters.items():\n        if parameter.kind == parameter.VAR_KEYWORD:\n            variadic_key = key\n            break\n\n    missing_parameters = set(parameters.keys()) - set(signature_parameters.keys())\n\n    if not variadic_key and missing_parameters:\n        raise ValueError(\n            f\"Signature for {fn} does not include any variadic keyword argument \"\n            \"but parameters were given that are not present in the signature.\"\n        )\n\n    if variadic_key and not missing_parameters:\n        # variadic key is present but no missing parameters, return parameters unchanged\n        return parameters\n\n    new_parameters = parameters.copy()\n    if variadic_key:\n        new_parameters[variadic_key] = {}\n\n    for key in missing_parameters:\n        new_parameters[variadic_key][key] = new_parameters.pop(key)\n\n    return new_parameters\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.explode_variadic_parameter","title":"explode_variadic_parameter","text":"

    Given a parameter dictionary, move any parameters stored in a variadic keyword argument parameter (i.e. **kwargs) into the top level.

    Example:

    ```python\ndef foo(a, b, **kwargs):\n    pass\n\nparameters = {\"a\": 1, \"b\": 2, \"kwargs\": {\"c\": 3, \"d\": 4}}\nexplode_variadic_parameter(foo, parameters)\n# {\"a\": 1, \"b\": 2, \"c\": 3, \"d\": 4}\n```\n
    Source code in prefect/utilities/callables.py
    def explode_variadic_parameter(\n    fn: Callable, parameters: Dict[str, Any]\n) -> Dict[str, Any]:\n    \"\"\"\n    Given a parameter dictionary, move any parameters stored in a variadic keyword\n    argument parameter (i.e. **kwargs) into the top level.\n\n    Example:\n\n        ```python\n        def foo(a, b, **kwargs):\n            pass\n\n        parameters = {\"a\": 1, \"b\": 2, \"kwargs\": {\"c\": 3, \"d\": 4}}\n        explode_variadic_parameter(foo, parameters)\n        # {\"a\": 1, \"b\": 2, \"c\": 3, \"d\": 4}\n        ```\n    \"\"\"\n    variadic_key = None\n    for key, parameter in inspect.signature(fn).parameters.items():\n        if parameter.kind == parameter.VAR_KEYWORD:\n            variadic_key = key\n            break\n\n    if not variadic_key:\n        return parameters\n\n    new_parameters = parameters.copy()\n    for key, value in new_parameters.pop(variadic_key, {}).items():\n        new_parameters[key] = value\n\n    return new_parameters\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.get_call_parameters","title":"get_call_parameters","text":"

    Bind a call to a function to get parameter/value mapping. Default values on the signature will be included if not overridden.

    Raises a ParameterBindError if the arguments/kwargs are not valid for the function

    Source code in prefect/utilities/callables.py
    def get_call_parameters(\n    fn: Callable,\n    call_args: Tuple[Any, ...],\n    call_kwargs: Dict[str, Any],\n    apply_defaults: bool = True,\n) -> Dict[str, Any]:\n    \"\"\"\n    Bind a call to a function to get parameter/value mapping. Default values on the\n    signature will be included if not overridden.\n\n    Raises a ParameterBindError if the arguments/kwargs are not valid for the function\n    \"\"\"\n    try:\n        bound_signature = inspect.signature(fn).bind(*call_args, **call_kwargs)\n    except TypeError as exc:\n        raise ParameterBindError.from_bind_failure(fn, exc, call_args, call_kwargs)\n\n    if apply_defaults:\n        bound_signature.apply_defaults()\n\n    # We cast from `OrderedDict` to `dict` because Dask will not convert futures in an\n    # ordered dictionary to values during execution; this is the default behavior in\n    # Python 3.9 anyway.\n    return dict(bound_signature.arguments)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.get_parameter_defaults","title":"get_parameter_defaults","text":"

    Get default parameter values for a callable.

    Source code in prefect/utilities/callables.py
    def get_parameter_defaults(\n    fn: Callable,\n) -> Dict[str, Any]:\n    \"\"\"\n    Get default parameter values for a callable.\n    \"\"\"\n    signature = inspect.signature(fn)\n\n    parameter_defaults = {}\n\n    for name, param in signature.parameters.items():\n        if param.default is not signature.empty:\n            parameter_defaults[name] = param.default\n\n    return parameter_defaults\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.parameter_docstrings","title":"parameter_docstrings","text":"

    Given a docstring in Google docstring format, parse the parameter section and return a dictionary that maps parameter names to docstring.

    Parameters:

    Name Type Description Default docstring Optional[str]

    The function's docstring.

    required

    Returns:

    Type Description Dict[str, str]

    Mapping from parameter names to docstrings.

    Source code in prefect/utilities/callables.py
    def parameter_docstrings(docstring: Optional[str]) -> Dict[str, str]:\n    \"\"\"\n    Given a docstring in Google docstring format, parse the parameter section\n    and return a dictionary that maps parameter names to docstring.\n\n    Args:\n        docstring: The function's docstring.\n\n    Returns:\n        Mapping from parameter names to docstrings.\n    \"\"\"\n    param_docstrings = {}\n\n    if not docstring:\n        return param_docstrings\n\n    with disable_logger(\"griffe.docstrings.google\"), disable_logger(\n        \"griffe.agents.nodes\"\n    ):\n        parsed = parse(Docstring(docstring), Parser.google)\n        for section in parsed:\n            if section.kind != DocstringSectionKind.parameters:\n                continue\n            param_docstrings = {\n                parameter.name: parameter.description for parameter in section.value\n            }\n\n    return param_docstrings\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.parameter_schema","title":"parameter_schema","text":"

    Given a function, generates an OpenAPI-compatible description of the function's arguments, including: - name - typing information - whether it is required - a default value - additional constraints (like possible enum values)

    Parameters:

    Name Type Description Default fn Callable

    The function whose arguments will be serialized

    required

    Returns:

    Name Type Description ParameterSchema ParameterSchema

    the argument schema

    Source code in prefect/utilities/callables.py
    def parameter_schema(fn: Callable) -> ParameterSchema:\n    \"\"\"Given a function, generates an OpenAPI-compatible description\n    of the function's arguments, including:\n        - name\n        - typing information\n        - whether it is required\n        - a default value\n        - additional constraints (like possible enum values)\n\n    Args:\n        fn (Callable): The function whose arguments will be serialized\n\n    Returns:\n        ParameterSchema: the argument schema\n    \"\"\"\n    signature = inspect.signature(fn)\n    model_fields = {}\n    aliases = {}\n    docstrings = parameter_docstrings(inspect.getdoc(fn))\n\n    class ModelConfig:\n        arbitrary_types_allowed = True\n\n    if HAS_PYDANTIC_V2 and has_v2_model_as_param(signature):\n        create_schema = create_v2_schema\n        process_params = process_v2_params\n    else:\n        create_schema = create_v1_schema\n        process_params = process_v1_params\n\n    for position, param in enumerate(signature.parameters.values()):\n        name, type_, field = process_params(\n            param, position=position, docstrings=docstrings, aliases=aliases\n        )\n        # Generate a Pydantic model at each step so we can check if this parameter\n        # type supports schema generation\n        try:\n            create_schema(\n                \"CheckParameter\", model_cfg=ModelConfig, **{name: (type_, field)}\n            )\n        except ValueError:\n            # This field's type is not valid for schema creation, update it to `Any`\n            type_ = Any\n        model_fields[name] = (type_, field)\n\n    # Generate the final model and schema\n    schema = create_schema(\"Parameters\", model_cfg=ModelConfig, **model_fields)\n    return ParameterSchema(**schema)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.parameters_to_args_kwargs","title":"parameters_to_args_kwargs","text":"

    Convert a parameters dictionary to positional and keyword arguments

    The function must have an identical signature to the original function or this will return an empty tuple and dict.

    Source code in prefect/utilities/callables.py
    def parameters_to_args_kwargs(\n    fn: Callable,\n    parameters: Dict[str, Any],\n) -> Tuple[Tuple[Any, ...], Dict[str, Any]]:\n    \"\"\"\n    Convert a `parameters` dictionary to positional and keyword arguments\n\n    The function _must_ have an identical signature to the original function or this\n    will return an empty tuple and dict.\n    \"\"\"\n    function_params = dict(inspect.signature(fn).parameters).keys()\n    # Check for parameters that are not present in the function signature\n    unknown_params = parameters.keys() - function_params\n    if unknown_params:\n        raise SignatureMismatchError.from_bad_params(\n            list(function_params), list(parameters.keys())\n        )\n    bound_signature = inspect.signature(fn).bind_partial()\n    bound_signature.arguments = parameters\n\n    return bound_signature.args, bound_signature.kwargs\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.raise_for_reserved_arguments","title":"raise_for_reserved_arguments","text":"

    Raise a ReservedArgumentError if fn has any parameters that conflict with the names contained in reserved_arguments.

    Source code in prefect/utilities/callables.py
    def raise_for_reserved_arguments(fn: Callable, reserved_arguments: Iterable[str]):\n    \"\"\"Raise a ReservedArgumentError if `fn` has any parameters that conflict\n    with the names contained in `reserved_arguments`.\"\"\"\n    function_paremeters = inspect.signature(fn).parameters\n\n    for argument in reserved_arguments:\n        if argument in function_paremeters:\n            raise ReservedArgumentError(\n                f\"{argument!r} is a reserved argument name and cannot be used.\"\n            )\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/collections/","title":"collections","text":"","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections","title":"prefect.utilities.collections","text":"

    Utilities for extensions of and operations on Python collections.

    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.AutoEnum","title":"AutoEnum","text":"

    Bases: str, Enum

    An enum class that automatically generates value from variable names.

    This guards against common errors where variable names are updated but values are not.

    In addition, because AutoEnums inherit from str, they are automatically JSON-serializable.

    See https://docs.python.org/3/library/enum.html#using-automatic-values

    Example
    class MyEnum(AutoEnum):\n    RED = AutoEnum.auto() # equivalent to RED = 'RED'\n    BLUE = AutoEnum.auto() # equivalent to BLUE = 'BLUE'\n
    Source code in prefect/utilities/collections.py
    class AutoEnum(str, Enum):\n    \"\"\"\n    An enum class that automatically generates value from variable names.\n\n    This guards against common errors where variable names are updated but values are\n    not.\n\n    In addition, because AutoEnums inherit from `str`, they are automatically\n    JSON-serializable.\n\n    See https://docs.python.org/3/library/enum.html#using-automatic-values\n\n    Example:\n        ```python\n        class MyEnum(AutoEnum):\n            RED = AutoEnum.auto() # equivalent to RED = 'RED'\n            BLUE = AutoEnum.auto() # equivalent to BLUE = 'BLUE'\n        ```\n    \"\"\"\n\n    def _generate_next_value_(name, start, count, last_values):\n        return name\n\n    @staticmethod\n    def auto():\n        \"\"\"\n        Exposes `enum.auto()` to avoid requiring a second import to use `AutoEnum`\n        \"\"\"\n        return auto()\n\n    def __repr__(self) -> str:\n        return f\"{type(self).__name__}.{self.value}\"\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.AutoEnum.auto","title":"auto staticmethod","text":"

    Exposes enum.auto() to avoid requiring a second import to use AutoEnum

    Source code in prefect/utilities/collections.py
    @staticmethod\ndef auto():\n    \"\"\"\n    Exposes `enum.auto()` to avoid requiring a second import to use `AutoEnum`\n    \"\"\"\n    return auto()\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.StopVisiting","title":"StopVisiting","text":"

    Bases: BaseException

    A special exception used to stop recursive visits in visit_collection.

    When raised, the expression is returned without modification and recursive visits in that path will end.

    Source code in prefect/utilities/collections.py
    class StopVisiting(BaseException):\n    \"\"\"\n    A special exception used to stop recursive visits in `visit_collection`.\n\n    When raised, the expression is returned without modification and recursive visits\n    in that path will end.\n    \"\"\"\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.batched_iterable","title":"batched_iterable","text":"

    Yield batches of a certain size from an iterable

    Parameters:

    Name Type Description Default iterable Iterable

    An iterable

    required size int

    The batch size to return

    required

    Yields:

    Name Type Description tuple T

    A batch of the iterable

    Source code in prefect/utilities/collections.py
    def batched_iterable(iterable: Iterable[T], size: int) -> Iterator[Tuple[T, ...]]:\n    \"\"\"\n    Yield batches of a certain size from an iterable\n\n    Args:\n        iterable (Iterable): An iterable\n        size (int): The batch size to return\n\n    Yields:\n        tuple: A batch of the iterable\n    \"\"\"\n    it = iter(iterable)\n    while True:\n        batch = tuple(itertools.islice(it, size))\n        if not batch:\n            break\n        yield batch\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.dict_to_flatdict","title":"dict_to_flatdict","text":"

    Converts a (nested) dictionary to a flattened representation.

    Each key of the flat dict will be a CompoundKey tuple containing the \"chain of keys\" for the corresponding value.

    Parameters:

    Name Type Description Default dct dict

    The dictionary to flatten

    required _parent Tuple

    The current parent for recursion

    None

    Returns:

    Type Description Dict[Tuple[KT, ...], Any]

    A flattened dict of the same type as dct

    Source code in prefect/utilities/collections.py
    def dict_to_flatdict(\n    dct: Dict[KT, Union[Any, Dict[KT, Any]]], _parent: Tuple[KT, ...] = None\n) -> Dict[Tuple[KT, ...], Any]:\n    \"\"\"Converts a (nested) dictionary to a flattened representation.\n\n    Each key of the flat dict will be a CompoundKey tuple containing the \"chain of keys\"\n    for the corresponding value.\n\n    Args:\n        dct (dict): The dictionary to flatten\n        _parent (Tuple, optional): The current parent for recursion\n\n    Returns:\n        A flattened dict of the same type as dct\n    \"\"\"\n    typ = cast(Type[Dict[Tuple[KT, ...], Any]], type(dct))\n    items: List[Tuple[Tuple[KT, ...], Any]] = []\n    parent = _parent or tuple()\n\n    for k, v in dct.items():\n        k_parent = tuple(parent + (k,))\n        # if v is a non-empty dict, recurse\n        if isinstance(v, dict) and v:\n            items.extend(dict_to_flatdict(v, _parent=k_parent).items())\n        else:\n            items.append((k_parent, v))\n    return typ(items)\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.extract_instances","title":"extract_instances","text":"

    Extract objects from a file and returns a dict of type -> instances

    Parameters:

    Name Type Description Default objects Iterable

    An iterable of objects

    required types Union[Type[T], Tuple[Type[T], ...]]

    A type or tuple of types to extract, defaults to all objects

    object

    Returns:

    Type Description Union[List[T], Dict[Type[T], T]]

    If a single type is given: a list of instances of that type

    Union[List[T], Dict[Type[T], T]]

    If a tuple of types is given: a mapping of type to a list of instances

    Source code in prefect/utilities/collections.py
    def extract_instances(\n    objects: Iterable,\n    types: Union[Type[T], Tuple[Type[T], ...]] = object,\n) -> Union[List[T], Dict[Type[T], T]]:\n    \"\"\"\n    Extract objects from a file and returns a dict of type -> instances\n\n    Args:\n        objects: An iterable of objects\n        types: A type or tuple of types to extract, defaults to all objects\n\n    Returns:\n        If a single type is given: a list of instances of that type\n        If a tuple of types is given: a mapping of type to a list of instances\n    \"\"\"\n    types = ensure_iterable(types)\n\n    # Create a mapping of type -> instance from the exec values\n    ret = defaultdict(list)\n\n    for o in objects:\n        # We iterate here so that the key is the passed type rather than type(o)\n        for type_ in types:\n            if isinstance(o, type_):\n                ret[type_].append(o)\n\n    if len(types) == 1:\n        return ret[types[0]]\n\n    return ret\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.flatdict_to_dict","title":"flatdict_to_dict","text":"

    Converts a flattened dictionary back to a nested dictionary.

    Parameters:

    Name Type Description Default dct dict

    The dictionary to be nested. Each key should be a tuple of keys as generated by dict_to_flatdict

    required

    Returns A nested dict of the same type as dct

    Source code in prefect/utilities/collections.py
    def flatdict_to_dict(\n    dct: Dict[Tuple[KT, ...], VT]\n) -> Dict[KT, Union[VT, Dict[KT, VT]]]:\n    \"\"\"Converts a flattened dictionary back to a nested dictionary.\n\n    Args:\n        dct (dict): The dictionary to be nested. Each key should be a tuple of keys\n            as generated by `dict_to_flatdict`\n\n    Returns\n        A nested dict of the same type as dct\n    \"\"\"\n    typ = type(dct)\n    result = cast(Dict[KT, Union[VT, Dict[KT, VT]]], typ())\n    for key_tuple, value in dct.items():\n        current_dict = result\n        for prefix_key in key_tuple[:-1]:\n            # Build nested dictionaries up for the current key tuple\n            # Use `setdefault` in case the nested dict has already been created\n            current_dict = current_dict.setdefault(prefix_key, typ())  # type: ignore\n        # Set the value\n        current_dict[key_tuple[-1]] = value\n\n    return result\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.get_from_dict","title":"get_from_dict","text":"

    Fetch a value from a nested dictionary or list using a sequence of keys.

    This function allows to fetch a value from a deeply nested structure of dictionaries and lists using either a dot-separated string or a list of keys. If a requested key does not exist, the function returns the provided default value.

    Parameters:

    Name Type Description Default dct Dict

    The nested dictionary or list from which to fetch the value.

    required keys Union[str, List[str]]

    The sequence of keys to use for access. Can be a dot-separated string or a list of keys. List indices can be included in the sequence as either integer keys or as string indices in square brackets.

    required default Any

    The default value to return if the requested key path does not exist. Defaults to None.

    None

    Returns:

    Type Description Any

    The fetched value if the key exists, or the default value if it does not.

    get_from_dict({'a': {'b': {'c': [1, 2, 3, 4]}}}, 'a.b.c[1]') 2 get_from_dict({'a': {'b': [0, {'c': [1, 2]}]}}, ['a', 'b', 1, 'c', 1]) 2 get_from_dict({'a': {'b': [0, {'c': [1, 2]}]}}, 'a.b.1.c.2', 'default') 'default'

    Source code in prefect/utilities/collections.py
    def get_from_dict(dct: Dict, keys: Union[str, List[str]], default: Any = None) -> Any:\n    \"\"\"\n    Fetch a value from a nested dictionary or list using a sequence of keys.\n\n    This function allows to fetch a value from a deeply nested structure\n    of dictionaries and lists using either a dot-separated string or a list\n    of keys. If a requested key does not exist, the function returns the\n    provided default value.\n\n    Args:\n        dct: The nested dictionary or list from which to fetch the value.\n        keys: The sequence of keys to use for access. Can be a\n            dot-separated string or a list of keys. List indices can be included\n            in the sequence as either integer keys or as string indices in square\n            brackets.\n        default: The default value to return if the requested key path does not\n            exist. Defaults to None.\n\n    Returns:\n        The fetched value if the key exists, or the default value if it does not.\n\n    Examples:\n    >>> get_from_dict({'a': {'b': {'c': [1, 2, 3, 4]}}}, 'a.b.c[1]')\n    2\n    >>> get_from_dict({'a': {'b': [0, {'c': [1, 2]}]}}, ['a', 'b', 1, 'c', 1])\n    2\n    >>> get_from_dict({'a': {'b': [0, {'c': [1, 2]}]}}, 'a.b.1.c.2', 'default')\n    'default'\n    \"\"\"\n    if isinstance(keys, str):\n        keys = keys.replace(\"[\", \".\").replace(\"]\", \"\").split(\".\")\n    try:\n        for key in keys:\n            try:\n                # Try to cast to int to handle list indices\n                key = int(key)\n            except ValueError:\n                # If it's not an int, use the key as-is\n                # for dict lookup\n                pass\n            dct = dct[key]\n        return dct\n    except (TypeError, KeyError, IndexError):\n        return default\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.isiterable","title":"isiterable","text":"

    Return a boolean indicating if an object is iterable.

    Excludes types that are iterable but typically used as singletons: - str - bytes - IO objects

    Source code in prefect/utilities/collections.py
    def isiterable(obj: Any) -> bool:\n    \"\"\"\n    Return a boolean indicating if an object is iterable.\n\n    Excludes types that are iterable but typically used as singletons:\n    - str\n    - bytes\n    - IO objects\n    \"\"\"\n    try:\n        iter(obj)\n    except TypeError:\n        return False\n    else:\n        return not isinstance(obj, (str, bytes, io.IOBase))\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.remove_nested_keys","title":"remove_nested_keys","text":"

    Recurses a dictionary returns a copy without all keys that match an entry in key_to_remove. Return obj unchanged if not a dictionary.

    Parameters:

    Name Type Description Default keys_to_remove List[Hashable]

    A list of keys to remove from obj obj: The object to remove keys from.

    required

    Returns:

    Type Description

    obj without keys matching an entry in keys_to_remove if obj is a dictionary. obj if obj is not a dictionary.

    Source code in prefect/utilities/collections.py
    def remove_nested_keys(keys_to_remove: List[Hashable], obj):\n    \"\"\"\n    Recurses a dictionary returns a copy without all keys that match an entry in\n    `key_to_remove`. Return `obj` unchanged if not a dictionary.\n\n    Args:\n        keys_to_remove: A list of keys to remove from obj obj: The object to remove keys\n            from.\n\n    Returns:\n        `obj` without keys matching an entry in `keys_to_remove` if `obj` is a\n            dictionary. `obj` if `obj` is not a dictionary.\n    \"\"\"\n    if not isinstance(obj, dict):\n        return obj\n    return {\n        key: remove_nested_keys(keys_to_remove, value)\n        for key, value in obj.items()\n        if key not in keys_to_remove\n    }\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.visit_collection","title":"visit_collection","text":"

    This function visits every element of an arbitrary Python collection. If an element is a Python collection, it will be visited recursively. If an element is not a collection, visit_fn will be called with the element. The return value of visit_fn can be used to alter the element if return_data is set.

    Note that when using return_data a copy of each collection is created to avoid mutating the original object. This may have significant performance penalties and should only be used if you intend to transform the collection.

    Supported types: - List - Tuple - Set - Dict (note: keys are also visited recursively) - Dataclass - Pydantic model - Prefect annotations

    Parameters:

    Name Type Description Default expr Any

    a Python object or expression

    required visit_fn Callable[[Any], Awaitable[Any]]

    an async function that will be applied to every non-collection element of expr.

    required return_data bool

    if True, a copy of expr containing data modified by visit_fn will be returned. This is slower than return_data=False (the default).

    False max_depth int

    Controls the depth of recursive visitation. If set to zero, no recursion will occur. If set to a positive integer N, visitation will only descend to N layers deep. If set to any negative integer, no limit will be enforced and recursion will continue until terminal items are reached. By default, recursion is unlimited.

    -1 context Optional[dict]

    An optional dictionary. If passed, the context will be sent to each call to the visit_fn. The context can be mutated by each visitor and will be available for later visits to expressions at the given depth. Values will not be available \"up\" a level from a given expression.

    The context will be automatically populated with an 'annotation' key when visiting collections within a BaseAnnotation type. This requires the caller to pass context={} and will not be activated by default.

    None remove_annotations bool

    If set, annotations will be replaced by their contents. By default, annotations are preserved but their contents are visited.

    False Source code in prefect/utilities/collections.py
    def visit_collection(\n    expr,\n    visit_fn: Callable[[Any], Any],\n    return_data: bool = False,\n    max_depth: int = -1,\n    context: Optional[dict] = None,\n    remove_annotations: bool = False,\n):\n    \"\"\"\n    This function visits every element of an arbitrary Python collection. If an element\n    is a Python collection, it will be visited recursively. If an element is not a\n    collection, `visit_fn` will be called with the element. The return value of\n    `visit_fn` can be used to alter the element if `return_data` is set.\n\n    Note that when using `return_data` a copy of each collection is created to avoid\n    mutating the original object. This may have significant performance penalties and\n    should only be used if you intend to transform the collection.\n\n    Supported types:\n    - List\n    - Tuple\n    - Set\n    - Dict (note: keys are also visited recursively)\n    - Dataclass\n    - Pydantic model\n    - Prefect annotations\n\n    Args:\n        expr (Any): a Python object or expression\n        visit_fn (Callable[[Any], Awaitable[Any]]): an async function that\n            will be applied to every non-collection element of expr.\n        return_data (bool): if `True`, a copy of `expr` containing data modified\n            by `visit_fn` will be returned. This is slower than `return_data=False`\n            (the default).\n        max_depth: Controls the depth of recursive visitation. If set to zero, no\n            recursion will occur. If set to a positive integer N, visitation will only\n            descend to N layers deep. If set to any negative integer, no limit will be\n            enforced and recursion will continue until terminal items are reached. By\n            default, recursion is unlimited.\n        context: An optional dictionary. If passed, the context will be sent to each\n            call to the `visit_fn`. The context can be mutated by each visitor and will\n            be available for later visits to expressions at the given depth. Values\n            will not be available \"up\" a level from a given expression.\n\n            The context will be automatically populated with an 'annotation' key when\n            visiting collections within a `BaseAnnotation` type. This requires the\n            caller to pass `context={}` and will not be activated by default.\n        remove_annotations: If set, annotations will be replaced by their contents. By\n            default, annotations are preserved but their contents are visited.\n    \"\"\"\n\n    def visit_nested(expr):\n        # Utility for a recursive call, preserving options and updating the depth.\n        return visit_collection(\n            expr,\n            visit_fn=visit_fn,\n            return_data=return_data,\n            remove_annotations=remove_annotations,\n            max_depth=max_depth - 1,\n            # Copy the context on nested calls so it does not \"propagate up\"\n            context=context.copy() if context is not None else None,\n        )\n\n    def visit_expression(expr):\n        if context is not None:\n            return visit_fn(expr, context)\n        else:\n            return visit_fn(expr)\n\n    # Visit every expression\n    try:\n        result = visit_expression(expr)\n    except StopVisiting:\n        max_depth = 0\n        result = expr\n\n    if return_data:\n        # Only mutate the expression while returning data, otherwise it could be null\n        expr = result\n\n    # Then, visit every child of the expression recursively\n\n    # If we have reached the maximum depth, do not perform any recursion\n    if max_depth == 0:\n        return result if return_data else None\n\n    # Get the expression type; treat iterators like lists\n    typ = list if isinstance(expr, IteratorABC) and isiterable(expr) else type(expr)\n    typ = cast(type, typ)  # mypy treats this as 'object' otherwise and complains\n\n    # Then visit every item in the expression if it is a collection\n    if isinstance(expr, Mock):\n        # Do not attempt to recurse into mock objects\n        result = expr\n\n    elif isinstance(expr, BaseAnnotation):\n        if context is not None:\n            context[\"annotation\"] = expr\n        value = visit_nested(expr.unwrap())\n\n        if remove_annotations:\n            result = value if return_data else None\n        else:\n            result = expr.rewrap(value) if return_data else None\n\n    elif typ in (list, tuple, set):\n        items = [visit_nested(o) for o in expr]\n        result = typ(items) if return_data else None\n\n    elif typ in (dict, OrderedDict):\n        assert isinstance(expr, (dict, OrderedDict))  # typecheck assertion\n        items = [(visit_nested(k), visit_nested(v)) for k, v in expr.items()]\n        result = typ(items) if return_data else None\n\n    elif is_dataclass(expr) and not isinstance(expr, type):\n        values = [visit_nested(getattr(expr, f.name)) for f in fields(expr)]\n        items = {field.name: value for field, value in zip(fields(expr), values)}\n        result = typ(**items) if return_data else None\n\n    elif isinstance(expr, pydantic.BaseModel):\n        # NOTE: This implementation *does not* traverse private attributes\n        # Pydantic does not expose extras in `__fields__` so we use `__fields_set__`\n        # as well to get all of the relevant attributes\n        # Check for presence of attrs even if they're in the field set due to pydantic#4916\n        model_fields = {\n            f for f in expr.__fields_set__.union(expr.__fields__) if hasattr(expr, f)\n        }\n        items = [visit_nested(getattr(expr, key)) for key in model_fields]\n\n        if return_data:\n            # Collect fields with aliases so reconstruction can use the correct field name\n            aliases = {\n                key: value.alias\n                for key, value in expr.__fields__.items()\n                if value.has_alias\n            }\n\n            model_instance = typ(\n                **{\n                    aliases.get(key) or key: value\n                    for key, value in zip(model_fields, items)\n                }\n            )\n\n            # Private attributes are not included in `__fields_set__` but we do not want\n            # to drop them from the model so we restore them after constructing a new\n            # model\n            for attr in expr.__private_attributes__:\n                # Use `object.__setattr__` to avoid errors on immutable models\n                object.__setattr__(model_instance, attr, getattr(expr, attr))\n\n            # Preserve data about which fields were explicitly set on the original model\n            object.__setattr__(model_instance, \"__fields_set__\", expr.__fields_set__)\n            result = model_instance\n        else:\n            result = None\n\n    else:\n        result = result if return_data else None\n\n    return result\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/compat/","title":"compat","text":"","tags":["Python API","utilities","compatibility"]},{"location":"api-ref/prefect/utilities/compat/#prefect.utilities.compat","title":"prefect.utilities.compat","text":"

    Utilities for Python version compatibility

    ","tags":["Python API","utilities","compatibility"]},{"location":"api-ref/prefect/utilities/context/","title":"context","text":"","tags":["Python API","utilities","context"]},{"location":"api-ref/prefect/utilities/context/#prefect.utilities.context","title":"prefect.utilities.context","text":"","tags":["Python API","utilities","context"]},{"location":"api-ref/prefect/utilities/dispatch/","title":"dispatch","text":"","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch","title":"prefect.utilities.dispatch","text":"

    Provides methods for performing dynamic dispatch for actions on base type to one of its subtypes.

    Example:

    @register_base_type\nclass Base:\n    @classmethod\n    def __dispatch_key__(cls):\n        return cls.__name__.lower()\n\n\nclass Foo(Base):\n    ...\n\nkey = get_dispatch_key(Foo)  # 'foo'\nlookup_type(Base, key) # Foo\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.get_dispatch_key","title":"get_dispatch_key","text":"

    Retrieve the unique dispatch key for a class type or instance.

    This key is defined at the __dispatch_key__ attribute. If it is a callable, it will be resolved.

    If allow_missing is False, an exception will be raised if the attribute is not defined or the key is null. If True, None will be returned in these cases.

    Source code in prefect/utilities/dispatch.py
    def get_dispatch_key(\n    cls_or_instance: Any, allow_missing: bool = False\n) -> Optional[str]:\n    \"\"\"\n    Retrieve the unique dispatch key for a class type or instance.\n\n    This key is defined at the `__dispatch_key__` attribute. If it is a callable, it\n    will be resolved.\n\n    If `allow_missing` is `False`, an exception will be raised if the attribute is not\n    defined or the key is null. If `True`, `None` will be returned in these cases.\n    \"\"\"\n    dispatch_key = getattr(cls_or_instance, \"__dispatch_key__\", None)\n\n    type_name = (\n        cls_or_instance.__name__\n        if isinstance(cls_or_instance, type)\n        else type(cls_or_instance).__name__\n    )\n\n    if dispatch_key is None:\n        if allow_missing:\n            return None\n        raise ValueError(\n            f\"Type {type_name!r} does not define a value for \"\n            \"'__dispatch_key__' which is required for registry lookup.\"\n        )\n\n    if callable(dispatch_key):\n        dispatch_key = dispatch_key()\n\n    if allow_missing and dispatch_key is None:\n        return None\n\n    if not isinstance(dispatch_key, str):\n        raise TypeError(\n            f\"Type {type_name!r} has a '__dispatch_key__' of type \"\n            f\"{type(dispatch_key).__name__} but a type of 'str' is required.\"\n        )\n\n    return dispatch_key\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.get_registry_for_type","title":"get_registry_for_type","text":"

    Get the first matching registry for a class or any of its base classes.

    If not found, None is returned.

    Source code in prefect/utilities/dispatch.py
    def get_registry_for_type(cls: T) -> Optional[Dict[str, T]]:\n    \"\"\"\n    Get the first matching registry for a class or any of its base classes.\n\n    If not found, `None` is returned.\n    \"\"\"\n    return next(\n        filter(\n            lambda registry: registry is not None,\n            (_TYPE_REGISTRIES.get(cls) for cls in cls.mro()),\n        ),\n        None,\n    )\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.lookup_type","title":"lookup_type","text":"

    Look up a dispatch key in the type registry for the given class.

    Source code in prefect/utilities/dispatch.py
    def lookup_type(cls: T, dispatch_key: str) -> T:\n    \"\"\"\n    Look up a dispatch key in the type registry for the given class.\n    \"\"\"\n    # Get the first matching registry for the class or one of its bases\n    registry = get_registry_for_type(cls)\n\n    # Look up this type in the registry\n    subcls = registry.get(dispatch_key)\n\n    if subcls is None:\n        raise KeyError(\n            f\"No class found for dispatch key {dispatch_key!r} in registry for type \"\n            f\"{cls.__name__!r}.\"\n        )\n\n    return subcls\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.register_base_type","title":"register_base_type","text":"

    Register a base type allowing child types to be registered for dispatch with register_type.

    The base class may or may not define a __dispatch_key__ to allow lookups of the base type.

    Source code in prefect/utilities/dispatch.py
    def register_base_type(cls: T) -> T:\n    \"\"\"\n    Register a base type allowing child types to be registered for dispatch with\n    `register_type`.\n\n    The base class may or may not define a `__dispatch_key__` to allow lookups of the\n    base type.\n    \"\"\"\n    registry = _TYPE_REGISTRIES.setdefault(cls, {})\n    base_key = get_dispatch_key(cls, allow_missing=True)\n    if base_key is not None:\n        registry[base_key] = cls\n\n    # Add automatic subtype registration\n    cls.__init_subclass_original__ = getattr(cls, \"__init_subclass__\")\n    cls.__init_subclass__ = _register_subclass_of_base_type\n\n    return cls\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.register_type","title":"register_type","text":"

    Register a type for lookup with dispatch.

    The type or one of its parents must define a unique __dispatch_key__.

    One of the classes base types must be registered using register_base_type.

    Source code in prefect/utilities/dispatch.py
    def register_type(cls: T) -> T:\n    \"\"\"\n    Register a type for lookup with dispatch.\n\n    The type or one of its parents must define a unique `__dispatch_key__`.\n\n    One of the classes base types must be registered using `register_base_type`.\n    \"\"\"\n    # Lookup the registry for this type\n    registry = get_registry_for_type(cls)\n\n    # Check if a base type is registered\n    if registry is None:\n        # Include a description of registered base types\n        known = \", \".join(repr(base.__name__) for base in _TYPE_REGISTRIES)\n        known_message = (\n            f\" Did you mean to inherit from one of the following known types: {known}.\"\n            if known\n            else \"\"\n        )\n\n        # And a list of all base types for the type they tried to register\n        bases = \", \".join(\n            repr(base.__name__) for base in cls.mro() if base not in (object, cls)\n        )\n\n        raise ValueError(\n            f\"No registry found for type {cls.__name__!r} with bases {bases}.\"\n            + known_message\n        )\n\n    key = get_dispatch_key(cls)\n    existing_value = registry.get(key)\n    if existing_value is not None and id(existing_value) != id(cls):\n        # Get line numbers for debugging\n        file = inspect.getsourcefile(cls)\n        line_number = inspect.getsourcelines(cls)[1]\n        existing_file = inspect.getsourcefile(existing_value)\n        existing_line_number = inspect.getsourcelines(existing_value)[1]\n        warnings.warn(\n            f\"Type {cls.__name__!r} at {file}:{line_number} has key {key!r} that \"\n            f\"matches existing registered type {existing_value.__name__!r} from \"\n            f\"{existing_file}:{existing_line_number}. The existing type will be \"\n            \"overridden.\"\n        )\n\n    # Add to the registry\n    registry[key] = cls\n\n    return cls\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dockerutils/","title":"dockerutils","text":"","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils","title":"prefect.utilities.dockerutils","text":"","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.BuildError","title":"BuildError","text":"

    Bases: Exception

    Raised when a Docker build fails

    Source code in prefect/utilities/dockerutils.py
    class BuildError(Exception):\n    \"\"\"Raised when a Docker build fails\"\"\"\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder","title":"ImageBuilder","text":"

    An interface for preparing Docker build contexts and building images

    Source code in prefect/utilities/dockerutils.py
    class ImageBuilder:\n    \"\"\"An interface for preparing Docker build contexts and building images\"\"\"\n\n    base_directory: Path\n    context: Optional[Path]\n    platform: Optional[str]\n    dockerfile_lines: List[str]\n\n    def __init__(\n        self,\n        base_image: str,\n        base_directory: Path = None,\n        platform: str = None,\n        context: Path = None,\n    ):\n        \"\"\"Create an ImageBuilder\n\n        Args:\n            base_image: the base image to use\n            base_directory: the starting point on your host for relative file locations,\n                defaulting to the current directory\n            context: use this path as the build context (if not provided, will create a\n                temporary directory for the context)\n\n        Returns:\n            The image ID\n        \"\"\"\n        self.base_directory = base_directory or context or Path().absolute()\n        self.temporary_directory = None\n        self.context = context\n        self.platform = platform\n        self.dockerfile_lines = []\n\n        if self.context:\n            dockerfile_path: Path = self.context / \"Dockerfile\"\n            if dockerfile_path.exists():\n                raise ValueError(f\"There is already a Dockerfile at {context}\")\n\n        self.add_line(f\"FROM {base_image}\")\n\n    def __enter__(self) -> Self:\n        if self.context and not self.temporary_directory:\n            return self\n\n        self.temporary_directory = TemporaryDirectory()\n        self.context = Path(self.temporary_directory.__enter__())\n        return self\n\n    def __exit__(\n        self, exc: Type[BaseException], value: BaseException, traceback: TracebackType\n    ) -> None:\n        if not self.temporary_directory:\n            return\n\n        self.temporary_directory.__exit__(exc, value, traceback)\n        self.temporary_directory = None\n        self.context = None\n\n    def add_line(self, line: str) -> None:\n        \"\"\"Add a line to this image's Dockerfile\"\"\"\n        self.add_lines([line])\n\n    def add_lines(self, lines: Iterable[str]) -> None:\n        \"\"\"Add lines to this image's Dockerfile\"\"\"\n        self.dockerfile_lines.extend(lines)\n\n    def copy(self, source: Union[str, Path], destination: Union[str, PurePosixPath]):\n        \"\"\"Copy a file to this image\"\"\"\n        if not self.context:\n            raise Exception(\"No context available\")\n\n        if not isinstance(destination, PurePosixPath):\n            destination = PurePosixPath(destination)\n\n        if not isinstance(source, Path):\n            source = Path(source)\n\n        if source.is_absolute():\n            source = source.resolve().relative_to(self.base_directory)\n\n        if self.temporary_directory:\n            os.makedirs(self.context / source.parent, exist_ok=True)\n\n            if source.is_dir():\n                shutil.copytree(self.base_directory / source, self.context / source)\n            else:\n                shutil.copy2(self.base_directory / source, self.context / source)\n\n        self.add_line(f\"COPY {source} {destination}\")\n\n    def write_text(self, text: str, destination: Union[str, PurePosixPath]):\n        if not self.context:\n            raise Exception(\"No context available\")\n\n        if not isinstance(destination, PurePosixPath):\n            destination = PurePosixPath(destination)\n\n        source_hash = hashlib.sha256(text.encode()).hexdigest()\n        (self.context / f\".{source_hash}\").write_text(text)\n        self.add_line(f\"COPY .{source_hash} {destination}\")\n\n    def build(\n        self, pull: bool = False, stream_progress_to: Optional[TextIO] = None\n    ) -> str:\n        \"\"\"Build the Docker image from the current state of the ImageBuilder\n\n        Args:\n            pull: True to pull the base image during the build\n            stream_progress_to: an optional stream (like sys.stdout, or an io.TextIO)\n                that will collect the build output as it is reported by Docker\n\n        Returns:\n            The image ID\n        \"\"\"\n        dockerfile_path: Path = self.context / \"Dockerfile\"\n\n        with dockerfile_path.open(\"w\") as dockerfile:\n            dockerfile.writelines(line + \"\\n\" for line in self.dockerfile_lines)\n\n        try:\n            return build_image(\n                self.context,\n                platform=self.platform,\n                pull=pull,\n                stream_progress_to=stream_progress_to,\n            )\n        finally:\n            os.unlink(dockerfile_path)\n\n    def assert_has_line(self, line: str) -> None:\n        \"\"\"Asserts that the given line is in the Dockerfile\"\"\"\n        all_lines = \"\\n\".join(\n            [f\"  {i+1:>3}: {line}\" for i, line in enumerate(self.dockerfile_lines)]\n        )\n        message = (\n            f\"Expected {line!r} not found in Dockerfile.  Dockerfile:\\n{all_lines}\"\n        )\n        assert line in self.dockerfile_lines, message\n\n    def assert_line_absent(self, line: str) -> None:\n        \"\"\"Asserts that the given line is absent from the Dockerfile\"\"\"\n        if line not in self.dockerfile_lines:\n            return\n\n        i = self.dockerfile_lines.index(line)\n\n        surrounding_lines = \"\\n\".join(\n            [\n                f\"  {i+1:>3}: {line}\"\n                for i, line in enumerate(self.dockerfile_lines[i - 2 : i + 2])\n            ]\n        )\n        message = (\n            f\"Unexpected {line!r} found in Dockerfile at line {i+1}.  \"\n            f\"Surrounding lines:\\n{surrounding_lines}\"\n        )\n\n        assert line not in self.dockerfile_lines, message\n\n    def assert_line_before(self, first: str, second: str) -> None:\n        \"\"\"Asserts that the first line appears before the second line\"\"\"\n        self.assert_has_line(first)\n        self.assert_has_line(second)\n\n        first_index = self.dockerfile_lines.index(first)\n        second_index = self.dockerfile_lines.index(second)\n\n        surrounding_lines = \"\\n\".join(\n            [\n                f\"  {i+1:>3}: {line}\"\n                for i, line in enumerate(\n                    self.dockerfile_lines[second_index - 2 : first_index + 2]\n                )\n            ]\n        )\n\n        message = (\n            f\"Expected {first!r} to appear before {second!r} in the Dockerfile, but \"\n            f\"{first!r} was at line {first_index+1} and {second!r} as at line \"\n            f\"{second_index+1}.  Surrounding lines:\\n{surrounding_lines}\"\n        )\n\n        assert first_index < second_index, message\n\n    def assert_line_after(self, second: str, first: str) -> None:\n        \"\"\"Asserts that the second line appears after the first line\"\"\"\n        self.assert_line_before(first, second)\n\n    def assert_has_file(self, source: Path, container_path: PurePosixPath) -> None:\n        \"\"\"Asserts that the given file or directory will be copied into the container\n        at the given path\"\"\"\n        if source.is_absolute():\n            source = source.relative_to(self.base_directory)\n\n        self.assert_has_line(f\"COPY {source} {container_path}\")\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.add_line","title":"add_line","text":"

    Add a line to this image's Dockerfile

    Source code in prefect/utilities/dockerutils.py
    def add_line(self, line: str) -> None:\n    \"\"\"Add a line to this image's Dockerfile\"\"\"\n    self.add_lines([line])\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.add_lines","title":"add_lines","text":"

    Add lines to this image's Dockerfile

    Source code in prefect/utilities/dockerutils.py
    def add_lines(self, lines: Iterable[str]) -> None:\n    \"\"\"Add lines to this image's Dockerfile\"\"\"\n    self.dockerfile_lines.extend(lines)\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_has_file","title":"assert_has_file","text":"

    Asserts that the given file or directory will be copied into the container at the given path

    Source code in prefect/utilities/dockerutils.py
    def assert_has_file(self, source: Path, container_path: PurePosixPath) -> None:\n    \"\"\"Asserts that the given file or directory will be copied into the container\n    at the given path\"\"\"\n    if source.is_absolute():\n        source = source.relative_to(self.base_directory)\n\n    self.assert_has_line(f\"COPY {source} {container_path}\")\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_has_line","title":"assert_has_line","text":"

    Asserts that the given line is in the Dockerfile

    Source code in prefect/utilities/dockerutils.py
    def assert_has_line(self, line: str) -> None:\n    \"\"\"Asserts that the given line is in the Dockerfile\"\"\"\n    all_lines = \"\\n\".join(\n        [f\"  {i+1:>3}: {line}\" for i, line in enumerate(self.dockerfile_lines)]\n    )\n    message = (\n        f\"Expected {line!r} not found in Dockerfile.  Dockerfile:\\n{all_lines}\"\n    )\n    assert line in self.dockerfile_lines, message\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_line_absent","title":"assert_line_absent","text":"

    Asserts that the given line is absent from the Dockerfile

    Source code in prefect/utilities/dockerutils.py
    def assert_line_absent(self, line: str) -> None:\n    \"\"\"Asserts that the given line is absent from the Dockerfile\"\"\"\n    if line not in self.dockerfile_lines:\n        return\n\n    i = self.dockerfile_lines.index(line)\n\n    surrounding_lines = \"\\n\".join(\n        [\n            f\"  {i+1:>3}: {line}\"\n            for i, line in enumerate(self.dockerfile_lines[i - 2 : i + 2])\n        ]\n    )\n    message = (\n        f\"Unexpected {line!r} found in Dockerfile at line {i+1}.  \"\n        f\"Surrounding lines:\\n{surrounding_lines}\"\n    )\n\n    assert line not in self.dockerfile_lines, message\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_line_after","title":"assert_line_after","text":"

    Asserts that the second line appears after the first line

    Source code in prefect/utilities/dockerutils.py
    def assert_line_after(self, second: str, first: str) -> None:\n    \"\"\"Asserts that the second line appears after the first line\"\"\"\n    self.assert_line_before(first, second)\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_line_before","title":"assert_line_before","text":"

    Asserts that the first line appears before the second line

    Source code in prefect/utilities/dockerutils.py
    def assert_line_before(self, first: str, second: str) -> None:\n    \"\"\"Asserts that the first line appears before the second line\"\"\"\n    self.assert_has_line(first)\n    self.assert_has_line(second)\n\n    first_index = self.dockerfile_lines.index(first)\n    second_index = self.dockerfile_lines.index(second)\n\n    surrounding_lines = \"\\n\".join(\n        [\n            f\"  {i+1:>3}: {line}\"\n            for i, line in enumerate(\n                self.dockerfile_lines[second_index - 2 : first_index + 2]\n            )\n        ]\n    )\n\n    message = (\n        f\"Expected {first!r} to appear before {second!r} in the Dockerfile, but \"\n        f\"{first!r} was at line {first_index+1} and {second!r} as at line \"\n        f\"{second_index+1}.  Surrounding lines:\\n{surrounding_lines}\"\n    )\n\n    assert first_index < second_index, message\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.build","title":"build","text":"

    Build the Docker image from the current state of the ImageBuilder

    Parameters:

    Name Type Description Default pull bool

    True to pull the base image during the build

    False stream_progress_to Optional[TextIO]

    an optional stream (like sys.stdout, or an io.TextIO) that will collect the build output as it is reported by Docker

    None

    Returns:

    Type Description str

    The image ID

    Source code in prefect/utilities/dockerutils.py
    def build(\n    self, pull: bool = False, stream_progress_to: Optional[TextIO] = None\n) -> str:\n    \"\"\"Build the Docker image from the current state of the ImageBuilder\n\n    Args:\n        pull: True to pull the base image during the build\n        stream_progress_to: an optional stream (like sys.stdout, or an io.TextIO)\n            that will collect the build output as it is reported by Docker\n\n    Returns:\n        The image ID\n    \"\"\"\n    dockerfile_path: Path = self.context / \"Dockerfile\"\n\n    with dockerfile_path.open(\"w\") as dockerfile:\n        dockerfile.writelines(line + \"\\n\" for line in self.dockerfile_lines)\n\n    try:\n        return build_image(\n            self.context,\n            platform=self.platform,\n            pull=pull,\n            stream_progress_to=stream_progress_to,\n        )\n    finally:\n        os.unlink(dockerfile_path)\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.copy","title":"copy","text":"

    Copy a file to this image

    Source code in prefect/utilities/dockerutils.py
    def copy(self, source: Union[str, Path], destination: Union[str, PurePosixPath]):\n    \"\"\"Copy a file to this image\"\"\"\n    if not self.context:\n        raise Exception(\"No context available\")\n\n    if not isinstance(destination, PurePosixPath):\n        destination = PurePosixPath(destination)\n\n    if not isinstance(source, Path):\n        source = Path(source)\n\n    if source.is_absolute():\n        source = source.resolve().relative_to(self.base_directory)\n\n    if self.temporary_directory:\n        os.makedirs(self.context / source.parent, exist_ok=True)\n\n        if source.is_dir():\n            shutil.copytree(self.base_directory / source, self.context / source)\n        else:\n            shutil.copy2(self.base_directory / source, self.context / source)\n\n    self.add_line(f\"COPY {source} {destination}\")\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.PushError","title":"PushError","text":"

    Bases: Exception

    Raised when a Docker image push fails

    Source code in prefect/utilities/dockerutils.py
    class PushError(Exception):\n    \"\"\"Raised when a Docker image push fails\"\"\"\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.build_image","title":"build_image","text":"

    Builds a Docker image, returning the image ID

    Parameters:

    Name Type Description Default context Path

    the root directory for the Docker build context

    required dockerfile str

    the path to the Dockerfile, relative to the context

    'Dockerfile' tag Optional[str]

    the tag to give this image

    None pull bool

    True to pull the base image during the build

    False stream_progress_to Optional[TextIO]

    an optional stream (like sys.stdout, or an io.TextIO) that will collect the build output as it is reported by Docker

    None

    Returns:

    Type Description str

    The image ID

    Source code in prefect/utilities/dockerutils.py
    @silence_docker_warnings()\ndef build_image(\n    context: Path,\n    dockerfile: str = \"Dockerfile\",\n    tag: Optional[str] = None,\n    pull: bool = False,\n    platform: str = None,\n    stream_progress_to: Optional[TextIO] = None,\n    **kwargs,\n) -> str:\n    \"\"\"Builds a Docker image, returning the image ID\n\n    Args:\n        context: the root directory for the Docker build context\n        dockerfile: the path to the Dockerfile, relative to the context\n        tag: the tag to give this image\n        pull: True to pull the base image during the build\n        stream_progress_to: an optional stream (like sys.stdout, or an io.TextIO) that\n            will collect the build output as it is reported by Docker\n\n    Returns:\n        The image ID\n    \"\"\"\n\n    if not context:\n        raise ValueError(\"context required to build an image\")\n\n    if not Path(context).exists():\n        raise ValueError(f\"Context path {context} does not exist\")\n\n    kwargs = {key: kwargs[key] for key in kwargs if key not in [\"decode\", \"labels\"]}\n\n    image_id = None\n    with docker_client() as client:\n        events = client.api.build(\n            path=str(context),\n            tag=tag,\n            dockerfile=dockerfile,\n            pull=pull,\n            decode=True,\n            labels=IMAGE_LABELS,\n            platform=platform,\n            **kwargs,\n        )\n\n        try:\n            for event in events:\n                if \"stream\" in event:\n                    if not stream_progress_to:\n                        continue\n                    stream_progress_to.write(event[\"stream\"])\n                    stream_progress_to.flush()\n                elif \"aux\" in event:\n                    image_id = event[\"aux\"][\"ID\"]\n                elif \"error\" in event:\n                    raise BuildError(event[\"error\"])\n                elif \"message\" in event:\n                    raise BuildError(event[\"message\"])\n        except docker.errors.APIError as e:\n            raise BuildError(e.explanation) from e\n\n    assert image_id, \"The Docker daemon did not return an image ID\"\n    return image_id\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.docker_client","title":"docker_client","text":"

    Get the environmentally-configured Docker client

    Source code in prefect/utilities/dockerutils.py
    @contextmanager\ndef docker_client() -> Generator[\"DockerClient\", None, None]:\n    \"\"\"Get the environmentally-configured Docker client\"\"\"\n    with silence_docker_warnings():\n        client = docker.DockerClient.from_env()\n\n    try:\n        yield client\n    finally:\n        client.close()\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.format_outlier_version_name","title":"format_outlier_version_name","text":"

    Formats outlier docker version names to pass packaging.version.parse validation - Current cases are simple, but creates stub for more complicated formatting if eventually needed. - Example outlier versions that throw a parsing exception: - \"20.10.0-ce\" (variant of community edition label) - \"20.10.0-ee\" (variant of enterprise edition label)

    Parameters:

    Name Type Description Default version str

    raw docker version value

    required

    Returns:

    Name Type Description str

    value that can pass packaging.version.parse validation

    Source code in prefect/utilities/dockerutils.py
    def format_outlier_version_name(version: str):\n    \"\"\"\n    Formats outlier docker version names to pass `packaging.version.parse` validation\n    - Current cases are simple, but creates stub for more complicated formatting if eventually needed.\n    - Example outlier versions that throw a parsing exception:\n      - \"20.10.0-ce\" (variant of community edition label)\n      - \"20.10.0-ee\" (variant of enterprise edition label)\n\n    Args:\n        version (str): raw docker version value\n\n    Returns:\n        str: value that can pass `packaging.version.parse` validation\n    \"\"\"\n    return version.replace(\"-ce\", \"\").replace(\"-ee\", \"\")\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.generate_default_dockerfile","title":"generate_default_dockerfile","text":"

    Generates a default Dockerfile used for deploying flows. The Dockerfile is written to a temporary file and yielded. The temporary file is removed after the context manager exits.

    Parameters:

    Name Type Description Default - context

    The context to use for the Dockerfile. Defaults to the current working directory.

    required Source code in prefect/utilities/dockerutils.py
    @contextmanager\ndef generate_default_dockerfile(context: Optional[Path] = None):\n    \"\"\"\n    Generates a default Dockerfile used for deploying flows. The Dockerfile is written\n    to a temporary file and yielded. The temporary file is removed after the context\n    manager exits.\n\n    Args:\n        - context: The context to use for the Dockerfile. Defaults to\n            the current working directory.\n    \"\"\"\n    if not context:\n        context = Path.cwd()\n    lines = []\n    base_image = get_prefect_image_name()\n    lines.append(f\"FROM {base_image}\")\n    dir_name = context.name\n\n    if (context / \"requirements.txt\").exists():\n        lines.append(f\"COPY requirements.txt /opt/prefect/{dir_name}/requirements.txt\")\n        lines.append(\n            f\"RUN python -m pip install -r /opt/prefect/{dir_name}/requirements.txt\"\n        )\n\n    lines.append(f\"COPY . /opt/prefect/{dir_name}/\")\n    lines.append(f\"WORKDIR /opt/prefect/{dir_name}/\")\n\n    temp_dockerfile = context / \"Dockerfile\"\n    if Path(temp_dockerfile).exists():\n        raise RuntimeError(\n            \"Failed to generate Dockerfile. Dockerfile already exists in the\"\n            \" current directory.\"\n        )\n\n    with Path(temp_dockerfile).open(\"w\") as f:\n        f.writelines(line + \"\\n\" for line in lines)\n\n    try:\n        yield temp_dockerfile\n    finally:\n        temp_dockerfile.unlink()\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.get_prefect_image_name","title":"get_prefect_image_name","text":"

    Get the Prefect image name matching the current Prefect and Python versions.

    Parameters:

    Name Type Description Default prefect_version str

    An optional override for the Prefect version.

    None python_version str

    An optional override for the Python version; must be at the minor level e.g. '3.9'.

    None flavor str

    An optional alternative image flavor to build, like 'conda'

    None Source code in prefect/utilities/dockerutils.py
    def get_prefect_image_name(\n    prefect_version: str = None, python_version: str = None, flavor: str = None\n) -> str:\n    \"\"\"\n    Get the Prefect image name matching the current Prefect and Python versions.\n\n    Args:\n        prefect_version: An optional override for the Prefect version.\n        python_version: An optional override for the Python version; must be at the\n            minor level e.g. '3.9'.\n        flavor: An optional alternative image flavor to build, like 'conda'\n    \"\"\"\n    parsed_version = (prefect_version or prefect.__version__).split(\"+\")\n    is_prod_build = len(parsed_version) == 1\n    prefect_version = (\n        parsed_version[0]\n        if is_prod_build\n        else \"sha-\" + prefect.__version_info__[\"full-revisionid\"][:7]\n    )\n\n    python_version = python_version or python_version_minor()\n\n    tag = slugify(\n        f\"{prefect_version}-python{python_version}\" + (f\"-{flavor}\" if flavor else \"\"),\n        lowercase=False,\n        max_length=128,\n        # Docker allows these characters for tag names\n        regex_pattern=r\"[^a-zA-Z0-9_.-]+\",\n    )\n\n    image = \"prefect\" if is_prod_build else \"prefect-dev\"\n    return f\"prefecthq/{image}:{tag}\"\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.parse_image_tag","title":"parse_image_tag","text":"

    Parse Docker Image String

    • If a tag exists, this function parses and returns the image registry and tag, separately as a tuple.
    • Example 1: 'prefecthq/prefect:latest' -> ('prefecthq/prefect', 'latest')
    • Example 2: 'hostname.io:5050/folder/subfolder:latest' -> ('hostname.io:5050/folder/subfolder', 'latest')
    • Supports parsing Docker Image strings that follow Docker Image Specification v1.1.0
    • Image building tools typically enforce this standard

    Parameters:

    Name Type Description Default name str

    Name of Docker Image

    required Return Source code in prefect/utilities/dockerutils.py
    def parse_image_tag(name: str) -> Tuple[str, Optional[str]]:\n    \"\"\"\n    Parse Docker Image String\n\n    - If a tag exists, this function parses and returns the image registry and tag,\n      separately as a tuple.\n      - Example 1: 'prefecthq/prefect:latest' -> ('prefecthq/prefect', 'latest')\n      - Example 2: 'hostname.io:5050/folder/subfolder:latest' -> ('hostname.io:5050/folder/subfolder', 'latest')\n    - Supports parsing Docker Image strings that follow Docker Image Specification v1.1.0\n      - Image building tools typically enforce this standard\n\n    Args:\n        name (str): Name of Docker Image\n\n    Return:\n        tuple: image registry, image tag\n    \"\"\"\n    tag = None\n    name_parts = name.split(\"/\")\n    # First handles the simplest image names (DockerHub-based, index-free, potentionally with a tag)\n    # - Example: simplename:latest\n    if len(name_parts) == 1:\n        if \":\" in name_parts[0]:\n            image_name, tag = name_parts[0].split(\":\")\n        else:\n            image_name = name_parts[0]\n    else:\n        # 1. Separates index (hostname.io or prefecthq) from path:tag (folder/subfolder:latest or prefect:latest)\n        # 2. Separates path and tag (if tag exists)\n        # 3. Reunites index and path (without tag) as image name\n        index_name = name_parts[0]\n        image_path = \"/\".join(name_parts[1:])\n        if \":\" in image_path:\n            image_path, tag = image_path.split(\":\")\n        image_name = f\"{index_name}/{image_path}\"\n    return image_name, tag\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.push_image","title":"push_image","text":"

    Pushes a local image to a Docker registry, returning the registry-qualified tag for that image

    This assumes that the environment's Docker daemon is already authenticated to the given registry, and currently makes no attempt to authenticate.

    Parameters:

    Name Type Description Default image_id str

    a Docker image ID

    required registry_url str

    the URL of a Docker registry

    required name str

    the name of this image

    required tag str

    the tag to give this image (defaults to a short representation of the image's ID)

    None stream_progress_to Optional[TextIO]

    an optional stream (like sys.stdout, or an io.TextIO) that will collect the build output as it is reported by Docker

    None

    Returns:

    Type Description str

    A registry-qualified tag, like my-registry.example.com/my-image:abcdefg

    Source code in prefect/utilities/dockerutils.py
    @silence_docker_warnings()\ndef push_image(\n    image_id: str,\n    registry_url: str,\n    name: str,\n    tag: Optional[str] = None,\n    stream_progress_to: Optional[TextIO] = None,\n) -> str:\n    \"\"\"Pushes a local image to a Docker registry, returning the registry-qualified tag\n    for that image\n\n    This assumes that the environment's Docker daemon is already authenticated to the\n    given registry, and currently makes no attempt to authenticate.\n\n    Args:\n        image_id (str): a Docker image ID\n        registry_url (str): the URL of a Docker registry\n        name (str): the name of this image\n        tag (str): the tag to give this image (defaults to a short representation of\n            the image's ID)\n        stream_progress_to: an optional stream (like sys.stdout, or an io.TextIO) that\n            will collect the build output as it is reported by Docker\n\n    Returns:\n        A registry-qualified tag, like my-registry.example.com/my-image:abcdefg\n    \"\"\"\n\n    if not tag:\n        tag = slugify(pendulum.now(\"utc\").isoformat())\n\n    _, registry, _, _, _ = urlsplit(registry_url)\n    repository = f\"{registry}/{name}\"\n\n    with docker_client() as client:\n        image: \"docker.Image\" = client.images.get(image_id)\n        image.tag(repository, tag=tag)\n        events = client.api.push(repository, tag=tag, stream=True, decode=True)\n        try:\n            for event in events:\n                if \"status\" in event:\n                    if not stream_progress_to:\n                        continue\n                    stream_progress_to.write(event[\"status\"])\n                    if \"progress\" in event:\n                        stream_progress_to.write(\" \" + event[\"progress\"])\n                    stream_progress_to.write(\"\\n\")\n                    stream_progress_to.flush()\n                elif \"error\" in event:\n                    raise PushError(event[\"error\"])\n        finally:\n            client.api.remove_image(f\"{repository}:{tag}\", noprune=True)\n\n    return f\"{repository}:{tag}\"\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.split_repository_path","title":"split_repository_path","text":"

    Splits a Docker repository path into its namespace and repository components.

    Parameters:

    Name Type Description Default repository_path str

    The Docker repository path to split.

    required

    Returns:

    Type Description Tuple[Optional[str], str]

    Tuple[Optional[str], str]: A tuple containing the namespace and repository components. - namespace (Optional[str]): The Docker namespace, combining the registry and organization. None if not present. - repository (Optionals[str]): The repository name.

    Source code in prefect/utilities/dockerutils.py
    def split_repository_path(repository_path: str) -> Tuple[Optional[str], str]:\n    \"\"\"\n    Splits a Docker repository path into its namespace and repository components.\n\n    Args:\n        repository_path: The Docker repository path to split.\n\n    Returns:\n        Tuple[Optional[str], str]: A tuple containing the namespace and repository components.\n            - namespace (Optional[str]): The Docker namespace, combining the registry and organization. None if not present.\n            - repository (Optionals[str]): The repository name.\n    \"\"\"\n    parts = repository_path.split(\"/\", 2)\n\n    # Check if the path includes a registry and organization or just organization/repository\n    if len(parts) == 3 or (len(parts) == 2 and (\".\" in parts[0] or \":\" in parts[0])):\n        # Namespace includes registry and organization\n        namespace = \"/\".join(parts[:-1])\n        repository = parts[-1]\n    elif len(parts) == 2:\n        # Only organization/repository provided, so namespace is just the first part\n        namespace = parts[0]\n        repository = parts[1]\n    else:\n        # No namespace provided\n        namespace = None\n        repository = parts[0]\n\n    return namespace, repository\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.to_run_command","title":"to_run_command","text":"

    Convert a process-style list of command arguments to a single Dockerfile RUN instruction.

    Source code in prefect/utilities/dockerutils.py
    def to_run_command(command: List[str]) -> str:\n    \"\"\"\n    Convert a process-style list of command arguments to a single Dockerfile RUN\n    instruction.\n    \"\"\"\n    if not command:\n        return \"\"\n\n    run_command = f\"RUN {command[0]}\"\n    if len(command) > 1:\n        run_command += \" \" + \" \".join([repr(arg) for arg in command[1:]])\n\n    # TODO: Consider performing text-wrapping to improve readability of the generated\n    #       Dockerfile\n    # return textwrap.wrap(\n    #     run_command,\n    #     subsequent_indent=\" \" * 4,\n    #     break_on_hyphens=False,\n    #     break_long_words=False\n    # )\n\n    return run_command\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/filesystem/","title":"filesystem","text":"","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem","title":"prefect.utilities.filesystem","text":"

    Utilities for working with file systems

    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.create_default_ignore_file","title":"create_default_ignore_file","text":"

    Creates default ignore file in the provided path if one does not already exist; returns boolean specifying whether a file was created.

    Source code in prefect/utilities/filesystem.py
    def create_default_ignore_file(path: str) -> bool:\n    \"\"\"\n    Creates default ignore file in the provided path if one does not already exist; returns boolean specifying\n    whether a file was created.\n    \"\"\"\n    path = pathlib.Path(path)\n    ignore_file = path / \".prefectignore\"\n    if ignore_file.exists():\n        return False\n    default_file = pathlib.Path(prefect.__module_path__) / \".prefectignore\"\n    with ignore_file.open(mode=\"w\") as f:\n        f.write(default_file.read_text())\n    return True\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.filename","title":"filename","text":"

    Extract the file name from a path with remote file system support

    Source code in prefect/utilities/filesystem.py
    def filename(path: str) -> str:\n    \"\"\"Extract the file name from a path with remote file system support\"\"\"\n    try:\n        of: OpenFile = fsspec.open(path)\n        sep = of.fs.sep\n    except (ImportError, AttributeError):\n        sep = \"\\\\\" if \"\\\\\" in path else \"/\"\n    return path.split(sep)[-1]\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.filter_files","title":"filter_files","text":"

    This function accepts a root directory path and a list of file patterns to ignore, and returns a list of files that excludes those that should be ignored.

    The specification matches that of .gitignore files.

    Source code in prefect/utilities/filesystem.py
    def filter_files(\n    root: str = \".\", ignore_patterns: list = None, include_dirs: bool = True\n) -> set:\n    \"\"\"\n    This function accepts a root directory path and a list of file patterns to ignore, and returns\n    a list of files that excludes those that should be ignored.\n\n    The specification matches that of [.gitignore files](https://git-scm.com/docs/gitignore).\n    \"\"\"\n    if ignore_patterns is None:\n        ignore_patterns = []\n    spec = pathspec.PathSpec.from_lines(\"gitwildmatch\", ignore_patterns)\n    ignored_files = {p.path for p in spec.match_tree_entries(root)}\n    if include_dirs:\n        all_files = {p.path for p in pathspec.util.iter_tree_entries(root)}\n    else:\n        all_files = set(pathspec.util.iter_tree_files(root))\n    included_files = all_files - ignored_files\n    return included_files\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.get_open_file_limit","title":"get_open_file_limit","text":"

    Get the maximum number of open files allowed for the current process

    Source code in prefect/utilities/filesystem.py
    def get_open_file_limit() -> int:\n    \"\"\"Get the maximum number of open files allowed for the current process\"\"\"\n\n    try:\n        if os.name == \"nt\":\n            import ctypes\n\n            return ctypes.cdll.ucrtbase._getmaxstdio()\n        else:\n            import resource\n\n            soft_limit, _ = resource.getrlimit(resource.RLIMIT_NOFILE)\n            return soft_limit\n    except Exception:\n        # Catch all exceptions, as ctypes can raise several errors\n        # depending on what went wrong. Return a safe default if we\n        # can't get the limit from the OS.\n        return 200\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.is_local_path","title":"is_local_path","text":"

    Check if the given path points to a local or remote file system

    Source code in prefect/utilities/filesystem.py
    def is_local_path(path: Union[str, pathlib.Path, OpenFile]):\n    \"\"\"Check if the given path points to a local or remote file system\"\"\"\n    if isinstance(path, str):\n        try:\n            of = fsspec.open(path)\n        except ImportError:\n            # The path is a remote file system that uses a lib that is not installed\n            return False\n    elif isinstance(path, pathlib.Path):\n        return True\n    elif isinstance(path, OpenFile):\n        of = path\n    else:\n        raise TypeError(f\"Invalid path of type {type(path).__name__!r}\")\n\n    return type(of.fs) == LocalFileSystem\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.relative_path_to_current_platform","title":"relative_path_to_current_platform","text":"

    Converts a relative path generated on any platform to a relative path for the current platform.

    Source code in prefect/utilities/filesystem.py
    def relative_path_to_current_platform(path_str: str) -> Path:\n    \"\"\"\n    Converts a relative path generated on any platform to a relative path for the\n    current platform.\n    \"\"\"\n\n    return Path(PureWindowsPath(path_str).as_posix())\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.tmpchdir","title":"tmpchdir","text":"

    Change current-working directories for the duration of the context

    Source code in prefect/utilities/filesystem.py
    @contextmanager\ndef tmpchdir(path: str):\n    \"\"\"\n    Change current-working directories for the duration of the context\n    \"\"\"\n    path = os.path.abspath(path)\n    if os.path.isfile(path) or (not os.path.exists(path) and not path.endswith(\"/\")):\n        path = os.path.dirname(path)\n\n    owd = os.getcwd()\n\n    try:\n        os.chdir(path)\n        yield path\n    finally:\n        os.chdir(owd)\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.to_display_path","title":"to_display_path","text":"

    Convert a path to a displayable path. The absolute path or relative path to the current (or given) directory will be returned, whichever is shorter.

    Source code in prefect/utilities/filesystem.py
    def to_display_path(\n    path: Union[pathlib.Path, str], relative_to: Union[pathlib.Path, str] = None\n) -> str:\n    \"\"\"\n    Convert a path to a displayable path. The absolute path or relative path to the\n    current (or given) directory will be returned, whichever is shorter.\n    \"\"\"\n    path, relative_to = (\n        pathlib.Path(path).resolve(),\n        pathlib.Path(relative_to or \".\").resolve(),\n    )\n    relative_path = str(path.relative_to(relative_to))\n    absolute_path = str(path)\n    return relative_path if len(relative_path) < len(absolute_path) else absolute_path\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/hashing/","title":"hashing","text":"","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/hashing/#prefect.utilities.hashing","title":"prefect.utilities.hashing","text":"","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/hashing/#prefect.utilities.hashing.file_hash","title":"file_hash","text":"

    Given a path to a file, produces a stable hash of the file contents.

    Parameters:

    Name Type Description Default path str

    the path to a file

    required hash_algo

    Hash algorithm from hashlib to use.

    _md5

    Returns:

    Name Type Description str str

    a hash of the file contents

    Source code in prefect/utilities/hashing.py
    def file_hash(path: str, hash_algo=_md5) -> str:\n    \"\"\"Given a path to a file, produces a stable hash of the file contents.\n\n    Args:\n        path (str): the path to a file\n        hash_algo: Hash algorithm from hashlib to use.\n\n    Returns:\n        str: a hash of the file contents\n    \"\"\"\n    contents = Path(path).read_bytes()\n    return stable_hash(contents, hash_algo=hash_algo)\n
    ","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/hashing/#prefect.utilities.hashing.hash_objects","title":"hash_objects","text":"

    Attempt to hash objects by dumping to JSON or serializing with cloudpickle. On failure of both, None will be returned

    Source code in prefect/utilities/hashing.py
    def hash_objects(*args, hash_algo=_md5, **kwargs) -> Optional[str]:\n    \"\"\"\n    Attempt to hash objects by dumping to JSON or serializing with cloudpickle.\n    On failure of both, `None` will be returned\n    \"\"\"\n    try:\n        serializer = JSONSerializer(dumps_kwargs={\"sort_keys\": True})\n        return stable_hash(serializer.dumps((args, kwargs)), hash_algo=hash_algo)\n    except Exception:\n        pass\n\n    try:\n        return stable_hash(cloudpickle.dumps((args, kwargs)), hash_algo=hash_algo)\n    except Exception:\n        pass\n\n    return None\n
    ","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/hashing/#prefect.utilities.hashing.stable_hash","title":"stable_hash","text":"

    Given some arguments, produces a stable 64-bit hash of their contents.

    Supports bytes and strings. Strings will be UTF-8 encoded.

    Parameters:

    Name Type Description Default *args Union[str, bytes]

    Items to include in the hash.

    () hash_algo

    Hash algorithm from hashlib to use.

    _md5

    Returns:

    Type Description str

    A hex hash.

    Source code in prefect/utilities/hashing.py
    def stable_hash(*args: Union[str, bytes], hash_algo=_md5) -> str:\n    \"\"\"Given some arguments, produces a stable 64-bit hash of their contents.\n\n    Supports bytes and strings. Strings will be UTF-8 encoded.\n\n    Args:\n        *args: Items to include in the hash.\n        hash_algo: Hash algorithm from hashlib to use.\n\n    Returns:\n        A hex hash.\n    \"\"\"\n    h = hash_algo()\n    for a in args:\n        if isinstance(a, str):\n            a = a.encode()\n        h.update(a)\n    return h.hexdigest()\n
    ","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/importtools/","title":"importtools","text":"","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools","title":"prefect.utilities.importtools","text":"","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.AliasedModuleDefinition","title":"AliasedModuleDefinition","text":"

    Bases: NamedTuple

    A definition for the AliasedModuleFinder.

    Parameters:

    Name Type Description Default alias

    The import name to create

    required real

    The import name of the module to reference for the alias

    required callback

    A function to call when the alias module is loaded

    required Source code in prefect/utilities/importtools.py
    class AliasedModuleDefinition(NamedTuple):\n    \"\"\"\n    A definition for the `AliasedModuleFinder`.\n\n    Args:\n        alias: The import name to create\n        real: The import name of the module to reference for the alias\n        callback: A function to call when the alias module is loaded\n    \"\"\"\n\n    alias: str\n    real: str\n    callback: Optional[Callable[[str], None]]\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.AliasedModuleFinder","title":"AliasedModuleFinder","text":"

    Bases: MetaPathFinder

    Source code in prefect/utilities/importtools.py
    class AliasedModuleFinder(MetaPathFinder):\n    def __init__(self, aliases: Iterable[AliasedModuleDefinition]):\n        \"\"\"\n        See `AliasedModuleDefinition` for alias specification.\n\n        Aliases apply to all modules nested within an alias.\n        \"\"\"\n        self.aliases = aliases\n\n    def find_spec(\n        self,\n        fullname: str,\n        path=None,\n        target=None,\n    ) -> Optional[ModuleSpec]:\n        \"\"\"\n        The fullname is the imported path, e.g. \"foo.bar\". If there is an alias \"phi\"\n        for \"foo\" then on import of \"phi.bar\" we will find the spec for \"foo.bar\" and\n        create a new spec for \"phi.bar\" that points to \"foo.bar\".\n        \"\"\"\n        for alias, real, callback in self.aliases:\n            if fullname.startswith(alias):\n                # Retrieve the spec of the real module\n                real_spec = importlib.util.find_spec(fullname.replace(alias, real, 1))\n                # Create a new spec for the alias\n                return ModuleSpec(\n                    fullname,\n                    AliasedModuleLoader(fullname, callback, real_spec),\n                    origin=real_spec.origin,\n                    is_package=real_spec.submodule_search_locations is not None,\n                )\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.AliasedModuleFinder.find_spec","title":"find_spec","text":"

    The fullname is the imported path, e.g. \"foo.bar\". If there is an alias \"phi\" for \"foo\" then on import of \"phi.bar\" we will find the spec for \"foo.bar\" and create a new spec for \"phi.bar\" that points to \"foo.bar\".

    Source code in prefect/utilities/importtools.py
    def find_spec(\n    self,\n    fullname: str,\n    path=None,\n    target=None,\n) -> Optional[ModuleSpec]:\n    \"\"\"\n    The fullname is the imported path, e.g. \"foo.bar\". If there is an alias \"phi\"\n    for \"foo\" then on import of \"phi.bar\" we will find the spec for \"foo.bar\" and\n    create a new spec for \"phi.bar\" that points to \"foo.bar\".\n    \"\"\"\n    for alias, real, callback in self.aliases:\n        if fullname.startswith(alias):\n            # Retrieve the spec of the real module\n            real_spec = importlib.util.find_spec(fullname.replace(alias, real, 1))\n            # Create a new spec for the alias\n            return ModuleSpec(\n                fullname,\n                AliasedModuleLoader(fullname, callback, real_spec),\n                origin=real_spec.origin,\n                is_package=real_spec.submodule_search_locations is not None,\n            )\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.DelayedImportErrorModule","title":"DelayedImportErrorModule","text":"

    Bases: ModuleType

    A fake module returned by lazy_import when the module cannot be found. When any of the module's attributes are accessed, we will throw a ModuleNotFoundError.

    Adapted from lazy_loader

    Source code in prefect/utilities/importtools.py
    class DelayedImportErrorModule(ModuleType):\n    \"\"\"\n    A fake module returned by `lazy_import` when the module cannot be found. When any\n    of the module's attributes are accessed, we will throw a `ModuleNotFoundError`.\n\n    Adapted from [lazy_loader][1]\n\n    [1]: https://github.com/scientific-python/lazy_loader\n    \"\"\"\n\n    def __init__(self, frame_data, help_message, *args, **kwargs):\n        self.__frame_data = frame_data\n        self.__help_message = (\n            help_message or \"Import errors for this module are only reported when used.\"\n        )\n        super().__init__(*args, **kwargs)\n\n    def __getattr__(self, attr):\n        if attr in (\"__class__\", \"__file__\", \"__frame_data\", \"__help_message\"):\n            super().__getattr__(attr)\n        else:\n            fd = self.__frame_data\n            raise ModuleNotFoundError(\n                f\"No module named '{fd['spec']}'\\n\\nThis module was originally imported\"\n                f\" at:\\n  File \\\"{fd['filename']}\\\", line {fd['lineno']}, in\"\n                f\" {fd['function']}\\n\\n    {''.join(fd['code_context']).strip()}\\n\"\n                + self.__help_message\n            )\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.from_qualified_name","title":"from_qualified_name","text":"

    Import an object given a fully-qualified name.

    Parameters:

    Name Type Description Default name str

    The fully-qualified name of the object to import.

    required

    Returns:

    Type Description Any

    the imported object

    Examples:

    >>> obj = from_qualified_name(\"random.randint\")\n>>> import random\n>>> obj == random.randint\nTrue\n
    Source code in prefect/utilities/importtools.py
    def from_qualified_name(name: str) -> Any:\n    \"\"\"\n    Import an object given a fully-qualified name.\n\n    Args:\n        name: The fully-qualified name of the object to import.\n\n    Returns:\n        the imported object\n\n    Examples:\n        >>> obj = from_qualified_name(\"random.randint\")\n        >>> import random\n        >>> obj == random.randint\n        True\n    \"\"\"\n    # Try importing it first so we support \"module\" or \"module.sub_module\"\n    try:\n        module = importlib.import_module(name)\n        return module\n    except ImportError:\n        # If no subitem was included raise the import error\n        if \".\" not in name:\n            raise\n\n    # Otherwise, we'll try to load it as an attribute of a module\n    mod_name, attr_name = name.rsplit(\".\", 1)\n    module = importlib.import_module(mod_name)\n    return getattr(module, attr_name)\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.import_object","title":"import_object","text":"

    Load an object from an import path.

    Import paths can be formatted as one of: - module.object - module:object - /path/to/script.py:object

    This function is not thread safe as it modifies the 'sys' module during execution.

    Source code in prefect/utilities/importtools.py
    def import_object(import_path: str):\n    \"\"\"\n    Load an object from an import path.\n\n    Import paths can be formatted as one of:\n    - module.object\n    - module:object\n    - /path/to/script.py:object\n\n    This function is not thread safe as it modifies the 'sys' module during execution.\n    \"\"\"\n    if \".py:\" in import_path:\n        script_path, object_name = import_path.rsplit(\":\", 1)\n        module = load_script_as_module(script_path)\n    else:\n        if \":\" in import_path:\n            module_name, object_name = import_path.rsplit(\":\", 1)\n        elif \".\" in import_path:\n            module_name, object_name = import_path.rsplit(\".\", 1)\n        else:\n            raise ValueError(\n                f\"Invalid format for object import. Received {import_path!r}.\"\n            )\n\n        module = load_module(module_name)\n\n    return getattr(module, object_name)\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.lazy_import","title":"lazy_import","text":"

    Create a lazily-imported module to use in place of the module of the given name. Use this to retain module-level imports for libraries that we don't want to actually import until they are needed.

    Adapted from the Python documentation and lazy_loader

    Source code in prefect/utilities/importtools.py
    def lazy_import(\n    name: str, error_on_import: bool = False, help_message: str = \"\"\n) -> ModuleType:\n    \"\"\"\n    Create a lazily-imported module to use in place of the module of the given name.\n    Use this to retain module-level imports for libraries that we don't want to\n    actually import until they are needed.\n\n    Adapted from the [Python documentation][1] and [lazy_loader][2]\n\n    [1]: https://docs.python.org/3/library/importlib.html#implementing-lazy-imports\n    [2]: https://github.com/scientific-python/lazy_loader\n    \"\"\"\n\n    try:\n        return sys.modules[name]\n    except KeyError:\n        pass\n\n    spec = importlib.util.find_spec(name)\n    if spec is None:\n        if error_on_import:\n            raise ModuleNotFoundError(f\"No module named '{name}'.\\n{help_message}\")\n        else:\n            try:\n                parent = inspect.stack()[1]\n                frame_data = {\n                    \"spec\": name,\n                    \"filename\": parent.filename,\n                    \"lineno\": parent.lineno,\n                    \"function\": parent.function,\n                    \"code_context\": parent.code_context,\n                }\n                return DelayedImportErrorModule(\n                    frame_data, help_message, \"DelayedImportErrorModule\"\n                )\n            finally:\n                del parent\n\n    module = importlib.util.module_from_spec(spec)\n    sys.modules[name] = module\n\n    loader = importlib.util.LazyLoader(spec.loader)\n    loader.exec_module(module)\n\n    return module\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.load_module","title":"load_module","text":"

    Import a module with support for relative imports within the module.

    Source code in prefect/utilities/importtools.py
    def load_module(module_name: str) -> ModuleType:\n    \"\"\"\n    Import a module with support for relative imports within the module.\n    \"\"\"\n    # Ensure relative imports within the imported module work if the user is in the\n    # correct working directory\n    working_directory = os.getcwd()\n    sys.path.insert(0, working_directory)\n\n    try:\n        return importlib.import_module(module_name)\n    finally:\n        sys.path.remove(working_directory)\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.load_script_as_module","title":"load_script_as_module","text":"

    Execute a script at the given path.

    Sets the module name to __prefect_loader__.

    If an exception occurs during execution of the script, a prefect.exceptions.ScriptError is created to wrap the exception and raised.

    During the duration of this function call, sys is modified to support loading. These changes are reverted after completion, but this function is not thread safe and use of it in threaded contexts may result in undesirable behavior.

    See https://docs.python.org/3/library/importlib.html#importing-a-source-file-directly

    Source code in prefect/utilities/importtools.py
    def load_script_as_module(path: str) -> ModuleType:\n    \"\"\"\n    Execute a script at the given path.\n\n    Sets the module name to `__prefect_loader__`.\n\n    If an exception occurs during execution of the script, a\n    `prefect.exceptions.ScriptError` is created to wrap the exception and raised.\n\n    During the duration of this function call, `sys` is modified to support loading.\n    These changes are reverted after completion, but this function is not thread safe\n    and use of it in threaded contexts may result in undesirable behavior.\n\n    See https://docs.python.org/3/library/importlib.html#importing-a-source-file-directly\n    \"\"\"\n    # We will add the parent directory to search locations to support relative imports\n    # during execution of the script\n    if not path.endswith(\".py\"):\n        raise ValueError(f\"The provided path does not point to a python file: {path!r}\")\n\n    parent_path = str(Path(path).resolve().parent)\n    working_directory = os.getcwd()\n\n    spec = importlib.util.spec_from_file_location(\n        \"__prefect_loader__\",\n        path,\n        # Support explicit relative imports i.e. `from .foo import bar`\n        submodule_search_locations=[parent_path, working_directory],\n    )\n    module = importlib.util.module_from_spec(spec)\n    sys.modules[\"__prefect_loader__\"] = module\n\n    # Support implicit relative imports i.e. `from foo import bar`\n    sys.path.insert(0, working_directory)\n    sys.path.insert(0, parent_path)\n    try:\n        spec.loader.exec_module(module)\n    except Exception as exc:\n        raise ScriptError(user_exc=exc, path=path) from exc\n    finally:\n        sys.modules.pop(\"__prefect_loader__\")\n        sys.path.remove(parent_path)\n        sys.path.remove(working_directory)\n\n    return module\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.objects_from_script","title":"objects_from_script","text":"

    Run a python script and return all the global variables

    Supports remote paths by copying to a local temporary file.

    WARNING: The Python documentation does not recommend using runpy for this pattern.

    Furthermore, any functions and classes defined by the executed code are not guaranteed to work correctly after a runpy function has returned. If that limitation is not acceptable for a given use case, importlib is likely to be a more suitable choice than this module.

    The function load_script_as_module uses importlib instead and should be used instead for loading objects from scripts.

    Parameters:

    Name Type Description Default path str

    The path to the script to run

    required text Union[str, bytes]

    Optionally, the text of the script. Skips loading the contents if given.

    None

    Returns:

    Type Description Dict[str, Any]

    A dictionary mapping variable name to value

    Raises:

    Type Description ScriptError

    if the script raises an exception during execution

    Source code in prefect/utilities/importtools.py
    def objects_from_script(path: str, text: Union[str, bytes] = None) -> Dict[str, Any]:\n    \"\"\"\n    Run a python script and return all the global variables\n\n    Supports remote paths by copying to a local temporary file.\n\n    WARNING: The Python documentation does not recommend using runpy for this pattern.\n\n    > Furthermore, any functions and classes defined by the executed code are not\n    > guaranteed to work correctly after a runpy function has returned. If that\n    > limitation is not acceptable for a given use case, importlib is likely to be a\n    > more suitable choice than this module.\n\n    The function `load_script_as_module` uses importlib instead and should be used\n    instead for loading objects from scripts.\n\n    Args:\n        path: The path to the script to run\n        text: Optionally, the text of the script. Skips loading the contents if given.\n\n    Returns:\n        A dictionary mapping variable name to value\n\n    Raises:\n        ScriptError: if the script raises an exception during execution\n    \"\"\"\n\n    def run_script(run_path: str):\n        # Cast to an absolute path before changing directories to ensure relative paths\n        # are not broken\n        abs_run_path = os.path.abspath(run_path)\n        with tmpchdir(run_path):\n            try:\n                return runpy.run_path(abs_run_path)\n            except Exception as exc:\n                raise ScriptError(user_exc=exc, path=path) from exc\n\n    if text:\n        with NamedTemporaryFile(\n            mode=\"wt\" if isinstance(text, str) else \"wb\",\n            prefix=f\"run-{filename(path)}\",\n            suffix=\".py\",\n        ) as tmpfile:\n            tmpfile.write(text)\n            tmpfile.flush()\n            return run_script(tmpfile.name)\n\n    else:\n        if not is_local_path(path):\n            # Remote paths need to be local to run\n            with fsspec.open(path) as f:\n                contents = f.read()\n            return objects_from_script(path, contents)\n        else:\n            return run_script(path)\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.to_qualified_name","title":"to_qualified_name","text":"

    Given an object, returns its fully-qualified name: a string that represents its Python import path.

    Parameters:

    Name Type Description Default obj Any

    an importable Python object

    required

    Returns:

    Name Type Description str str

    the qualified name

    Source code in prefect/utilities/importtools.py
    def to_qualified_name(obj: Any) -> str:\n    \"\"\"\n    Given an object, returns its fully-qualified name: a string that represents its\n    Python import path.\n\n    Args:\n        obj (Any): an importable Python object\n\n    Returns:\n        str: the qualified name\n    \"\"\"\n    if sys.version_info < (3, 10):\n        # These attributes are only available in Python 3.10+\n        if isinstance(obj, (classmethod, staticmethod)):\n            obj = obj.__func__\n    return obj.__module__ + \".\" + obj.__qualname__\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/math/","title":"math","text":"","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math","title":"prefect.utilities.math","text":"","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math.bounded_poisson_interval","title":"bounded_poisson_interval","text":"

    Bounds Poisson \"inter-arrival times\" to a range.

    Unlike clamped_poisson_interval this does not take a target average interval. Instead, the interval is predetermined and the average is calculated as their midpoint. This allows Poisson intervals to be used in cases where a lower bound must be enforced.

    Source code in prefect/utilities/math.py
    def bounded_poisson_interval(lower_bound, upper_bound):\n    \"\"\"\n    Bounds Poisson \"inter-arrival times\" to a range.\n\n    Unlike `clamped_poisson_interval` this does not take a target average interval.\n    Instead, the interval is predetermined and the average is calculated as their\n    midpoint. This allows Poisson intervals to be used in cases where a lower bound\n    must be enforced.\n    \"\"\"\n    average = (float(lower_bound) + float(upper_bound)) / 2.0\n    upper_rv = exponential_cdf(upper_bound, average)\n    lower_rv = exponential_cdf(lower_bound, average)\n    return poisson_interval(average, lower_rv, upper_rv)\n
    ","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math.clamped_poisson_interval","title":"clamped_poisson_interval","text":"

    Bounds Poisson \"inter-arrival times\" to a range defined by the clamping factor.

    The upper bound for this random variate is: average_interval * (1 + clamping_factor). A lower bound is picked so that the average interval remains approximately fixed.

    Source code in prefect/utilities/math.py
    def clamped_poisson_interval(average_interval, clamping_factor=0.3):\n    \"\"\"\n    Bounds Poisson \"inter-arrival times\" to a range defined by the clamping factor.\n\n    The upper bound for this random variate is: average_interval * (1 + clamping_factor).\n    A lower bound is picked so that the average interval remains approximately fixed.\n    \"\"\"\n    if clamping_factor <= 0:\n        raise ValueError(\"`clamping_factor` must be >= 0.\")\n\n    upper_clamp_multiple = 1 + clamping_factor\n    upper_bound = average_interval * upper_clamp_multiple\n    lower_bound = max(0, average_interval * lower_clamp_multiple(upper_clamp_multiple))\n\n    upper_rv = exponential_cdf(upper_bound, average_interval)\n    lower_rv = exponential_cdf(lower_bound, average_interval)\n    return poisson_interval(average_interval, lower_rv, upper_rv)\n
    ","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math.lower_clamp_multiple","title":"lower_clamp_multiple","text":"

    Computes a lower clamp multiple that can be used to bound a random variate drawn from an exponential distribution.

    Given an upper clamp multiple k (and corresponding upper bound k * average_interval), this function computes a lower clamp multiple c (corresponding to a lower bound c * average_interval) where the probability mass between the lower bound and the median is equal to the probability mass between the median and the upper bound.

    Source code in prefect/utilities/math.py
    def lower_clamp_multiple(k):\n    \"\"\"\n    Computes a lower clamp multiple that can be used to bound a random variate drawn\n    from an exponential distribution.\n\n    Given an upper clamp multiple `k` (and corresponding upper bound k * average_interval),\n    this function computes a lower clamp multiple `c` (corresponding to a lower bound\n    c * average_interval) where the probability mass between the lower bound and the\n    median is equal to the probability mass between the median and the upper bound.\n    \"\"\"\n    if k >= 50:\n        # return 0 for large values of `k` to prevent numerical overflow\n        return 0.0\n\n    return math.log(max(2**k / (2**k - 1), 1e-10), 2)\n
    ","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math.poisson_interval","title":"poisson_interval","text":"

    Generates an \"inter-arrival time\" for a Poisson process.

    Draws a random variable from an exponential distribution using the inverse-CDF method. Can optionally be passed a lower and upper bound between (0, 1] to clamp the potential output values.

    Source code in prefect/utilities/math.py
    def poisson_interval(average_interval, lower=0, upper=1):\n    \"\"\"\n    Generates an \"inter-arrival time\" for a Poisson process.\n\n    Draws a random variable from an exponential distribution using the inverse-CDF\n    method. Can optionally be passed a lower and upper bound between (0, 1] to clamp\n    the potential output values.\n    \"\"\"\n\n    # note that we ensure the argument to the logarithm is stabilized to prevent\n    # calling log(0), which results in a DomainError\n    return -math.log(max(1 - random.uniform(lower, upper), 1e-10)) * average_interval\n
    ","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/names/","title":"names","text":"","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/names/#prefect.utilities.names","title":"prefect.utilities.names","text":"","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/names/#prefect.utilities.names.generate_slug","title":"generate_slug","text":"

    Generates a random slug.

    Parameters:

    Name Type Description Default - n_words (int

    the number of words in the slug

    required Source code in prefect/utilities/names.py
    def generate_slug(n_words: int) -> str:\n    \"\"\"\n    Generates a random slug.\n\n    Args:\n        - n_words (int): the number of words in the slug\n    \"\"\"\n    words = coolname.generate(n_words)\n\n    # regenerate words if they include ignored words\n    while IGNORE_LIST.intersection(words):\n        words = coolname.generate(n_words)\n\n    return \"-\".join(words)\n
    ","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/names/#prefect.utilities.names.obfuscate","title":"obfuscate","text":"

    Obfuscates any data type's string representation. See obfuscate_string.

    Source code in prefect/utilities/names.py
    def obfuscate(s: Any, show_tail=False) -> str:\n    \"\"\"\n    Obfuscates any data type's string representation. See `obfuscate_string`.\n    \"\"\"\n    if s is None:\n        return OBFUSCATED_PREFIX + \"*\" * 4\n\n    return obfuscate_string(str(s), show_tail=show_tail)\n
    ","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/names/#prefect.utilities.names.obfuscate_string","title":"obfuscate_string","text":"

    Obfuscates a string by returning a new string of 8 characters. If the input string is longer than 10 characters and show_tail is True, then up to 4 of its final characters will become final characters of the obfuscated string; all other characters are \"*\".

    \"abc\" -> \"*\" \"abcdefgh\" -> \"*\" \"abcdefghijk\" -> \"*k\" \"abcdefghijklmnopqrs\" -> \"****pqrs\"

    Source code in prefect/utilities/names.py
    def obfuscate_string(s: str, show_tail=False) -> str:\n    \"\"\"\n    Obfuscates a string by returning a new string of 8 characters. If the input\n    string is longer than 10 characters and show_tail is True, then up to 4 of\n    its final characters will become final characters of the obfuscated string;\n    all other characters are \"*\".\n\n    \"abc\"      -> \"********\"\n    \"abcdefgh\" -> \"********\"\n    \"abcdefghijk\" -> \"*******k\"\n    \"abcdefghijklmnopqrs\" -> \"****pqrs\"\n    \"\"\"\n    result = OBFUSCATED_PREFIX + \"*\" * 4\n    # take up to 4 characters, but only after the 10th character\n    suffix = s[10:][-4:]\n    if suffix and show_tail:\n        result = f\"{result[:-len(suffix)]}{suffix}\"\n    return result\n
    ","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/processutils/","title":"processutils","text":""},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils","title":"prefect.utilities.processutils","text":""},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.forward_signal_handler","title":"forward_signal_handler","text":"

    Forward subsequent signum events (e.g. interrupts) to respective signums.

    Source code in prefect/utilities/processutils.py
    def forward_signal_handler(\n    pid: int, signum: int, *signums: int, process_name: str, print_fn: Callable\n):\n    \"\"\"Forward subsequent signum events (e.g. interrupts) to respective signums.\"\"\"\n    current_signal, future_signals = signums[0], signums[1:]\n\n    # avoid RecursionError when setting up a direct signal forward to the same signal for the main pid\n    avoid_infinite_recursion = signum == current_signal and pid == os.getpid()\n    if avoid_infinite_recursion:\n        # store the vanilla handler so it can be temporarily restored below\n        original_handler = signal.getsignal(current_signal)\n\n    def handler(*args):\n        print_fn(\n            f\"Received {getattr(signum, 'name', signum)}. \"\n            f\"Sending {getattr(current_signal, 'name', current_signal)} to\"\n            f\" {process_name} (PID {pid})...\"\n        )\n        if avoid_infinite_recursion:\n            signal.signal(current_signal, original_handler)\n        os.kill(pid, current_signal)\n        if future_signals:\n            forward_signal_handler(\n                pid,\n                signum,\n                *future_signals,\n                process_name=process_name,\n                print_fn=print_fn,\n            )\n\n    # register current and future signal handlers\n    _register_signal(signum, handler)\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.open_process","title":"open_process async","text":"

    Like anyio.open_process but with: - Support for Windows command joining - Termination of the process on exception during yield - Forced cleanup of process resources during cancellation

    Source code in prefect/utilities/processutils.py
    @asynccontextmanager\nasync def open_process(command: List[str], **kwargs):\n    \"\"\"\n    Like `anyio.open_process` but with:\n    - Support for Windows command joining\n    - Termination of the process on exception during yield\n    - Forced cleanup of process resources during cancellation\n    \"\"\"\n    # Passing a string to open_process is equivalent to shell=True which is\n    # generally necessary for Unix-like commands on Windows but otherwise should\n    # be avoided\n    if not isinstance(command, list):\n        raise TypeError(\n            \"The command passed to open process must be a list. You passed the command\"\n            f\"'{command}', which is type '{type(command)}'.\"\n        )\n\n    if sys.platform == \"win32\":\n        command = \" \".join(command)\n        process = await _open_anyio_process(command, **kwargs)\n    else:\n        process = await anyio.open_process(command, **kwargs)\n\n    # if there's a creationflags kwarg and it contains CREATE_NEW_PROCESS_GROUP,\n    # use SetConsoleCtrlHandler to handle CTRL-C\n    win32_process_group = False\n    if (\n        sys.platform == \"win32\"\n        and \"creationflags\" in kwargs\n        and kwargs[\"creationflags\"] & subprocess.CREATE_NEW_PROCESS_GROUP\n    ):\n        win32_process_group = True\n        _windows_process_group_pids.add(process.pid)\n        # Add a handler for CTRL-C. Re-adding the handler is safe as Windows\n        # will not add a duplicate handler if _win32_ctrl_handler is\n        # already registered.\n        windll.kernel32.SetConsoleCtrlHandler(_win32_ctrl_handler, 1)\n\n    try:\n        async with process:\n            yield process\n    finally:\n        try:\n            process.terminate()\n            if win32_process_group:\n                _windows_process_group_pids.remove(process.pid)\n\n        except OSError:\n            # Occurs if the process is already terminated\n            pass\n\n        # Ensure the process resource is closed. If not shielded from cancellation,\n        # this resource can be left open and the subprocess output can appear after\n        # the parent process has exited.\n        with anyio.CancelScope(shield=True):\n            await process.aclose()\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.run_process","title":"run_process async","text":"

    Like anyio.run_process but with:

    • Use of our open_process utility to ensure resources are cleaned up
    • Simple stream_output support to connect the subprocess to the parent stdout/err
    • Support for submission with TaskGroup.start marking as 'started' after the process has been created. When used, the PID is returned to the task status.
    Source code in prefect/utilities/processutils.py
    async def run_process(\n    command: List[str],\n    stream_output: Union[bool, Tuple[Optional[TextSink], Optional[TextSink]]] = False,\n    task_status: Optional[anyio.abc.TaskStatus] = None,\n    task_status_handler: Optional[Callable[[anyio.abc.Process], Any]] = None,\n    **kwargs,\n):\n    \"\"\"\n    Like `anyio.run_process` but with:\n\n    - Use of our `open_process` utility to ensure resources are cleaned up\n    - Simple `stream_output` support to connect the subprocess to the parent stdout/err\n    - Support for submission with `TaskGroup.start` marking as 'started' after the\n        process has been created. When used, the PID is returned to the task status.\n\n    \"\"\"\n    if stream_output is True:\n        stream_output = (sys.stdout, sys.stderr)\n\n    async with open_process(\n        command,\n        stdout=subprocess.PIPE if stream_output else subprocess.DEVNULL,\n        stderr=subprocess.PIPE if stream_output else subprocess.DEVNULL,\n        **kwargs,\n    ) as process:\n        if task_status is not None:\n            if not task_status_handler:\n\n                def task_status_handler(process):\n                    return process.pid\n\n            task_status.started(task_status_handler(process))\n\n        if stream_output:\n            await consume_process_output(\n                process, stdout_sink=stream_output[0], stderr_sink=stream_output[1]\n            )\n\n        await process.wait()\n\n    return process\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.setup_signal_handlers_agent","title":"setup_signal_handlers_agent","text":"

    Handle interrupts of the agent gracefully.

    Source code in prefect/utilities/processutils.py
    def setup_signal_handlers_agent(pid: int, process_name: str, print_fn: Callable):\n    \"\"\"Handle interrupts of the agent gracefully.\"\"\"\n    setup_handler = partial(\n        forward_signal_handler, pid, process_name=process_name, print_fn=print_fn\n    )\n    # when agent receives SIGINT, it stops dequeueing new FlowRuns, and runs until the subprocesses finish\n    # the signal is not forwarded to subprocesses, so they can continue to run and hopefully still complete\n    if sys.platform == \"win32\":\n        # on Windows, use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        setup_handler(signal.SIGINT, signal.CTRL_BREAK_EVENT)\n    else:\n        # forward first SIGINT directly, send SIGKILL on subsequent interrupt\n        setup_handler(signal.SIGINT, signal.SIGINT, signal.SIGKILL)\n        # first SIGTERM: send SIGINT, send SIGKILL on subsequent SIGTERM\n        setup_handler(signal.SIGTERM, signal.SIGINT, signal.SIGKILL)\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.setup_signal_handlers_server","title":"setup_signal_handlers_server","text":"

    Handle interrupts of the server gracefully.

    Source code in prefect/utilities/processutils.py
    def setup_signal_handlers_server(pid: int, process_name: str, print_fn: Callable):\n    \"\"\"Handle interrupts of the server gracefully.\"\"\"\n    setup_handler = partial(\n        forward_signal_handler, pid, process_name=process_name, print_fn=print_fn\n    )\n    # when server receives a signal, it needs to be propagated to the uvicorn subprocess\n    if sys.platform == \"win32\":\n        # on Windows, use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        setup_handler(signal.SIGINT, signal.CTRL_BREAK_EVENT)\n    else:\n        # first interrupt: SIGTERM, second interrupt: SIGKILL\n        setup_handler(signal.SIGINT, signal.SIGTERM, signal.SIGKILL)\n        # forward first SIGTERM directly, send SIGKILL on subsequent SIGTERM\n        setup_handler(signal.SIGTERM, signal.SIGTERM, signal.SIGKILL)\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.setup_signal_handlers_worker","title":"setup_signal_handlers_worker","text":"

    Handle interrupts of workers gracefully.

    Source code in prefect/utilities/processutils.py
    def setup_signal_handlers_worker(pid: int, process_name: str, print_fn: Callable):\n    \"\"\"Handle interrupts of workers gracefully.\"\"\"\n    setup_handler = partial(\n        forward_signal_handler, pid, process_name=process_name, print_fn=print_fn\n    )\n    # when agent receives SIGINT, it stops dequeueing new FlowRuns, and runs until the subprocesses finish\n    # the signal is not forwarded to subprocesses, so they can continue to run and hopefully still complete\n    if sys.platform == \"win32\":\n        # on Windows, use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        setup_handler(signal.SIGINT, signal.CTRL_BREAK_EVENT)\n    else:\n        # forward first SIGINT directly, send SIGKILL on subsequent interrupt\n        setup_handler(signal.SIGINT, signal.SIGINT, signal.SIGKILL)\n        # first SIGTERM: send SIGINT, send SIGKILL on subsequent SIGTERM\n        setup_handler(signal.SIGTERM, signal.SIGINT, signal.SIGKILL)\n
    "},{"location":"api-ref/prefect/utilities/pydantic/","title":"pydantic","text":"","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic","title":"prefect.utilities.pydantic","text":"","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic.PartialModel","title":"PartialModel","text":"

    Bases: Generic[M]

    A utility for creating a Pydantic model in several steps.

    Fields may be set at initialization, via attribute assignment, or at finalization when the concrete model is returned.

    Pydantic validation does not occur until finalization.

    Each field can only be set once and a ValueError will be raised on assignment if a field already has a value.

    Example

    class MyModel(pydantic.BaseModel): x: int y: str z: float

    partial_model = PartialModel(MyModel, x=1) partial_model.y = \"two\" model = partial_model.finalize(z=3.0)

    Source code in prefect/utilities/pydantic.py
    class PartialModel(Generic[M]):\n    \"\"\"\n    A utility for creating a Pydantic model in several steps.\n\n    Fields may be set at initialization, via attribute assignment, or at finalization\n    when the concrete model is returned.\n\n    Pydantic validation does not occur until finalization.\n\n    Each field can only be set once and a `ValueError` will be raised on assignment if\n    a field already has a value.\n\n    Example:\n        >>> class MyModel(pydantic.BaseModel):\n        >>>     x: int\n        >>>     y: str\n        >>>     z: float\n        >>>\n        >>> partial_model = PartialModel(MyModel, x=1)\n        >>> partial_model.y = \"two\"\n        >>> model = partial_model.finalize(z=3.0)\n    \"\"\"\n\n    def __init__(self, __model_cls: Type[M], **kwargs: Any) -> None:\n        self.fields = kwargs\n        # Set fields first to avoid issues if `fields` is also set on the `model_cls`\n        # in our custom `setattr` implementation.\n        self.model_cls = __model_cls\n\n        for name in kwargs.keys():\n            self.raise_if_not_in_model(name)\n\n    def finalize(self, **kwargs: Any) -> M:\n        for name in kwargs.keys():\n            self.raise_if_already_set(name)\n            self.raise_if_not_in_model(name)\n        return self.model_cls(**self.fields, **kwargs)\n\n    def raise_if_already_set(self, name):\n        if name in self.fields:\n            raise ValueError(f\"Field {name!r} has already been set.\")\n\n    def raise_if_not_in_model(self, name):\n        if name not in self.model_cls.__fields__:\n            raise ValueError(f\"Field {name!r} is not present in the model.\")\n\n    def __setattr__(self, __name: str, __value: Any) -> None:\n        if __name in {\"fields\", \"model_cls\"}:\n            return super().__setattr__(__name, __value)\n\n        self.raise_if_already_set(__name)\n        self.raise_if_not_in_model(__name)\n        self.fields[__name] = __value\n\n    def __repr__(self) -> str:\n        dsp_fields = \", \".join(\n            f\"{key}={repr(value)}\" for key, value in self.fields.items()\n        )\n        return f\"PartialModel(cls={self.model_cls.__name__}, {dsp_fields})\"\n
    ","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic.add_cloudpickle_reduction","title":"add_cloudpickle_reduction","text":"

    Adds a __reducer__ to the given class that ensures it is cloudpickle compatible.

    Workaround for issues with cloudpickle when using cythonized pydantic which throws exceptions when attempting to pickle the class which has \"compiled\" validator methods dynamically attached to it.

    We cannot define this utility in the model class itself because the class is the type that contains unserializable methods.

    Any model using some features of Pydantic (e.g. Path validation) with a Cython compiled Pydantic installation may encounter pickling issues.

    See related issue at https://github.com/cloudpipe/cloudpickle/issues/408

    Source code in prefect/utilities/pydantic.py
    def add_cloudpickle_reduction(__model_cls: Type[M] = None, **kwargs: Any):\n    \"\"\"\n    Adds a `__reducer__` to the given class that ensures it is cloudpickle compatible.\n\n    Workaround for issues with cloudpickle when using cythonized pydantic which\n    throws exceptions when attempting to pickle the class which has \"compiled\"\n    validator methods dynamically attached to it.\n\n    We cannot define this utility in the model class itself because the class is the\n    type that contains unserializable methods.\n\n    Any model using some features of Pydantic (e.g. `Path` validation) with a Cython\n    compiled Pydantic installation may encounter pickling issues.\n\n    See related issue at https://github.com/cloudpipe/cloudpickle/issues/408\n    \"\"\"\n    if __model_cls:\n        __model_cls.__reduce__ = _reduce_model\n        __model_cls.__reduce_kwargs__ = kwargs\n        return __model_cls\n    else:\n        return cast(\n            Callable[[Type[M]], Type[M]],\n            partial(\n                add_cloudpickle_reduction,\n                **kwargs,\n            ),\n        )\n
    ","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic.add_type_dispatch","title":"add_type_dispatch","text":"

    Extend a Pydantic model to add a 'type' field that is used a discriminator field to dynamically determine the subtype that when deserializing models.

    This allows automatic resolution to subtypes of the decorated model.

    If a type field already exists, it should be a string literal field that has a constant value for each subclass. The default value of this field will be used as the dispatch key.

    If a type field does not exist, one will be added. In this case, the value of the field will be set to the value of the __dispatch_key__. The base class should define a __dispatch_key__ class method that is used to determine the unique key for each subclass. Alternatively, each subclass can define the __dispatch_key__ as a string literal.

    The base class must not define a 'type' field. If it is not desirable to add a field to the model and the dispatch key can be tracked separately, the lower level utilities in prefect.utilities.dispatch should be used directly.

    Source code in prefect/utilities/pydantic.py
    def add_type_dispatch(model_cls: Type[M]) -> Type[M]:\n    \"\"\"\n    Extend a Pydantic model to add a 'type' field that is used a discriminator field\n    to dynamically determine the subtype that when deserializing models.\n\n    This allows automatic resolution to subtypes of the decorated model.\n\n    If a type field already exists, it should be a string literal field that has a\n    constant value for each subclass. The default value of this field will be used as\n    the dispatch key.\n\n    If a type field does not exist, one will be added. In this case, the value of the\n    field will be set to the value of the `__dispatch_key__`. The base class should\n    define a `__dispatch_key__` class method that is used to determine the unique key\n    for each subclass. Alternatively, each subclass can define the `__dispatch_key__`\n    as a string literal.\n\n    The base class must not define a 'type' field. If it is not desirable to add a field\n    to the model and the dispatch key can be tracked separately, the lower level\n    utilities in `prefect.utilities.dispatch` should be used directly.\n    \"\"\"\n    defines_dispatch_key = hasattr(\n        model_cls, \"__dispatch_key__\"\n    ) or \"__dispatch_key__\" in getattr(model_cls, \"__annotations__\", {})\n\n    defines_type_field = \"type\" in model_cls.__fields__\n\n    if not defines_dispatch_key and not defines_type_field:\n        raise ValueError(\n            f\"Model class {model_cls.__name__!r} does not define a `__dispatch_key__` \"\n            \"or a type field. One of these is required for dispatch.\"\n        )\n\n    elif defines_dispatch_key and not defines_type_field:\n        # Add a type field to store the value of the dispatch key\n        model_cls.__fields__[\"type\"] = pydantic.fields.ModelField(\n            name=\"type\",\n            type_=str,\n            required=True,\n            class_validators=None,\n            model_config=model_cls.__config__,\n        )\n\n    elif not defines_dispatch_key and defines_type_field:\n        field_type_annotation = model_cls.__fields__[\"type\"].type_\n        if field_type_annotation != str:\n            raise TypeError(\n                f\"Model class {model_cls.__name__!r} defines a 'type' field with \"\n                f\"type {field_type_annotation.__name__!r} but it must be 'str'.\"\n            )\n\n        # Set the dispatch key to retrieve the value from the type field\n        @classmethod\n        def dispatch_key_from_type_field(cls):\n            return cls.__fields__[\"type\"].default\n\n        model_cls.__dispatch_key__ = dispatch_key_from_type_field\n\n    else:\n        raise ValueError(\n            f\"Model class {model_cls.__name__!r} defines a `__dispatch_key__` \"\n            \"and a type field. Only one of these may be defined for dispatch.\"\n        )\n\n    cls_init = model_cls.__init__\n    cls_new = model_cls.__new__\n\n    def __init__(__pydantic_self__, **data: Any) -> None:\n        type_string = (\n            get_dispatch_key(__pydantic_self__)\n            if type(__pydantic_self__) != model_cls\n            else \"__base__\"\n        )\n        data.setdefault(\"type\", type_string)\n        cls_init(__pydantic_self__, **data)\n\n    def __new__(cls: Type[Self], **kwargs) -> Self:\n        if \"type\" in kwargs:\n            try:\n                subcls = lookup_type(cls, dispatch_key=kwargs[\"type\"])\n            except KeyError as exc:\n                raise pydantic.ValidationError(errors=[exc], model=cls)\n            return cls_new(subcls)\n        else:\n            return cls_new(cls)\n\n    model_cls.__init__ = __init__\n    model_cls.__new__ = __new__\n\n    register_base_type(model_cls)\n\n    return model_cls\n
    ","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic.get_class_fields_only","title":"get_class_fields_only","text":"

    Gets all the field names defined on the model class but not any parent classes. Any fields that are on the parent but redefined on the subclass are included.

    Source code in prefect/utilities/pydantic.py
    def get_class_fields_only(model: Type[pydantic.BaseModel]) -> set:\n    \"\"\"\n    Gets all the field names defined on the model class but not any parent classes.\n    Any fields that are on the parent but redefined on the subclass are included.\n    \"\"\"\n    subclass_class_fields = set(model.__annotations__.keys())\n    parent_class_fields = set()\n\n    for base in model.__class__.__bases__:\n        if issubclass(base, pydantic.BaseModel):\n            parent_class_fields.update(base.__annotations__.keys())\n\n    return (subclass_class_fields - parent_class_fields) | (\n        subclass_class_fields & parent_class_fields\n    )\n
    ","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/render_swagger/","title":"render_swagger","text":"","tags":["Python API","Swagger"]},{"location":"api-ref/prefect/utilities/render_swagger/#prefect.utilities.render_swagger","title":"prefect.utilities.render_swagger","text":"","tags":["Python API","Swagger"]},{"location":"api-ref/prefect/utilities/render_swagger/#prefect.utilities.render_swagger.swagger_lib","title":"swagger_lib","text":"

    Provides the actual swagger library used

    Source code in prefect/utilities/render_swagger.py
    def swagger_lib(config) -> dict:\n    \"\"\"\n    Provides the actual swagger library used\n    \"\"\"\n    lib_swagger = {\n        \"css\": \"https://unpkg.com/swagger-ui-dist@3/swagger-ui.css\",\n        \"js\": \"https://unpkg.com/swagger-ui-dist@3/swagger-ui-bundle.js\",\n    }\n\n    extra_javascript = config.get(\"extra_javascript\", [])\n    extra_css = config.get(\"extra_css\", [])\n    for lib in extra_javascript:\n        if os.path.basename(urllib.parse.urlparse(lib).path) == \"swagger-ui-bundle.js\":\n            lib_swagger[\"js\"] = lib\n            break\n\n    for css in extra_css:\n        if os.path.basename(urllib.parse.urlparse(css).path) == \"swagger-ui.css\":\n            lib_swagger[\"css\"] = css\n            break\n    return lib_swagger\n
    ","tags":["Python API","Swagger"]},{"location":"api-ref/prefect/utilities/services/","title":"services","text":"","tags":["Python API","services"]},{"location":"api-ref/prefect/utilities/services/#prefect.utilities.services","title":"prefect.utilities.services","text":"","tags":["Python API","services"]},{"location":"api-ref/prefect/utilities/services/#prefect.utilities.services.critical_service_loop","title":"critical_service_loop async","text":"

    Runs the given workload function on the specified interval, while being forgiving of intermittent issues like temporary HTTP errors. If more than a certain number of consecutive errors occur, print a summary of up to memory recent exceptions to printer, then begin backoff.

    The loop will exit after reaching the consecutive error limit backoff times. On each backoff, the interval will be doubled. On a successful loop, the backoff will be reset.

    Parameters:

    Name Type Description Default workload Callable[..., Coroutine]

    the function to call

    required interval float

    how frequently to call it

    required memory int

    how many recent errors to remember

    10 consecutive int

    how many consecutive errors must we see before we begin backoff

    3 backoff int

    how many times we should allow consecutive errors before exiting

    1 printer Callable[..., None]

    a print-like function where errors will be reported

    print run_once bool

    if set, the loop will only run once then return

    False jitter_range float

    if set, the interval will be a random variable (rv) drawn from a clamped Poisson distribution where lambda = interval and the rv is bound between interval * (1 - range) < rv < interval * (1 + range)

    None Source code in prefect/utilities/services.py
    async def critical_service_loop(\n    workload: Callable[..., Coroutine],\n    interval: float,\n    memory: int = 10,\n    consecutive: int = 3,\n    backoff: int = 1,\n    printer: Callable[..., None] = print,\n    run_once: bool = False,\n    jitter_range: float = None,\n):\n    \"\"\"\n    Runs the given `workload` function on the specified `interval`, while being\n    forgiving of intermittent issues like temporary HTTP errors.  If more than a certain\n    number of `consecutive` errors occur, print a summary of up to `memory` recent\n    exceptions to `printer`, then begin backoff.\n\n    The loop will exit after reaching the consecutive error limit `backoff` times.\n    On each backoff, the interval will be doubled. On a successful loop, the backoff\n    will be reset.\n\n    Args:\n        workload: the function to call\n        interval: how frequently to call it\n        memory: how many recent errors to remember\n        consecutive: how many consecutive errors must we see before we begin backoff\n        backoff: how many times we should allow consecutive errors before exiting\n        printer: a `print`-like function where errors will be reported\n        run_once: if set, the loop will only run once then return\n        jitter_range: if set, the interval will be a random variable (rv) drawn from\n            a clamped Poisson distribution where lambda = interval and the rv is bound\n            between `interval * (1 - range) < rv < interval * (1 + range)`\n    \"\"\"\n\n    track_record: Deque[bool] = deque([True] * consecutive, maxlen=consecutive)\n    failures: Deque[Tuple[Exception, TracebackType]] = deque(maxlen=memory)\n    backoff_count = 0\n\n    while True:\n        try:\n            logger.debug(f\"Starting run of {workload!r}\")\n            await workload()\n\n            # Reset the backoff count on success; we may want to consider resetting\n            # this only if the track record is _all_ successful to avoid ending backoff\n            # prematurely\n            if backoff_count > 0:\n                printer(\"Resetting backoff due to successful run.\")\n                backoff_count = 0\n\n            track_record.append(True)\n        except httpx.TransportError as exc:\n            # httpx.TransportError is the base class for any kind of communications\n            # error, like timeouts, connection failures, etc.  This does _not_ cover\n            # routine HTTP error codes (even 5xx errors like 502/503) so this\n            # handler should not be attempting to cover cases where the Prefect server\n            # or Prefect Cloud is having an outage (which will be covered by the\n            # exception clause below)\n            track_record.append(False)\n            failures.append((exc, sys.exc_info()[-1]))\n            logger.debug(\n                f\"Run of {workload!r} failed with TransportError\", exc_info=exc\n            )\n        except httpx.HTTPStatusError as exc:\n            if exc.response.status_code >= 500:\n                # 5XX codes indicate a potential outage of the Prefect API which is\n                # likely to be temporary and transient.  Don't quit over these unless\n                # it is prolonged.\n                track_record.append(False)\n                failures.append((exc, sys.exc_info()[-1]))\n                logger.debug(\n                    f\"Run of {workload!r} failed with HTTPStatusError\", exc_info=exc\n                )\n            else:\n                raise\n\n        # Decide whether to exit now based on recent history.\n        #\n        # Given some typical background error rate of, say, 1%, we may still observe\n        # quite a few errors in our recent samples, but this is not necessarily a cause\n        # for concern.\n        #\n        # Imagine two distributions that could reflect our situation at any time: the\n        # everything-is-fine distribution of errors, and the everything-is-on-fire\n        # distribution of errors. We are trying to determine which of those two worlds\n        # we are currently experiencing.  We compare the likelihood that we'd draw N\n        # consecutive errors from each.  In the everything-is-fine distribution, that\n        # would be a very low-probability occurrence, but in the everything-is-on-fire\n        # distribution, that is a high-probability occurrence.\n        #\n        # Remarkably, we only need to look back for a small number of consecutive\n        # errors to have reasonable confidence that this is indeed an anomaly.\n        # @anticorrelator and @chrisguidry estimated that we should only need to look\n        # back for 3 consecutive errors.\n        if not any(track_record):\n            # We've failed enough times to be sure something is wrong, the writing is\n            # on the wall.  Let's explain what we've seen and exit.\n            printer(\n                f\"\\nFailed the last {consecutive} attempts. \"\n                \"Please check your environment and configuration.\"\n            )\n\n            printer(\"Examples of recent errors:\\n\")\n\n            failures_by_type = distinct(\n                reversed(failures),\n                key=lambda pair: type(pair[0]),  # Group by the type of exception\n            )\n            for exception, traceback in failures_by_type:\n                printer(\"\".join(format_exception(None, exception, traceback)))\n                printer()\n\n            backoff_count += 1\n\n            if backoff_count >= backoff:\n                raise RuntimeError(\"Service exceeded error threshold.\")\n\n            # Reset the track record\n            track_record.extend([True] * consecutive)\n            failures.clear()\n            printer(\n                \"Backing off due to consecutive errors, using increased interval of \"\n                f\" {interval * 2**backoff_count}s.\"\n            )\n\n        if run_once:\n            return\n\n        if jitter_range is not None:\n            sleep = clamped_poisson_interval(interval, clamping_factor=jitter_range)\n        else:\n            sleep = interval * 2**backoff_count\n\n        await anyio.sleep(sleep)\n
    ","tags":["Python API","services"]},{"location":"api-ref/prefect/utilities/slugify/","title":"slugify","text":"","tags":["Python API","slugify"]},{"location":"api-ref/prefect/utilities/slugify/#prefect.utilities.slugify","title":"prefect.utilities.slugify","text":"","tags":["Python API","slugify"]},{"location":"api-ref/prefect/utilities/templating/","title":"templating","text":"","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating","title":"prefect.utilities.templating","text":"","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.apply_values","title":"apply_values","text":"

    Replaces placeholders in a template with values from a supplied dictionary.

    Will recursively replace placeholders in dictionaries and lists.

    If a value has no placeholders, it will be returned unchanged.

    If a template contains only a single placeholder, the placeholder will be fully replaced with the value.

    If a template contains text before or after a placeholder or there are multiple placeholders, the placeholders will be replaced with the corresponding variable values.

    If a template contains a placeholder that is not in values, NotSet will be returned to signify that no placeholder replacement occurred. If template is a dictionary that contains a key with a value of NotSet, the key will be removed in the return value unless remove_notset is set to False.

    Parameters:

    Name Type Description Default template T

    template to discover and replace values in

    required values Dict[str, Any]

    The values to apply to placeholders in the template

    required remove_notset bool

    If True, remove keys with an unset value

    True

    Returns:

    Type Description Union[T, Type[NotSet]]

    The template with the values applied

    Source code in prefect/utilities/templating.py
    def apply_values(\n    template: T, values: Dict[str, Any], remove_notset: bool = True\n) -> Union[T, Type[NotSet]]:\n    \"\"\"\n    Replaces placeholders in a template with values from a supplied dictionary.\n\n    Will recursively replace placeholders in dictionaries and lists.\n\n    If a value has no placeholders, it will be returned unchanged.\n\n    If a template contains only a single placeholder, the placeholder will be\n    fully replaced with the value.\n\n    If a template contains text before or after a placeholder or there are\n    multiple placeholders, the placeholders will be replaced with the\n    corresponding variable values.\n\n    If a template contains a placeholder that is not in `values`, NotSet will\n    be returned to signify that no placeholder replacement occurred. If\n    `template` is a dictionary that contains a key with a value of NotSet,\n    the key will be removed in the return value unless `remove_notset` is set to False.\n\n    Args:\n        template: template to discover and replace values in\n        values: The values to apply to placeholders in the template\n        remove_notset: If True, remove keys with an unset value\n\n    Returns:\n        The template with the values applied\n    \"\"\"\n    if isinstance(template, (int, float, bool, type(NotSet), type(None))):\n        return template\n    if isinstance(template, str):\n        placeholders = find_placeholders(template)\n        if not placeholders:\n            # If there are no values, we can just use the template\n            return template\n        elif (\n            len(placeholders) == 1\n            and list(placeholders)[0].full_match == template\n            and list(placeholders)[0].type is PlaceholderType.STANDARD\n        ):\n            # If there is only one variable with no surrounding text,\n            # we can replace it. If there is no variable value, we\n            # return NotSet to indicate that the value should not be included.\n            return get_from_dict(values, list(placeholders)[0].name, NotSet)\n        else:\n            for full_match, name, placeholder_type in placeholders:\n                if placeholder_type is PlaceholderType.STANDARD:\n                    value = get_from_dict(values, name, NotSet)\n                elif placeholder_type is PlaceholderType.ENV_VAR:\n                    name = name.lstrip(ENV_VAR_PLACEHOLDER_PREFIX)\n                    value = os.environ.get(name, NotSet)\n                else:\n                    continue\n\n                if value is NotSet and not remove_notset:\n                    continue\n                elif value is NotSet:\n                    template = template.replace(full_match, \"\")\n                else:\n                    template = template.replace(full_match, str(value))\n\n            return template\n    elif isinstance(template, dict):\n        updated_template = {}\n        for key, value in template.items():\n            updated_value = apply_values(value, values, remove_notset=remove_notset)\n            if updated_value is not NotSet:\n                updated_template[key] = updated_value\n            elif not remove_notset:\n                updated_template[key] = value\n\n        return updated_template\n    elif isinstance(template, list):\n        updated_list = []\n        for value in template:\n            updated_value = apply_values(value, values, remove_notset=remove_notset)\n            if updated_value is not NotSet:\n                updated_list.append(updated_value)\n        return updated_list\n    else:\n        raise ValueError(f\"Unexpected template type {type(template).__name__!r}\")\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.determine_placeholder_type","title":"determine_placeholder_type","text":"

    Determines the type of a placeholder based on its name.

    Parameters:

    Name Type Description Default name str

    The name of the placeholder

    required

    Returns:

    Type Description PlaceholderType

    The type of the placeholder

    Source code in prefect/utilities/templating.py
    def determine_placeholder_type(name: str) -> PlaceholderType:\n    \"\"\"\n    Determines the type of a placeholder based on its name.\n\n    Args:\n        name: The name of the placeholder\n\n    Returns:\n        The type of the placeholder\n    \"\"\"\n    if name.startswith(BLOCK_DOCUMENT_PLACEHOLDER_PREFIX):\n        return PlaceholderType.BLOCK_DOCUMENT\n    elif name.startswith(VARIABLE_PLACEHOLDER_PREFIX):\n        return PlaceholderType.VARIABLE\n    elif name.startswith(ENV_VAR_PLACEHOLDER_PREFIX):\n        return PlaceholderType.ENV_VAR\n    else:\n        return PlaceholderType.STANDARD\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.find_placeholders","title":"find_placeholders","text":"

    Finds all placeholders in a template.

    Parameters:

    Name Type Description Default template T

    template to discover placeholders in

    required

    Returns:

    Type Description Set[Placeholder]

    A set of all placeholders in the template

    Source code in prefect/utilities/templating.py
    def find_placeholders(template: T) -> Set[Placeholder]:\n    \"\"\"\n    Finds all placeholders in a template.\n\n    Args:\n        template: template to discover placeholders in\n\n    Returns:\n        A set of all placeholders in the template\n    \"\"\"\n    if isinstance(template, (int, float, bool)):\n        return set()\n    if isinstance(template, str):\n        result = PLACEHOLDER_CAPTURE_REGEX.findall(template)\n        return {\n            Placeholder(full_match, name, determine_placeholder_type(name))\n            for full_match, name in result\n        }\n    elif isinstance(template, dict):\n        return set().union(\n            *[find_placeholders(value) for key, value in template.items()]\n        )\n    elif isinstance(template, list):\n        return set().union(*[find_placeholders(item) for item in template])\n    else:\n        raise ValueError(f\"Unexpected type: {type(template)}\")\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.resolve_block_document_references","title":"resolve_block_document_references async","text":"

    Resolve block document references in a template by replacing each reference with the data of the block document.

    Recursively searches for block document references in dictionaries and lists.

    Identifies block document references by the as dictionary with the following structure:

    {\n    \"$ref\": {\n        \"block_document_id\": <block_document_id>\n    }\n}\n
    where <block_document_id> is the ID of the block document to resolve.

    Once the block document is retrieved from the API, the data of the block document is used to replace the reference.

    Parameters:

    Name Type Description Default template T

    The template to resolve block documents in

    required

    Returns:

    Type Description Union[T, Dict[str, Any]]

    The template with block documents resolved

    Source code in prefect/utilities/templating.py
    @inject_client\nasync def resolve_block_document_references(\n    template: T, client: \"PrefectClient\" = None\n) -> Union[T, Dict[str, Any]]:\n    \"\"\"\n    Resolve block document references in a template by replacing each reference with\n    the data of the block document.\n\n    Recursively searches for block document references in dictionaries and lists.\n\n    Identifies block document references by the as dictionary with the following\n    structure:\n    ```\n    {\n        \"$ref\": {\n            \"block_document_id\": <block_document_id>\n        }\n    }\n    ```\n    where `<block_document_id>` is the ID of the block document to resolve.\n\n    Once the block document is retrieved from the API, the data of the block document\n    is used to replace the reference.\n\n    Args:\n        template: The template to resolve block documents in\n\n    Returns:\n        The template with block documents resolved\n    \"\"\"\n    if isinstance(template, dict):\n        block_document_id = template.get(\"$ref\", {}).get(\"block_document_id\")\n        if block_document_id:\n            block_document = await client.read_block_document(block_document_id)\n            return block_document.data\n        updated_template = {}\n        for key, value in template.items():\n            updated_value = await resolve_block_document_references(\n                value, client=client\n            )\n            updated_template[key] = updated_value\n        return updated_template\n    elif isinstance(template, list):\n        return [\n            await resolve_block_document_references(item, client=client)\n            for item in template\n        ]\n    elif isinstance(template, str):\n        placeholders = find_placeholders(template)\n        has_block_document_placeholder = any(\n            placeholder.type is PlaceholderType.BLOCK_DOCUMENT\n            for placeholder in placeholders\n        )\n        if len(placeholders) == 0 or not has_block_document_placeholder:\n            return template\n        elif (\n            len(placeholders) == 1\n            and list(placeholders)[0].full_match == template\n            and list(placeholders)[0].type is PlaceholderType.BLOCK_DOCUMENT\n        ):\n            block_type_slug, block_document_name = (\n                list(placeholders)[0]\n                .name.replace(BLOCK_DOCUMENT_PLACEHOLDER_PREFIX, \"\")\n                .split(\".\")\n            )\n            block_document = await client.read_block_document_by_name(\n                name=block_document_name, block_type_slug=block_type_slug\n            )\n            # Handling for system blocks like Secret that have a value field\n            # These blocks will be replaced by variables in the future and this\n            # logic can be removed at that time.\n            value = block_document.data.get(\"value\", block_document.data)\n            return value\n        else:\n            raise ValueError(\n                f\"Invalid template: {template!r}. Only a single block placeholder is\"\n                \" allowed in a string and no surrounding text is allowed.\"\n            )\n\n    return template\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.resolve_variables","title":"resolve_variables async","text":"

    Resolve variables in a template by replacing each variable placeholder with the value of the variable.

    Recursively searches for variable placeholders in dictionaries and lists.

    Strips variable placeholders if the variable is not found.

    Parameters:

    Name Type Description Default template T

    The template to resolve variables in

    required

    Returns:

    Type Description

    The template with variables resolved

    Source code in prefect/utilities/templating.py
    @inject_client\nasync def resolve_variables(template: T, client: \"PrefectClient\" = None):\n    \"\"\"\n    Resolve variables in a template by replacing each variable placeholder with the\n    value of the variable.\n\n    Recursively searches for variable placeholders in dictionaries and lists.\n\n    Strips variable placeholders if the variable is not found.\n\n    Args:\n        template: The template to resolve variables in\n\n    Returns:\n        The template with variables resolved\n    \"\"\"\n    if isinstance(template, str):\n        placeholders = find_placeholders(template)\n        has_variable_placeholder = any(\n            placeholder.type is PlaceholderType.VARIABLE for placeholder in placeholders\n        )\n        if not placeholders or not has_variable_placeholder:\n            # If there are no values, we can just use the template\n            return template\n        elif (\n            len(placeholders) == 1\n            and list(placeholders)[0].full_match == template\n            and list(placeholders)[0].type is PlaceholderType.VARIABLE\n        ):\n            variable_name = list(placeholders)[0].name.replace(\n                VARIABLE_PLACEHOLDER_PREFIX, \"\"\n            )\n            variable = await client.read_variable_by_name(name=variable_name)\n            if variable is None:\n                return \"\"\n            else:\n                return variable.value\n        else:\n            for full_match, name, placeholder_type in placeholders:\n                if placeholder_type is PlaceholderType.VARIABLE:\n                    variable_name = name.replace(VARIABLE_PLACEHOLDER_PREFIX, \"\")\n                    variable = await client.read_variable_by_name(name=variable_name)\n                    if variable is None:\n                        template = template.replace(full_match, \"\")\n                    else:\n                        template = template.replace(full_match, variable.value)\n            return template\n    elif isinstance(template, dict):\n        return {\n            key: await resolve_variables(value, client=client)\n            for key, value in template.items()\n        }\n    elif isinstance(template, list):\n        return [await resolve_variables(item, client=client) for item in template]\n    else:\n        return template\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/text/","title":"text","text":"","tags":["Python API","text"]},{"location":"api-ref/prefect/utilities/text/#prefect.utilities.text","title":"prefect.utilities.text","text":"","tags":["Python API","text"]},{"location":"api-ref/prefect/utilities/validation/","title":"validation","text":"","tags":["Python API","validation"]},{"location":"api-ref/prefect/utilities/validation/#prefect.utilities.validation","title":"prefect.utilities.validation","text":"","tags":["Python API","validation"]},{"location":"api-ref/prefect/utilities/validation/#prefect.utilities.validation.validate_schema","title":"validate_schema","text":"

    Validate that the provided schema is a valid json schema.

    Parameters:

    Name Type Description Default schema dict

    The schema to validate.

    required

    Raises:

    Type Description ValueError

    If the provided schema is not a valid json schema.

    Source code in prefect/utilities/validation.py
    def validate_schema(schema: dict):\n    \"\"\"\n    Validate that the provided schema is a valid json schema.\n\n    Args:\n        schema: The schema to validate.\n\n    Raises:\n        ValueError: If the provided schema is not a valid json schema.\n\n    \"\"\"\n    try:\n        if schema is not None:\n            # Most closely matches the schemas generated by pydantic\n            jsonschema.Draft4Validator.check_schema(schema)\n    except jsonschema.SchemaError as exc:\n        raise ValueError(\n            \"The provided schema is not a valid json schema. Schema error:\"\n            f\" {exc.message}\"\n        ) from exc\n
    ","tags":["Python API","validation"]},{"location":"api-ref/prefect/utilities/validation/#prefect.utilities.validation.validate_values_conform_to_schema","title":"validate_values_conform_to_schema","text":"

    Validate that the provided values conform to the provided json schema.

    Parameters:

    Name Type Description Default values dict

    The values to validate.

    required schema dict

    The schema to validate against.

    required ignore_required bool

    Whether to ignore the required fields in the schema. Should be used when a partial set of values is acceptable.

    False

    Raises:

    Type Description ValueError

    If the parameters do not conform to the schema.

    Source code in prefect/utilities/validation.py
    def validate_values_conform_to_schema(\n    values: dict, schema: dict, ignore_required: bool = False\n):\n    \"\"\"\n    Validate that the provided values conform to the provided json schema.\n\n    Args:\n        values: The values to validate.\n        schema: The schema to validate against.\n        ignore_required: Whether to ignore the required fields in the schema. Should be\n            used when a partial set of values is acceptable.\n\n    Raises:\n        ValueError: If the parameters do not conform to the schema.\n\n    \"\"\"\n    if ignore_required:\n        schema = remove_nested_keys([\"required\"], schema)\n\n    try:\n        if schema is not None and values is not None:\n            jsonschema.validate(values, schema)\n    except jsonschema.ValidationError as exc:\n        if exc.json_path == \"$\":\n            error_message = \"Validation failed.\"\n        else:\n            error_message = (\n                f\"Validation failed for field {exc.json_path.replace('$.', '')!r}.\"\n            )\n        error_message += f\" Failure reason: {exc.message}\"\n        raise ValueError(error_message) from exc\n    except jsonschema.SchemaError as exc:\n        raise ValueError(\n            \"The provided schema is not a valid json schema. Schema error:\"\n            f\" {exc.message}\"\n        ) from exc\n
    ","tags":["Python API","validation"]},{"location":"api-ref/prefect/utilities/visualization/","title":"visualization","text":"","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization","title":"prefect.utilities.visualization","text":"

    Utilities for working with Flow.visualize()

    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.TaskVizTracker","title":"TaskVizTracker","text":"Source code in prefect/utilities/visualization.py
    class TaskVizTracker:\n    def __init__(self):\n        self.tasks = []\n        self.dynamic_task_counter = {}\n        self.object_id_to_task = {}\n\n    def add_task(self, task: VizTask):\n        if task.name not in self.dynamic_task_counter:\n            self.dynamic_task_counter[task.name] = 0\n        else:\n            self.dynamic_task_counter[task.name] += 1\n\n        task.name = f\"{task.name}-{self.dynamic_task_counter[task.name]}\"\n        self.tasks.append(task)\n\n    def __enter__(self):\n        TaskVizTrackerState.current = self\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        TaskVizTrackerState.current = None\n\n    def link_viz_return_value_to_viz_task(\n        self, viz_return_value: Any, viz_task: VizTask\n    ) -> None:\n        \"\"\"\n        We cannot track booleans, Ellipsis, None, NotImplemented, or the integers from -5 to 256\n        because they are singletons.\n        \"\"\"\n        from prefect.engine import UNTRACKABLE_TYPES\n\n        if (type(viz_return_value) in UNTRACKABLE_TYPES) or (\n            isinstance(viz_return_value, int) and (-5 <= viz_return_value <= 256)\n        ):\n            return\n        self.object_id_to_task[id(viz_return_value)] = viz_task\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.TaskVizTracker.link_viz_return_value_to_viz_task","title":"link_viz_return_value_to_viz_task","text":"

    We cannot track booleans, Ellipsis, None, NotImplemented, or the integers from -5 to 256 because they are singletons.

    Source code in prefect/utilities/visualization.py
    def link_viz_return_value_to_viz_task(\n    self, viz_return_value: Any, viz_task: VizTask\n) -> None:\n    \"\"\"\n    We cannot track booleans, Ellipsis, None, NotImplemented, or the integers from -5 to 256\n    because they are singletons.\n    \"\"\"\n    from prefect.engine import UNTRACKABLE_TYPES\n\n    if (type(viz_return_value) in UNTRACKABLE_TYPES) or (\n        isinstance(viz_return_value, int) and (-5 <= viz_return_value <= 256)\n    ):\n        return\n    self.object_id_to_task[id(viz_return_value)] = viz_task\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.build_task_dependencies","title":"build_task_dependencies","text":"

    Constructs a Graphviz directed graph object that represents the dependencies between tasks in the given TaskVizTracker.

    • task_run_tracker (TaskVizTracker): An object containing tasks and their dependencies.
    • graphviz.Digraph: A directed graph object depicting the relationships and dependencies between tasks.

    Raises: - GraphvizImportError: If there's an ImportError related to graphviz. - FlowVisualizationError: If there's any other error during the visualization process or if return values of tasks are directly accessed without specifying a viz_return_value.

    Source code in prefect/utilities/visualization.py
    def build_task_dependencies(task_run_tracker: TaskVizTracker):\n    \"\"\"\n    Constructs a Graphviz directed graph object that represents the dependencies\n    between tasks in the given TaskVizTracker.\n\n    Parameters:\n    - task_run_tracker (TaskVizTracker): An object containing tasks and their\n      dependencies.\n\n    Returns:\n    - graphviz.Digraph: A directed graph object depicting the relationships and\n      dependencies between tasks.\n\n    Raises:\n    - GraphvizImportError: If there's an ImportError related to graphviz.\n    - FlowVisualizationError: If there's any other error during the visualization\n      process or if return values of tasks are directly accessed without\n      specifying a `viz_return_value`.\n    \"\"\"\n    try:\n        g = graphviz.Digraph()\n        for task in task_run_tracker.tasks:\n            g.node(task.name)\n            for upstream in task.upstream_tasks:\n                g.edge(upstream.name, task.name)\n        return g\n    except ImportError as exc:\n        raise GraphvizImportError from exc\n    except Exception:\n        raise FlowVisualizationError(\n            \"Something went wrong building the flow's visualization.\"\n            \" If you're interacting with the return value of a task\"\n            \" directly inside of your flow, you must set a set a `viz_return_value`\"\n            \", for example `@task(viz_return_value=[1, 2, 3])`.\"\n        )\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.track_viz_task","title":"track_viz_task","text":"

    Return a result if sync otherwise return a coroutine that returns the result

    Source code in prefect/utilities/visualization.py
    def track_viz_task(\n    is_async: bool,\n    task_name: str,\n    parameters: dict,\n    viz_return_value: Optional[Any] = None,\n):\n    \"\"\"Return a result if sync otherwise return a coroutine that returns the result\"\"\"\n    if is_async:\n        return from_async.wait_for_call_in_loop_thread(\n            partial(_track_viz_task, task_name, parameters, viz_return_value)\n        )\n    else:\n        return _track_viz_task(task_name, parameters, viz_return_value)\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.visualize_task_dependencies","title":"visualize_task_dependencies","text":"

    Renders and displays a Graphviz directed graph representing task dependencies.

    The graph is rendered in PNG format and saved with the name specified by flow_run_name. After rendering, the visualization is opened and displayed.

    Parameters: - graph (graphviz.Digraph): The directed graph object to visualize. - flow_run_name (str): The name to use when saving the rendered graph image.

    Raises: - GraphvizExecutableNotFoundError: If Graphviz isn't found on the system. - FlowVisualizationError: If there's any other error during the visualization process or if return values of tasks are directly accessed without specifying a viz_return_value.

    Source code in prefect/utilities/visualization.py
    def visualize_task_dependencies(graph: graphviz.Digraph, flow_run_name: str):\n    \"\"\"\n    Renders and displays a Graphviz directed graph representing task dependencies.\n\n    The graph is rendered in PNG format and saved with the name specified by\n    flow_run_name. After rendering, the visualization is opened and displayed.\n\n    Parameters:\n    - graph (graphviz.Digraph): The directed graph object to visualize.\n    - flow_run_name (str): The name to use when saving the rendered graph image.\n\n    Raises:\n    - GraphvizExecutableNotFoundError: If Graphviz isn't found on the system.\n    - FlowVisualizationError: If there's any other error during the visualization\n      process or if return values of tasks are directly accessed without\n      specifying a `viz_return_value`.\n    \"\"\"\n    try:\n        graph.render(filename=flow_run_name, view=True, format=\"png\", cleanup=True)\n    except graphviz.backend.ExecutableNotFound as exc:\n        msg = (\n            \"It appears you do not have Graphviz installed, or it is not on your \"\n            \"PATH. Please install Graphviz from http://www.graphviz.org/download/. \"\n            \"Note: Just installing the `graphviz` python package is not \"\n            \"sufficient.\"\n        )\n        raise GraphvizExecutableNotFoundError(msg) from exc\n    except Exception:\n        raise FlowVisualizationError(\n            \"Something went wrong building the flow's visualization.\"\n            \" If you're interacting with the return value of a task\"\n            \" directly inside of your flow, you must set a set a `viz_return_value`\"\n            \", for example `@task(viz_return_value=[1, 2, 3])`.\"\n        )\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/workers/base/","title":"base","text":"","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base","title":"prefect.workers.base","text":"","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseJobConfiguration","title":"BaseJobConfiguration","text":"

    Bases: BaseModel

    Source code in prefect/workers/base.py
    class BaseJobConfiguration(BaseModel):\n    command: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The command to use when starting a flow run. \"\n            \"In most cases, this should be left blank and the command \"\n            \"will be automatically generated by the worker.\"\n        ),\n    )\n    env: Dict[str, Optional[str]] = Field(\n        default_factory=dict,\n        title=\"Environment Variables\",\n        description=\"Environment variables to set when starting a flow run.\",\n    )\n    labels: Dict[str, str] = Field(\n        default_factory=dict,\n        description=(\n            \"Labels applied to infrastructure created by the worker using \"\n            \"this job configuration.\"\n        ),\n    )\n    name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"Name given to infrastructure created by the worker using this \"\n            \"job configuration.\"\n        ),\n    )\n\n    _related_objects: Dict[str, Any] = PrivateAttr(default_factory=dict)\n\n    @property\n    def is_using_a_runner(self):\n        return self.command is not None and \"prefect flow-run execute\" in self.command\n\n    @validator(\"command\")\n    def _coerce_command(cls, v):\n        \"\"\"Make sure that empty strings are treated as None\"\"\"\n        if not v:\n            return None\n        return v\n\n    @staticmethod\n    def _get_base_config_defaults(variables: dict) -> dict:\n        \"\"\"Get default values from base config for all variables that have them.\"\"\"\n        defaults = dict()\n        for variable_name, attrs in variables.items():\n            if \"default\" in attrs:\n                defaults[variable_name] = attrs[\"default\"]\n\n        return defaults\n\n    @classmethod\n    @inject_client\n    async def from_template_and_values(\n        cls, base_job_template: dict, values: dict, client: \"PrefectClient\" = None\n    ):\n        \"\"\"Creates a valid worker configuration object from the provided base\n        configuration and overrides.\n\n        Important: this method expects that the base_job_template was already\n        validated server-side.\n        \"\"\"\n        job_config: Dict[str, Any] = base_job_template[\"job_configuration\"]\n        variables_schema = base_job_template[\"variables\"]\n        variables = cls._get_base_config_defaults(\n            variables_schema.get(\"properties\", {})\n        )\n        variables.update(values)\n\n        populated_configuration = apply_values(template=job_config, values=variables)\n        populated_configuration = await resolve_block_document_references(\n            template=populated_configuration, client=client\n        )\n        populated_configuration = await resolve_variables(\n            template=populated_configuration, client=client\n        )\n        return cls(**populated_configuration)\n\n    @classmethod\n    def json_template(cls) -> dict:\n        \"\"\"Returns a dict with job configuration as keys and the corresponding templates as values\n\n        Defaults to using the job configuration parameter name as the template variable name.\n\n        e.g.\n        {\n            key1: '{{ key1 }}',     # default variable template\n            key2: '{{ template2 }}', # `template2` specifically provide as template\n        }\n        \"\"\"\n        configuration = {}\n        properties = cls.schema()[\"properties\"]\n        for k, v in properties.items():\n            if v.get(\"template\"):\n                template = v[\"template\"]\n            else:\n                template = \"{{ \" + k + \" }}\"\n            configuration[k] = template\n\n        return configuration\n\n    def prepare_for_flow_run(\n        self,\n        flow_run: \"FlowRun\",\n        deployment: Optional[\"DeploymentResponse\"] = None,\n        flow: Optional[\"Flow\"] = None,\n    ):\n        \"\"\"\n        Prepare the job configuration for a flow run.\n\n        This method is called by the worker before starting a flow run. It\n        should be used to set any configuration values that are dependent on\n        the flow run.\n\n        Args:\n            flow_run: The flow run to be executed.\n            deployment: The deployment that the flow run is associated with.\n            flow: The flow that the flow run is associated with.\n        \"\"\"\n\n        self._related_objects = {\n            \"deployment\": deployment,\n            \"flow\": flow,\n            \"flow-run\": flow_run,\n        }\n        if deployment is not None:\n            deployment_labels = self._base_deployment_labels(deployment)\n        else:\n            deployment_labels = {}\n\n        if flow is not None:\n            flow_labels = self._base_flow_labels(flow)\n        else:\n            flow_labels = {}\n\n        env = {\n            **self._base_environment(),\n            **self._base_flow_run_environment(flow_run),\n            **self.env,\n        }\n        self.env = {key: value for key, value in env.items() if value is not None}\n        self.labels = {\n            **self._base_flow_run_labels(flow_run),\n            **deployment_labels,\n            **flow_labels,\n            **self.labels,\n        }\n        self.name = self.name or flow_run.name\n        self.command = self.command or self._base_flow_run_command()\n\n    @staticmethod\n    def _base_flow_run_command() -> str:\n        \"\"\"\n        Generate a command for a flow run job.\n        \"\"\"\n        if experiment_enabled(\"enhanced_cancellation\"):\n            if (\n                PREFECT_EXPERIMENTAL_WARN\n                and PREFECT_EXPERIMENTAL_WARN_ENHANCED_CANCELLATION\n            ):\n                warnings.warn(\n                    EXPERIMENTAL_WARNING.format(\n                        feature=\"Enhanced flow run cancellation\",\n                        group=\"enhanced_cancellation\",\n                        help=\"\",\n                    ),\n                    ExperimentalFeature,\n                    stacklevel=3,\n                )\n            return \"prefect flow-run execute\"\n        return \"python -m prefect.engine\"\n\n    @staticmethod\n    def _base_flow_run_labels(flow_run: \"FlowRun\") -> Dict[str, str]:\n        \"\"\"\n        Generate a dictionary of labels for a flow run job.\n        \"\"\"\n        return {\n            \"prefect.io/flow-run-id\": str(flow_run.id),\n            \"prefect.io/flow-run-name\": flow_run.name,\n            \"prefect.io/version\": prefect.__version__,\n        }\n\n    @classmethod\n    def _base_environment(cls) -> Dict[str, str]:\n        \"\"\"\n        Environment variables that should be passed to all created infrastructure.\n\n        These values should be overridable with the `env` field.\n        \"\"\"\n        return get_current_settings().to_environment_variables(exclude_unset=True)\n\n    @staticmethod\n    def _base_flow_run_environment(flow_run: \"FlowRun\") -> Dict[str, str]:\n        \"\"\"\n        Generate a dictionary of environment variables for a flow run job.\n        \"\"\"\n        return {\"PREFECT__FLOW_RUN_ID\": str(flow_run.id)}\n\n    @staticmethod\n    def _base_deployment_labels(deployment: \"DeploymentResponse\") -> Dict[str, str]:\n        labels = {\n            \"prefect.io/deployment-id\": str(deployment.id),\n            \"prefect.io/deployment-name\": deployment.name,\n        }\n        if deployment.updated is not None:\n            labels[\"prefect.io/deployment-updated\"] = deployment.updated.in_timezone(\n                \"utc\"\n            ).to_iso8601_string()\n        return labels\n\n    @staticmethod\n    def _base_flow_labels(flow: \"Flow\") -> Dict[str, str]:\n        return {\n            \"prefect.io/flow-id\": str(flow.id),\n            \"prefect.io/flow-name\": flow.name,\n        }\n\n    def _related_resources(self) -> List[RelatedResource]:\n        tags = set()\n        related = []\n\n        for kind, obj in self._related_objects.items():\n            if obj is None:\n                continue\n            if hasattr(obj, \"tags\"):\n                tags.update(obj.tags)\n            related.append(object_as_related_resource(kind=kind, role=kind, object=obj))\n\n        return related + tags_as_related_resources(tags)\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseJobConfiguration.from_template_and_values","title":"from_template_and_values async classmethod","text":"

    Creates a valid worker configuration object from the provided base configuration and overrides.

    Important: this method expects that the base_job_template was already validated server-side.

    Source code in prefect/workers/base.py
    @classmethod\n@inject_client\nasync def from_template_and_values(\n    cls, base_job_template: dict, values: dict, client: \"PrefectClient\" = None\n):\n    \"\"\"Creates a valid worker configuration object from the provided base\n    configuration and overrides.\n\n    Important: this method expects that the base_job_template was already\n    validated server-side.\n    \"\"\"\n    job_config: Dict[str, Any] = base_job_template[\"job_configuration\"]\n    variables_schema = base_job_template[\"variables\"]\n    variables = cls._get_base_config_defaults(\n        variables_schema.get(\"properties\", {})\n    )\n    variables.update(values)\n\n    populated_configuration = apply_values(template=job_config, values=variables)\n    populated_configuration = await resolve_block_document_references(\n        template=populated_configuration, client=client\n    )\n    populated_configuration = await resolve_variables(\n        template=populated_configuration, client=client\n    )\n    return cls(**populated_configuration)\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseJobConfiguration.json_template","title":"json_template classmethod","text":"

    Returns a dict with job configuration as keys and the corresponding templates as values

    Defaults to using the job configuration parameter name as the template variable name.

    e.g. { key1: '{{ key1 }}', # default variable template key2: '{{ template2 }}', # template2 specifically provide as template }

    Source code in prefect/workers/base.py
    @classmethod\ndef json_template(cls) -> dict:\n    \"\"\"Returns a dict with job configuration as keys and the corresponding templates as values\n\n    Defaults to using the job configuration parameter name as the template variable name.\n\n    e.g.\n    {\n        key1: '{{ key1 }}',     # default variable template\n        key2: '{{ template2 }}', # `template2` specifically provide as template\n    }\n    \"\"\"\n    configuration = {}\n    properties = cls.schema()[\"properties\"]\n    for k, v in properties.items():\n        if v.get(\"template\"):\n            template = v[\"template\"]\n        else:\n            template = \"{{ \" + k + \" }}\"\n        configuration[k] = template\n\n    return configuration\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseJobConfiguration.prepare_for_flow_run","title":"prepare_for_flow_run","text":"

    Prepare the job configuration for a flow run.

    This method is called by the worker before starting a flow run. It should be used to set any configuration values that are dependent on the flow run.

    Parameters:

    Name Type Description Default flow_run FlowRun

    The flow run to be executed.

    required deployment Optional[DeploymentResponse]

    The deployment that the flow run is associated with.

    None flow Optional[Flow]

    The flow that the flow run is associated with.

    None Source code in prefect/workers/base.py
    def prepare_for_flow_run(\n    self,\n    flow_run: \"FlowRun\",\n    deployment: Optional[\"DeploymentResponse\"] = None,\n    flow: Optional[\"Flow\"] = None,\n):\n    \"\"\"\n    Prepare the job configuration for a flow run.\n\n    This method is called by the worker before starting a flow run. It\n    should be used to set any configuration values that are dependent on\n    the flow run.\n\n    Args:\n        flow_run: The flow run to be executed.\n        deployment: The deployment that the flow run is associated with.\n        flow: The flow that the flow run is associated with.\n    \"\"\"\n\n    self._related_objects = {\n        \"deployment\": deployment,\n        \"flow\": flow,\n        \"flow-run\": flow_run,\n    }\n    if deployment is not None:\n        deployment_labels = self._base_deployment_labels(deployment)\n    else:\n        deployment_labels = {}\n\n    if flow is not None:\n        flow_labels = self._base_flow_labels(flow)\n    else:\n        flow_labels = {}\n\n    env = {\n        **self._base_environment(),\n        **self._base_flow_run_environment(flow_run),\n        **self.env,\n    }\n    self.env = {key: value for key, value in env.items() if value is not None}\n    self.labels = {\n        **self._base_flow_run_labels(flow_run),\n        **deployment_labels,\n        **flow_labels,\n        **self.labels,\n    }\n    self.name = self.name or flow_run.name\n    self.command = self.command or self._base_flow_run_command()\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker","title":"BaseWorker","text":"

    Bases: ABC

    Source code in prefect/workers/base.py
    @register_base_type\nclass BaseWorker(abc.ABC):\n    type: str\n    job_configuration: Type[BaseJobConfiguration] = BaseJobConfiguration\n    job_configuration_variables: Optional[Type[BaseVariables]] = None\n\n    _documentation_url = \"\"\n    _logo_url = \"\"\n    _description = \"\"\n\n    def __init__(\n        self,\n        work_pool_name: str,\n        work_queues: Optional[List[str]] = None,\n        name: Optional[str] = None,\n        prefetch_seconds: Optional[float] = None,\n        create_pool_if_not_found: bool = True,\n        limit: Optional[int] = None,\n        heartbeat_interval_seconds: Optional[int] = None,\n        *,\n        base_job_template: Optional[Dict[str, Any]] = None,\n    ):\n        \"\"\"\n        Base class for all Prefect workers.\n\n        Args:\n            name: The name of the worker. If not provided, a random one\n                will be generated. If provided, it cannot contain '/' or '%'.\n                The name is used to identify the worker in the UI; if two\n                processes have the same name, they will be treated as the same\n                worker.\n            work_pool_name: The name of the work pool to poll.\n            work_queues: A list of work queues to poll. If not provided, all\n                work queue in the work pool will be polled.\n            prefetch_seconds: The number of seconds to prefetch flow runs for.\n            create_pool_if_not_found: Whether to create the work pool\n                if it is not found. Defaults to `True`, but can be set to `False` to\n                ensure that work pools are not created accidentally.\n            limit: The maximum number of flow runs this worker should be running at\n                a given time.\n            base_job_template: If creating the work pool, provide the base job\n                template to use. Logs a warning if the pool already exists.\n        \"\"\"\n        if name and (\"/\" in name or \"%\" in name):\n            raise ValueError(\"Worker name cannot contain '/' or '%'\")\n        self.name = name or f\"{self.__class__.__name__} {uuid4()}\"\n        self._logger = get_logger(f\"worker.{self.__class__.type}.{self.name.lower()}\")\n\n        self.is_setup = False\n        self._create_pool_if_not_found = create_pool_if_not_found\n        self._base_job_template = base_job_template\n        self._work_pool_name = work_pool_name\n        self._work_queues: Set[str] = set(work_queues) if work_queues else set()\n\n        self._prefetch_seconds: float = (\n            prefetch_seconds or PREFECT_WORKER_PREFETCH_SECONDS.value()\n        )\n        self.heartbeat_interval_seconds = (\n            heartbeat_interval_seconds or PREFECT_WORKER_HEARTBEAT_SECONDS.value()\n        )\n\n        self._work_pool: Optional[WorkPool] = None\n        self._runs_task_group: Optional[anyio.abc.TaskGroup] = None\n        self._client: Optional[PrefectClient] = None\n        self._last_polled_time: pendulum.DateTime = pendulum.now(\"utc\")\n        self._limit = limit\n        self._limiter: Optional[anyio.CapacityLimiter] = None\n        self._submitting_flow_run_ids = set()\n        self._cancelling_flow_run_ids = set()\n        self._scheduled_task_scopes = set()\n\n    @classmethod\n    def get_documentation_url(cls) -> str:\n        return cls._documentation_url\n\n    @classmethod\n    def get_logo_url(cls) -> str:\n        return cls._logo_url\n\n    @classmethod\n    def get_description(cls) -> str:\n        return cls._description\n\n    @classmethod\n    def get_default_base_job_template(cls) -> Dict:\n        if cls.job_configuration_variables is None:\n            schema = cls.job_configuration.schema()\n            # remove \"template\" key from all dicts in schema['properties'] because it is not a\n            # relevant field\n            for key, value in schema[\"properties\"].items():\n                if isinstance(value, dict):\n                    schema[\"properties\"][key].pop(\"template\", None)\n            variables_schema = schema\n        else:\n            variables_schema = cls.job_configuration_variables.schema()\n        variables_schema.pop(\"title\", None)\n        return {\n            \"job_configuration\": cls.job_configuration.json_template(),\n            \"variables\": variables_schema,\n        }\n\n    @staticmethod\n    def get_worker_class_from_type(type: str) -> Optional[Type[\"BaseWorker\"]]:\n        \"\"\"\n        Returns the worker class for a given worker type. If the worker type\n        is not recognized, returns None.\n        \"\"\"\n        load_prefect_collections()\n        worker_registry = get_registry_for_type(BaseWorker)\n        if worker_registry is not None:\n            return worker_registry.get(type)\n\n    @staticmethod\n    def get_all_available_worker_types() -> List[str]:\n        \"\"\"\n        Returns all worker types available in the local registry.\n        \"\"\"\n        load_prefect_collections()\n        worker_registry = get_registry_for_type(BaseWorker)\n        if worker_registry is not None:\n            return list(worker_registry.keys())\n        return []\n\n    def get_name_slug(self):\n        return slugify(self.name)\n\n    def get_flow_run_logger(self, flow_run: \"FlowRun\") -> PrefectLogAdapter:\n        return flow_run_logger(flow_run=flow_run).getChild(\n            \"worker\",\n            extra={\n                \"worker_name\": self.name,\n                \"work_pool_name\": (\n                    self._work_pool_name if self._work_pool else \"<unknown>\"\n                ),\n                \"work_pool_id\": str(getattr(self._work_pool, \"id\", \"unknown\")),\n            },\n        )\n\n    @abc.abstractmethod\n    async def run(\n        self,\n        flow_run: \"FlowRun\",\n        configuration: BaseJobConfiguration,\n        task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> BaseWorkerResult:\n        \"\"\"\n        Runs a given flow run on the current worker.\n        \"\"\"\n        raise NotImplementedError(\n            \"Workers must implement a method for running submitted flow runs\"\n        )\n\n    async def kill_infrastructure(\n        self,\n        infrastructure_pid: str,\n        configuration: BaseJobConfiguration,\n        grace_seconds: int = 30,\n    ):\n        \"\"\"\n        Method for killing infrastructure created by a worker. Should be implemented by\n        individual workers if they support killing infrastructure.\n        \"\"\"\n        raise NotImplementedError(\n            \"This worker does not support killing infrastructure.\"\n        )\n\n    @classmethod\n    def __dispatch_key__(cls):\n        if cls.__name__ == \"BaseWorker\":\n            return None  # The base class is abstract\n        return cls.type\n\n    async def setup(self):\n        \"\"\"Prepares the worker to run.\"\"\"\n        self._logger.debug(\"Setting up worker...\")\n        self._runs_task_group = anyio.create_task_group()\n        self._limiter = (\n            anyio.CapacityLimiter(self._limit) if self._limit is not None else None\n        )\n        self._client = get_client()\n        await self._client.__aenter__()\n        await self._runs_task_group.__aenter__()\n\n        self.is_setup = True\n\n    async def teardown(self, *exc_info):\n        \"\"\"Cleans up resources after the worker is stopped.\"\"\"\n        self._logger.debug(\"Tearing down worker...\")\n        self.is_setup = False\n        for scope in self._scheduled_task_scopes:\n            scope.cancel()\n        if self._runs_task_group:\n            await self._runs_task_group.__aexit__(*exc_info)\n        if self._client:\n            await self._client.__aexit__(*exc_info)\n        self._runs_task_group = None\n        self._client = None\n\n    def is_worker_still_polling(self, query_interval_seconds: int) -> bool:\n        \"\"\"\n        This method is invoked by a webserver healthcheck handler\n        and returns a boolean indicating if the worker has recorded a\n        scheduled flow run poll within a variable amount of time.\n\n        The `query_interval_seconds` is the same value that is used by\n        the loop services - we will evaluate if the _last_polled_time\n        was within that interval x 30 (so 10s -> 5m)\n\n        The instance property `self._last_polled_time`\n        is currently set/updated in `get_and_submit_flow_runs()`\n        \"\"\"\n        threshold_seconds = query_interval_seconds * 30\n\n        seconds_since_last_poll = (\n            pendulum.now(\"utc\") - self._last_polled_time\n        ).in_seconds()\n\n        is_still_polling = seconds_since_last_poll <= threshold_seconds\n\n        if not is_still_polling:\n            self._logger.error(\n                f\"Worker has not polled in the last {seconds_since_last_poll} seconds \"\n                \"and should be restarted\"\n            )\n\n        return is_still_polling\n\n    async def get_and_submit_flow_runs(self):\n        runs_response = await self._get_scheduled_flow_runs()\n\n        self._last_polled_time = pendulum.now(\"utc\")\n\n        return await self._submit_scheduled_flow_runs(flow_run_response=runs_response)\n\n    async def check_for_cancelled_flow_runs(self):\n        if not self.is_setup:\n            raise RuntimeError(\n                \"Worker is not set up. Please make sure you are running this worker \"\n                \"as an async context manager.\"\n            )\n\n        self._logger.debug(\"Checking for cancelled flow runs...\")\n\n        work_queue_filter = (\n            WorkQueueFilter(name=WorkQueueFilterName(any_=list(self._work_queues)))\n            if self._work_queues\n            else None\n        )\n\n        named_cancelling_flow_runs = await self._client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLED]),\n                    name=FlowRunFilterStateName(any_=[\"Cancelling\"]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(not_any_=list(self._cancelling_flow_run_ids)),\n            ),\n            work_pool_filter=WorkPoolFilter(\n                name=WorkPoolFilterName(any_=[self._work_pool_name])\n            ),\n            work_queue_filter=work_queue_filter,\n        )\n\n        typed_cancelling_flow_runs = await self._client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLING]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(not_any_=list(self._cancelling_flow_run_ids)),\n            ),\n            work_pool_filter=WorkPoolFilter(\n                name=WorkPoolFilterName(any_=[self._work_pool_name])\n            ),\n            work_queue_filter=work_queue_filter,\n        )\n\n        cancelling_flow_runs = named_cancelling_flow_runs + typed_cancelling_flow_runs\n\n        if cancelling_flow_runs:\n            self._logger.info(\n                f\"Found {len(cancelling_flow_runs)} flow runs awaiting cancellation.\"\n            )\n\n        for flow_run in cancelling_flow_runs:\n            self._cancelling_flow_run_ids.add(flow_run.id)\n            self._runs_task_group.start_soon(self.cancel_run, flow_run)\n\n        return cancelling_flow_runs\n\n    async def cancel_run(self, flow_run: \"FlowRun\"):\n        run_logger = self.get_flow_run_logger(flow_run)\n\n        try:\n            configuration = await self._get_configuration(flow_run)\n            if configuration.is_using_a_runner:\n                self._logger.info(\n                    f\"Skipping cancellation because flow run {str(flow_run.id)!r} is\"\n                    \" using enhanced cancellation. A dedicated runner will handle\"\n                    \" cancellation.\"\n                )\n                return\n        except ObjectNotFound:\n            self._logger.warning(\n                f\"Flow run {flow_run.id!r} cannot be cancelled by this worker:\"\n                f\" associated deployment {flow_run.deployment_id!r} does not exist.\"\n            )\n\n        if not flow_run.infrastructure_pid:\n            run_logger.error(\n                f\"Flow run '{flow_run.id}' does not have an infrastructure pid\"\n                \" attached. Cancellation cannot be guaranteed.\"\n            )\n            await self._mark_flow_run_as_cancelled(\n                flow_run,\n                state_updates={\n                    \"message\": (\n                        \"This flow run is missing infrastructure tracking information\"\n                        \" and cancellation cannot be guaranteed.\"\n                    )\n                },\n            )\n            return\n\n        try:\n            await self.kill_infrastructure(\n                infrastructure_pid=flow_run.infrastructure_pid,\n                configuration=configuration,\n            )\n        except NotImplementedError:\n            self._logger.error(\n                f\"Worker type {self.type!r} does not support killing created \"\n                \"infrastructure. Cancellation cannot be guaranteed.\"\n            )\n        except InfrastructureNotFound as exc:\n            self._logger.warning(f\"{exc} Marking flow run as cancelled.\")\n            await self._mark_flow_run_as_cancelled(flow_run)\n        except InfrastructureNotAvailable as exc:\n            self._logger.warning(f\"{exc} Flow run cannot be cancelled by this worker.\")\n        except Exception:\n            run_logger.exception(\n                \"Encountered exception while killing infrastructure for flow run \"\n                f\"'{flow_run.id}'. Flow run may not be cancelled.\"\n            )\n            # We will try again on generic exceptions\n            self._cancelling_flow_run_ids.remove(flow_run.id)\n            return\n        else:\n            self._emit_flow_run_cancelled_event(\n                flow_run=flow_run, configuration=configuration\n            )\n            await self._mark_flow_run_as_cancelled(flow_run)\n            run_logger.info(f\"Cancelled flow run '{flow_run.id}'!\")\n\n    async def _update_local_work_pool_info(self):\n        try:\n            work_pool = await self._client.read_work_pool(\n                work_pool_name=self._work_pool_name\n            )\n        except ObjectNotFound:\n            if self._create_pool_if_not_found:\n                wp = WorkPoolCreate(\n                    name=self._work_pool_name,\n                    type=self.type,\n                )\n                if self._base_job_template is not None:\n                    wp.base_job_template = self._base_job_template\n\n                work_pool = await self._client.create_work_pool(work_pool=wp)\n                self._logger.info(f\"Work pool {self._work_pool_name!r} created.\")\n            else:\n                self._logger.warning(f\"Work pool {self._work_pool_name!r} not found!\")\n                if self._base_job_template is not None:\n                    self._logger.warning(\n                        \"Ignoring supplied base job template because the work pool\"\n                        \" already exists\"\n                    )\n                return\n\n        # if the remote config type changes (or if it's being loaded for the\n        # first time), check if it matches the local type and warn if not\n        if getattr(self._work_pool, \"type\", 0) != work_pool.type:\n            if work_pool.type != self.__class__.type:\n                self._logger.warning(\n                    \"Worker type mismatch! This worker process expects type \"\n                    f\"{self.type!r} but received {work_pool.type!r}\"\n                    \" from the server. Unexpected behavior may occur.\"\n                )\n\n        # once the work pool is loaded, verify that it has a `base_job_template` and\n        # set it if not\n        if not work_pool.base_job_template:\n            job_template = self.__class__.get_default_base_job_template()\n            await self._set_work_pool_template(work_pool, job_template)\n            work_pool.base_job_template = job_template\n\n        self._work_pool = work_pool\n\n    async def _send_worker_heartbeat(self):\n        if self._work_pool:\n            await self._client.send_worker_heartbeat(\n                work_pool_name=self._work_pool_name,\n                worker_name=self.name,\n                heartbeat_interval_seconds=self.heartbeat_interval_seconds,\n            )\n\n    async def sync_with_backend(self):\n        \"\"\"\n        Updates the worker's local information about it's current work pool and\n        queues. Sends a worker heartbeat to the API.\n        \"\"\"\n        await self._update_local_work_pool_info()\n\n        await self._send_worker_heartbeat()\n\n        self._logger.debug(\"Worker synchronized with the Prefect API server.\")\n\n    async def _get_scheduled_flow_runs(\n        self,\n    ) -> List[\"WorkerFlowRunResponse\"]:\n        \"\"\"\n        Retrieve scheduled flow runs from the work pool's queues.\n        \"\"\"\n        scheduled_before = pendulum.now(\"utc\").add(seconds=int(self._prefetch_seconds))\n        self._logger.debug(\n            f\"Querying for flow runs scheduled before {scheduled_before}\"\n        )\n        try:\n            scheduled_flow_runs = (\n                await self._client.get_scheduled_flow_runs_for_work_pool(\n                    work_pool_name=self._work_pool_name,\n                    scheduled_before=scheduled_before,\n                    work_queue_names=list(self._work_queues),\n                )\n            )\n            self._logger.debug(\n                f\"Discovered {len(scheduled_flow_runs)} scheduled_flow_runs\"\n            )\n            return scheduled_flow_runs\n        except ObjectNotFound:\n            # the pool doesn't exist; it will be created on the next\n            # heartbeat (or an appropriate warning will be logged)\n            return []\n\n    async def _submit_scheduled_flow_runs(\n        self, flow_run_response: List[\"WorkerFlowRunResponse\"]\n    ) -> List[\"FlowRun\"]:\n        \"\"\"\n        Takes a list of WorkerFlowRunResponses and submits the referenced flow runs\n        for execution by the worker.\n        \"\"\"\n        submittable_flow_runs = [entry.flow_run for entry in flow_run_response]\n        submittable_flow_runs.sort(key=lambda run: run.next_scheduled_start_time)\n        for flow_run in submittable_flow_runs:\n            if flow_run.id in self._submitting_flow_run_ids:\n                continue\n\n            try:\n                if self._limiter:\n                    self._limiter.acquire_on_behalf_of_nowait(flow_run.id)\n            except anyio.WouldBlock:\n                self._logger.info(\n                    f\"Flow run limit reached; {self._limiter.borrowed_tokens} flow runs\"\n                    \" in progress.\"\n                )\n                break\n            else:\n                run_logger = self.get_flow_run_logger(flow_run)\n                run_logger.info(\n                    f\"Worker '{self.name}' submitting flow run '{flow_run.id}'\"\n                )\n                self._submitting_flow_run_ids.add(flow_run.id)\n                self._runs_task_group.start_soon(\n                    self._submit_run,\n                    flow_run,\n                )\n\n        return list(\n            filter(\n                lambda run: run.id in self._submitting_flow_run_ids,\n                submittable_flow_runs,\n            )\n        )\n\n    async def _check_flow_run(self, flow_run: \"FlowRun\") -> None:\n        \"\"\"\n        Performs a check on a submitted flow run to warn the user if the flow run\n        was created from a deployment with a storage block.\n        \"\"\"\n        if flow_run.deployment_id:\n            deployment = await self._client.read_deployment(flow_run.deployment_id)\n            if deployment.storage_document_id:\n                raise ValueError(\n                    f\"Flow run {flow_run.id!r} was created from deployment\"\n                    f\" {deployment.name!r} which is configured with a storage block.\"\n                    \" Please use an\"\n                    \" agent to execute this flow run.\"\n                )\n\n    async def _submit_run(self, flow_run: \"FlowRun\") -> None:\n        \"\"\"\n        Submits a given flow run for execution by the worker.\n        \"\"\"\n        run_logger = self.get_flow_run_logger(flow_run)\n\n        try:\n            await self._check_flow_run(flow_run)\n        except (ValueError, ObjectNotFound):\n            self._logger.exception(\n                (\n                    \"Flow run %s did not pass checks and will not be submitted for\"\n                    \" execution\"\n                ),\n                flow_run.id,\n            )\n            self._submitting_flow_run_ids.remove(flow_run.id)\n            return\n\n        ready_to_submit = await self._propose_pending_state(flow_run)\n\n        if ready_to_submit:\n            readiness_result = await self._runs_task_group.start(\n                self._submit_run_and_capture_errors, flow_run\n            )\n\n            if readiness_result and not isinstance(readiness_result, Exception):\n                try:\n                    await self._client.update_flow_run(\n                        flow_run_id=flow_run.id,\n                        infrastructure_pid=str(readiness_result),\n                    )\n                except Exception:\n                    run_logger.exception(\n                        \"An error occurred while setting the `infrastructure_pid` on \"\n                        f\"flow run {flow_run.id!r}. The flow run will \"\n                        \"not be cancellable.\"\n                    )\n\n            run_logger.info(f\"Completed submission of flow run '{flow_run.id}'\")\n\n        else:\n            # If the run is not ready to submit, release the concurrency slot\n            if self._limiter:\n                self._limiter.release_on_behalf_of(flow_run.id)\n\n        self._submitting_flow_run_ids.remove(flow_run.id)\n\n    async def _submit_run_and_capture_errors(\n        self, flow_run: \"FlowRun\", task_status: anyio.abc.TaskStatus = None\n    ) -> Union[BaseWorkerResult, Exception]:\n        run_logger = self.get_flow_run_logger(flow_run)\n\n        try:\n            configuration = await self._get_configuration(flow_run)\n            submitted_event = self._emit_flow_run_submitted_event(configuration)\n            result = await self.run(\n                flow_run=flow_run,\n                task_status=task_status,\n                configuration=configuration,\n            )\n        except Exception as exc:\n            if not task_status._future.done():\n                # This flow run was being submitted and did not start successfully\n                run_logger.exception(\n                    f\"Failed to submit flow run '{flow_run.id}' to infrastructure.\"\n                )\n                # Mark the task as started to prevent agent crash\n                task_status.started(exc)\n                await self._propose_crashed_state(\n                    flow_run, \"Flow run could not be submitted to infrastructure\"\n                )\n            else:\n                run_logger.exception(\n                    f\"An error occurred while monitoring flow run '{flow_run.id}'. \"\n                    \"The flow run will not be marked as failed, but an issue may have \"\n                    \"occurred.\"\n                )\n            return exc\n        finally:\n            if self._limiter:\n                self._limiter.release_on_behalf_of(flow_run.id)\n\n        if not task_status._future.done():\n            run_logger.error(\n                f\"Infrastructure returned without reporting flow run '{flow_run.id}' \"\n                \"as started or raising an error. This behavior is not expected and \"\n                \"generally indicates improper implementation of infrastructure. The \"\n                \"flow run will not be marked as failed, but an issue may have occurred.\"\n            )\n            # Mark the task as started to prevent agent crash\n            task_status.started()\n\n        if result.status_code != 0:\n            await self._propose_crashed_state(\n                flow_run,\n                (\n                    \"Flow run infrastructure exited with non-zero status code\"\n                    f\" {result.status_code}.\"\n                ),\n            )\n\n        self._emit_flow_run_executed_event(result, configuration, submitted_event)\n\n        return result\n\n    def get_status(self):\n        \"\"\"\n        Retrieves the status of the current worker including its name, current worker\n        pool, the work pool queues it is polling, and its local settings.\n        \"\"\"\n        return {\n            \"name\": self.name,\n            \"work_pool\": (\n                self._work_pool.dict(json_compatible=True)\n                if self._work_pool is not None\n                else None\n            ),\n            \"settings\": {\n                \"prefetch_seconds\": self._prefetch_seconds,\n            },\n        }\n\n    async def _get_configuration(\n        self,\n        flow_run: \"FlowRun\",\n    ) -> BaseJobConfiguration:\n        deployment = await self._client.read_deployment(flow_run.deployment_id)\n        flow = await self._client.read_flow(flow_run.flow_id)\n        configuration = await self.job_configuration.from_template_and_values(\n            base_job_template=self._work_pool.base_job_template,\n            values=deployment.infra_overrides or {},\n            client=self._client,\n        )\n        configuration.prepare_for_flow_run(\n            flow_run=flow_run, deployment=deployment, flow=flow\n        )\n        return configuration\n\n    async def _propose_pending_state(self, flow_run: \"FlowRun\") -> bool:\n        run_logger = self.get_flow_run_logger(flow_run)\n        state = flow_run.state\n        try:\n            state = await propose_state(\n                self._client, Pending(), flow_run_id=flow_run.id\n            )\n        except Abort as exc:\n            run_logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}'. \"\n                    f\"Server sent an abort signal: {exc}\"\n                ),\n            )\n            return False\n        except Exception:\n            run_logger.exception(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n            )\n            return False\n\n        if not state.is_pending():\n            run_logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}': \"\n                    f\"Server returned a non-pending state {state.type.value!r}\"\n                ),\n            )\n            return False\n\n        return True\n\n    async def _propose_failed_state(self, flow_run: \"FlowRun\", exc: Exception) -> None:\n        run_logger = self.get_flow_run_logger(flow_run)\n        try:\n            await propose_state(\n                self._client,\n                await exception_to_failed_state(message=\"Submission failed.\", exc=exc),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # We've already failed, no need to note the abort but we don't want it to\n            # raise in the agent process\n            pass\n        except Exception:\n            run_logger.error(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n                exc_info=True,\n            )\n\n    async def _propose_crashed_state(self, flow_run: \"FlowRun\", message: str) -> None:\n        run_logger = self.get_flow_run_logger(flow_run)\n        try:\n            state = await propose_state(\n                self._client,\n                Crashed(message=message),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # Flow run already marked as failed\n            pass\n        except Exception:\n            run_logger.exception(f\"Failed to update state of flow run '{flow_run.id}'\")\n        else:\n            if state.is_crashed():\n                run_logger.info(\n                    f\"Reported flow run '{flow_run.id}' as crashed: {message}\"\n                )\n\n    async def _mark_flow_run_as_cancelled(\n        self, flow_run: \"FlowRun\", state_updates: Optional[dict] = None\n    ) -> None:\n        state_updates = state_updates or {}\n        state_updates.setdefault(\"name\", \"Cancelled\")\n        state_updates.setdefault(\"type\", StateType.CANCELLED)\n        state = flow_run.state.copy(update=state_updates)\n\n        await self._client.set_flow_run_state(flow_run.id, state, force=True)\n\n        # Do not remove the flow run from the cancelling set immediately because\n        # the API caches responses for the `read_flow_runs` and we do not want to\n        # duplicate cancellations.\n        await self._schedule_task(\n            60 * 10, self._cancelling_flow_run_ids.remove, flow_run.id\n        )\n\n    async def _set_work_pool_template(self, work_pool, job_template):\n        \"\"\"Updates the `base_job_template` for the worker's work pool server side.\"\"\"\n        await self._client.update_work_pool(\n            work_pool_name=work_pool.name,\n            work_pool=WorkPoolUpdate(\n                base_job_template=job_template,\n            ),\n        )\n\n    async def _schedule_task(self, __in_seconds: int, fn, *args, **kwargs):\n        \"\"\"\n        Schedule a background task to start after some time.\n\n        These tasks will be run immediately when the worker exits instead of waiting.\n\n        The function may be async or sync. Async functions will be awaited.\n        \"\"\"\n\n        async def wrapper(task_status):\n            # If we are shutting down, do not sleep; otherwise sleep until the scheduled\n            # time or shutdown\n            if self.is_setup:\n                with anyio.CancelScope() as scope:\n                    self._scheduled_task_scopes.add(scope)\n                    task_status.started()\n                    await anyio.sleep(__in_seconds)\n\n                self._scheduled_task_scopes.remove(scope)\n            else:\n                task_status.started()\n\n            result = fn(*args, **kwargs)\n            if inspect.iscoroutine(result):\n                await result\n\n        await self._runs_task_group.start(wrapper)\n\n    async def __aenter__(self):\n        self._logger.debug(\"Entering worker context...\")\n        await self.setup()\n        return self\n\n    async def __aexit__(self, *exc_info):\n        self._logger.debug(\"Exiting worker context...\")\n        await self.teardown(*exc_info)\n\n    def __repr__(self):\n        return f\"Worker(pool={self._work_pool_name!r}, name={self.name!r})\"\n\n    def _event_resource(self):\n        return {\n            \"prefect.resource.id\": f\"prefect.worker.{self.type}.{self.get_name_slug()}\",\n            \"prefect.resource.name\": self.name,\n            \"prefect.version\": prefect.__version__,\n            \"prefect.worker-type\": self.type,\n        }\n\n    def _event_related_resources(\n        self,\n        configuration: Optional[BaseJobConfiguration] = None,\n        include_self: bool = False,\n    ) -> List[RelatedResource]:\n        related = []\n        if configuration:\n            related += configuration._related_resources()\n\n        if self._work_pool:\n            related.append(\n                object_as_related_resource(\n                    kind=\"work-pool\", role=\"work-pool\", object=self._work_pool\n                )\n            )\n\n        if include_self:\n            worker_resource = self._event_resource()\n            worker_resource[\"prefect.resource.role\"] = \"worker\"\n            related.append(RelatedResource(__root__=worker_resource))\n\n        return related\n\n    def _emit_flow_run_submitted_event(\n        self, configuration: BaseJobConfiguration\n    ) -> Event:\n        return emit_event(\n            event=\"prefect.worker.submitted-flow-run\",\n            resource=self._event_resource(),\n            related=self._event_related_resources(configuration=configuration),\n        )\n\n    def _emit_flow_run_executed_event(\n        self,\n        result: BaseWorkerResult,\n        configuration: BaseJobConfiguration,\n        submitted_event: Event,\n    ):\n        related = self._event_related_resources(configuration=configuration)\n\n        for resource in related:\n            if resource.role == \"flow-run\":\n                resource[\"prefect.infrastructure.identifier\"] = str(result.identifier)\n                resource[\"prefect.infrastructure.status-code\"] = str(result.status_code)\n\n        emit_event(\n            event=\"prefect.worker.executed-flow-run\",\n            resource=self._event_resource(),\n            related=related,\n            follows=submitted_event,\n        )\n\n    async def _emit_worker_started_event(self) -> Event:\n        return emit_event(\n            \"prefect.worker.started\",\n            resource=self._event_resource(),\n            related=self._event_related_resources(),\n        )\n\n    async def _emit_worker_stopped_event(self, started_event: Event):\n        emit_event(\n            \"prefect.worker.stopped\",\n            resource=self._event_resource(),\n            related=self._event_related_resources(),\n            follows=started_event,\n        )\n\n    def _emit_flow_run_cancelled_event(\n        self, flow_run: \"FlowRun\", configuration: BaseJobConfiguration\n    ):\n        related = self._event_related_resources(configuration=configuration)\n\n        for resource in related:\n            if resource.role == \"flow-run\":\n                resource[\"prefect.infrastructure.identifier\"] = str(\n                    flow_run.infrastructure_pid\n                )\n\n        emit_event(\n            event=\"prefect.worker.cancelled-flow-run\",\n            resource=self._event_resource(),\n            related=related,\n        )\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.get_all_available_worker_types","title":"get_all_available_worker_types staticmethod","text":"

    Returns all worker types available in the local registry.

    Source code in prefect/workers/base.py
    @staticmethod\ndef get_all_available_worker_types() -> List[str]:\n    \"\"\"\n    Returns all worker types available in the local registry.\n    \"\"\"\n    load_prefect_collections()\n    worker_registry = get_registry_for_type(BaseWorker)\n    if worker_registry is not None:\n        return list(worker_registry.keys())\n    return []\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.get_status","title":"get_status","text":"

    Retrieves the status of the current worker including its name, current worker pool, the work pool queues it is polling, and its local settings.

    Source code in prefect/workers/base.py
    def get_status(self):\n    \"\"\"\n    Retrieves the status of the current worker including its name, current worker\n    pool, the work pool queues it is polling, and its local settings.\n    \"\"\"\n    return {\n        \"name\": self.name,\n        \"work_pool\": (\n            self._work_pool.dict(json_compatible=True)\n            if self._work_pool is not None\n            else None\n        ),\n        \"settings\": {\n            \"prefetch_seconds\": self._prefetch_seconds,\n        },\n    }\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.get_worker_class_from_type","title":"get_worker_class_from_type staticmethod","text":"

    Returns the worker class for a given worker type. If the worker type is not recognized, returns None.

    Source code in prefect/workers/base.py
    @staticmethod\ndef get_worker_class_from_type(type: str) -> Optional[Type[\"BaseWorker\"]]:\n    \"\"\"\n    Returns the worker class for a given worker type. If the worker type\n    is not recognized, returns None.\n    \"\"\"\n    load_prefect_collections()\n    worker_registry = get_registry_for_type(BaseWorker)\n    if worker_registry is not None:\n        return worker_registry.get(type)\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.is_worker_still_polling","title":"is_worker_still_polling","text":"

    This method is invoked by a webserver healthcheck handler and returns a boolean indicating if the worker has recorded a scheduled flow run poll within a variable amount of time.

    The query_interval_seconds is the same value that is used by the loop services - we will evaluate if the _last_polled_time was within that interval x 30 (so 10s -> 5m)

    The instance property self._last_polled_time is currently set/updated in get_and_submit_flow_runs()

    Source code in prefect/workers/base.py
    def is_worker_still_polling(self, query_interval_seconds: int) -> bool:\n    \"\"\"\n    This method is invoked by a webserver healthcheck handler\n    and returns a boolean indicating if the worker has recorded a\n    scheduled flow run poll within a variable amount of time.\n\n    The `query_interval_seconds` is the same value that is used by\n    the loop services - we will evaluate if the _last_polled_time\n    was within that interval x 30 (so 10s -> 5m)\n\n    The instance property `self._last_polled_time`\n    is currently set/updated in `get_and_submit_flow_runs()`\n    \"\"\"\n    threshold_seconds = query_interval_seconds * 30\n\n    seconds_since_last_poll = (\n        pendulum.now(\"utc\") - self._last_polled_time\n    ).in_seconds()\n\n    is_still_polling = seconds_since_last_poll <= threshold_seconds\n\n    if not is_still_polling:\n        self._logger.error(\n            f\"Worker has not polled in the last {seconds_since_last_poll} seconds \"\n            \"and should be restarted\"\n        )\n\n    return is_still_polling\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.kill_infrastructure","title":"kill_infrastructure async","text":"

    Method for killing infrastructure created by a worker. Should be implemented by individual workers if they support killing infrastructure.

    Source code in prefect/workers/base.py
    async def kill_infrastructure(\n    self,\n    infrastructure_pid: str,\n    configuration: BaseJobConfiguration,\n    grace_seconds: int = 30,\n):\n    \"\"\"\n    Method for killing infrastructure created by a worker. Should be implemented by\n    individual workers if they support killing infrastructure.\n    \"\"\"\n    raise NotImplementedError(\n        \"This worker does not support killing infrastructure.\"\n    )\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.run","title":"run abstractmethod async","text":"

    Runs a given flow run on the current worker.

    Source code in prefect/workers/base.py
    @abc.abstractmethod\nasync def run(\n    self,\n    flow_run: \"FlowRun\",\n    configuration: BaseJobConfiguration,\n    task_status: Optional[anyio.abc.TaskStatus] = None,\n) -> BaseWorkerResult:\n    \"\"\"\n    Runs a given flow run on the current worker.\n    \"\"\"\n    raise NotImplementedError(\n        \"Workers must implement a method for running submitted flow runs\"\n    )\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.setup","title":"setup async","text":"

    Prepares the worker to run.

    Source code in prefect/workers/base.py
    async def setup(self):\n    \"\"\"Prepares the worker to run.\"\"\"\n    self._logger.debug(\"Setting up worker...\")\n    self._runs_task_group = anyio.create_task_group()\n    self._limiter = (\n        anyio.CapacityLimiter(self._limit) if self._limit is not None else None\n    )\n    self._client = get_client()\n    await self._client.__aenter__()\n    await self._runs_task_group.__aenter__()\n\n    self.is_setup = True\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.sync_with_backend","title":"sync_with_backend async","text":"

    Updates the worker's local information about it's current work pool and queues. Sends a worker heartbeat to the API.

    Source code in prefect/workers/base.py
    async def sync_with_backend(self):\n    \"\"\"\n    Updates the worker's local information about it's current work pool and\n    queues. Sends a worker heartbeat to the API.\n    \"\"\"\n    await self._update_local_work_pool_info()\n\n    await self._send_worker_heartbeat()\n\n    self._logger.debug(\"Worker synchronized with the Prefect API server.\")\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.teardown","title":"teardown async","text":"

    Cleans up resources after the worker is stopped.

    Source code in prefect/workers/base.py
    async def teardown(self, *exc_info):\n    \"\"\"Cleans up resources after the worker is stopped.\"\"\"\n    self._logger.debug(\"Tearing down worker...\")\n    self.is_setup = False\n    for scope in self._scheduled_task_scopes:\n        scope.cancel()\n    if self._runs_task_group:\n        await self._runs_task_group.__aexit__(*exc_info)\n    if self._client:\n        await self._client.__aexit__(*exc_info)\n    self._runs_task_group = None\n    self._client = None\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/block/","title":"block","text":"","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block","title":"prefect.workers.block","text":"","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block.BlockWorkerJobConfiguration","title":"BlockWorkerJobConfiguration","text":"

    Bases: BaseModel

    Source code in prefect/workers/block.py
    class BlockWorkerJobConfiguration(BaseModel):\n    block: Block = Field(\n        default=..., description=\"The infrastructure block to use for job creation.\"\n    )\n\n    @validator(\"block\")\n    def _validate_block_is_infrastructure(cls, v):\n        print(\"v: \", v)\n        if not isinstance(v, Infrastructure):\n            raise TypeError(\"Provided block is not a valid infrastructure block.\")\n\n        return v\n\n    _related_objects: Dict[str, Any] = PrivateAttr(default_factory=dict)\n\n    @property\n    def is_using_a_runner(self):\n        return (\n            self.block.command is not None\n            and \"prefect flow-run execute\" in shlex.join(self.block.command)\n        )\n\n    @staticmethod\n    def _get_base_config_defaults(variables: dict) -> dict:\n        \"\"\"Get default values from base config for all variables that have them.\"\"\"\n        defaults = dict()\n        for variable_name, attrs in variables.items():\n            if \"default\" in attrs:\n                defaults[variable_name] = attrs[\"default\"]\n\n        return defaults\n\n    @classmethod\n    @inject_client\n    async def from_template_and_values(\n        cls, base_job_template: dict, values: dict, client: \"PrefectClient\" = None\n    ):\n        \"\"\"Creates a valid worker configuration object from the provided base\n        configuration and overrides.\n\n        Important: this method expects that the base_job_template was already\n        validated server-side.\n        \"\"\"\n        job_config: Dict[str, Any] = base_job_template[\"job_configuration\"]\n        variables_schema = base_job_template[\"variables\"]\n        variables = cls._get_base_config_defaults(\n            variables_schema.get(\"properties\", {})\n        )\n        variables.update(values)\n\n        populated_configuration = apply_values(template=job_config, values=variables)\n\n        block_document_id = get_from_dict(\n            populated_configuration, \"block.$ref.block_document_id\"\n        )\n        if not block_document_id:\n            raise ValueError(\n                \"Base job template is invalid for this worker type because it does not\"\n                \" contain a block_document_id after variable resolution.\"\n            )\n\n        block_document = await client.read_block_document(\n            block_document_id=block_document_id\n        )\n        infrastructure_block = Block._from_block_document(block_document)\n\n        populated_configuration[\"block\"] = infrastructure_block\n\n        return cls(**populated_configuration)\n\n    @classmethod\n    def json_template(cls) -> dict:\n        \"\"\"Returns a dict with job configuration as keys and the corresponding templates as values\n\n        Defaults to using the job configuration parameter name as the template variable name.\n\n        e.g.\n        {\n            key1: '{{ key1 }}',     # default variable template\n            key2: '{{ template2 }}', # `template2` specifically provide as template\n        }\n        \"\"\"\n        configuration = {}\n        properties = cls.schema()[\"properties\"]\n        for k, v in properties.items():\n            if v.get(\"template\"):\n                template = v[\"template\"]\n            else:\n                template = \"{{ \" + k + \" }}\"\n            configuration[k] = template\n\n        return configuration\n\n    def _related_resources(self) -> List[RelatedResource]:\n        tags = set()\n        related = []\n\n        for kind, obj in self._related_objects.items():\n            if obj is None:\n                continue\n            if hasattr(obj, \"tags\"):\n                tags.update(obj.tags)\n            related.append(object_as_related_resource(kind=kind, role=kind, object=obj))\n\n        return related + tags_as_related_resources(tags)\n\n    def prepare_for_flow_run(\n        self,\n        flow_run: \"FlowRun\",\n        deployment: Optional[\"DeploymentResponse\"] = None,\n        flow: Optional[\"Flow\"] = None,\n    ):\n        self.block = self.block.prepare_for_flow_run(\n            flow_run=flow_run, deployment=deployment, flow=flow\n        )\n
    ","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block.BlockWorkerJobConfiguration.from_template_and_values","title":"from_template_and_values async classmethod","text":"

    Creates a valid worker configuration object from the provided base configuration and overrides.

    Important: this method expects that the base_job_template was already validated server-side.

    Source code in prefect/workers/block.py
    @classmethod\n@inject_client\nasync def from_template_and_values(\n    cls, base_job_template: dict, values: dict, client: \"PrefectClient\" = None\n):\n    \"\"\"Creates a valid worker configuration object from the provided base\n    configuration and overrides.\n\n    Important: this method expects that the base_job_template was already\n    validated server-side.\n    \"\"\"\n    job_config: Dict[str, Any] = base_job_template[\"job_configuration\"]\n    variables_schema = base_job_template[\"variables\"]\n    variables = cls._get_base_config_defaults(\n        variables_schema.get(\"properties\", {})\n    )\n    variables.update(values)\n\n    populated_configuration = apply_values(template=job_config, values=variables)\n\n    block_document_id = get_from_dict(\n        populated_configuration, \"block.$ref.block_document_id\"\n    )\n    if not block_document_id:\n        raise ValueError(\n            \"Base job template is invalid for this worker type because it does not\"\n            \" contain a block_document_id after variable resolution.\"\n        )\n\n    block_document = await client.read_block_document(\n        block_document_id=block_document_id\n    )\n    infrastructure_block = Block._from_block_document(block_document)\n\n    populated_configuration[\"block\"] = infrastructure_block\n\n    return cls(**populated_configuration)\n
    ","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block.BlockWorkerJobConfiguration.json_template","title":"json_template classmethod","text":"

    Returns a dict with job configuration as keys and the corresponding templates as values

    Defaults to using the job configuration parameter name as the template variable name.

    e.g. { key1: '{{ key1 }}', # default variable template key2: '{{ template2 }}', # template2 specifically provide as template }

    Source code in prefect/workers/block.py
    @classmethod\ndef json_template(cls) -> dict:\n    \"\"\"Returns a dict with job configuration as keys and the corresponding templates as values\n\n    Defaults to using the job configuration parameter name as the template variable name.\n\n    e.g.\n    {\n        key1: '{{ key1 }}',     # default variable template\n        key2: '{{ template2 }}', # `template2` specifically provide as template\n    }\n    \"\"\"\n    configuration = {}\n    properties = cls.schema()[\"properties\"]\n    for k, v in properties.items():\n        if v.get(\"template\"):\n            template = v[\"template\"]\n        else:\n            template = \"{{ \" + k + \" }}\"\n        configuration[k] = template\n\n    return configuration\n
    ","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block.BlockWorkerResult","title":"BlockWorkerResult","text":"

    Bases: BaseWorkerResult

    Result of a block worker job

    Source code in prefect/workers/block.py
    class BlockWorkerResult(BaseWorkerResult):\n    \"\"\"Result of a block worker job\"\"\"\n
    ","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/process/","title":"process","text":"","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/process/#prefect.workers.process","title":"prefect.workers.process","text":"

    Module containing the Process worker used for executing flow runs as subprocesses.

    To start a Process worker, run the following command:

    prefect worker start --pool 'my-work-pool' --type process\n

    Replace my-work-pool with the name of the work pool you want the worker to poll for flow runs.

    For more information about work pools and workers, checkout out the Prefect docs.

    ","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/process/#prefect.workers.process.ProcessJobConfiguration","title":"ProcessJobConfiguration","text":"

    Bases: BaseJobConfiguration

    Source code in prefect/workers/process.py
    class ProcessJobConfiguration(BaseJobConfiguration):\n    stream_output: bool = Field(default=True)\n    working_dir: Optional[Path] = Field(default=None)\n\n    @validator(\"working_dir\")\n    def validate_command(cls, v):\n        \"\"\"Make sure that the working directory is formatted for the current platform.\"\"\"\n        if v:\n            return relative_path_to_current_platform(v)\n        return v\n\n    def prepare_for_flow_run(\n        self,\n        flow_run: \"FlowRun\",\n        deployment: Optional[\"DeploymentResponse\"] = None,\n        flow: Optional[\"Flow\"] = None,\n    ):\n        super().prepare_for_flow_run(flow_run, deployment, flow)\n\n        self.env = {**os.environ, **self.env}\n        self.command = (\n            f\"{get_sys_executable()} -m prefect.engine\"\n            if self.command == self._base_flow_run_command()\n            else self.command\n        )\n\n    def _base_flow_run_command(self) -> str:\n        \"\"\"\n        Override the base flow run command because enhanced cancellation doesn't\n        work with the process worker.\n        \"\"\"\n        return \"python -m prefect.engine\"\n
    ","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/process/#prefect.workers.process.ProcessJobConfiguration.validate_command","title":"validate_command","text":"

    Make sure that the working directory is formatted for the current platform.

    Source code in prefect/workers/process.py
    @validator(\"working_dir\")\ndef validate_command(cls, v):\n    \"\"\"Make sure that the working directory is formatted for the current platform.\"\"\"\n    if v:\n        return relative_path_to_current_platform(v)\n    return v\n
    ","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/process/#prefect.workers.process.ProcessWorkerResult","title":"ProcessWorkerResult","text":"

    Bases: BaseWorkerResult

    Contains information about the final state of a completed process

    Source code in prefect/workers/process.py
    class ProcessWorkerResult(BaseWorkerResult):\n    \"\"\"Contains information about the final state of a completed process\"\"\"\n
    ","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/server/","title":"server","text":"","tags":["Python API","workers","server"]},{"location":"api-ref/prefect/workers/server/#prefect.workers.server","title":"prefect.workers.server","text":"","tags":["Python API","workers","server"]},{"location":"api-ref/prefect/workers/server/#prefect.workers.server.start_healthcheck_server","title":"start_healthcheck_server","text":"

    Run a healthcheck FastAPI server for a worker.

    Parameters:

    Name Type Description Default worker BaseWorker | ProcessWorker

    the worker whose health we will check

    required log_level str

    the log level to use for the server

    'error' Source code in prefect/workers/server.py
    def start_healthcheck_server(\n    worker: Union[BaseWorker, ProcessWorker],\n    query_interval_seconds: int,\n    log_level: str = \"error\",\n) -> None:\n    \"\"\"\n    Run a healthcheck FastAPI server for a worker.\n\n    Args:\n        worker (BaseWorker | ProcessWorker): the worker whose health we will check\n        log_level (str): the log level to use for the server\n    \"\"\"\n    webserver = FastAPI()\n    router = APIRouter()\n\n    def perform_health_check():\n        did_recently_poll = worker.is_worker_still_polling(\n            query_interval_seconds=query_interval_seconds\n        )\n\n        if not did_recently_poll:\n            return JSONResponse(\n                status_code=status.HTTP_503_SERVICE_UNAVAILABLE,\n                content={\"message\": \"Worker may be unresponsive at this time\"},\n            )\n        return JSONResponse(status_code=status.HTTP_200_OK, content={\"message\": \"OK\"})\n\n    router.add_api_route(\"/health\", perform_health_check, methods=[\"GET\"])\n\n    webserver.include_router(router)\n\n    uvicorn.run(\n        webserver,\n        host=PREFECT_WORKER_WEBSERVER_HOST.value(),\n        port=PREFECT_WORKER_WEBSERVER_PORT.value(),\n        log_level=log_level,\n    )\n
    ","tags":["Python API","workers","server"]},{"location":"api-ref/prefect/workers/utilities/","title":"utilities","text":"","tags":["Python API","workers","utilities"]},{"location":"api-ref/prefect/workers/utilities/#prefect.workers.utilities","title":"prefect.workers.utilities","text":"","tags":["Python API","workers","utilities"]},{"location":"api-ref/python/","title":"Python SDK","text":"

    The Prefect Python SDK is used to build, test, and execute workflows against the Prefect API.

    Explore the modules in the navigation bar to the left to learn more.

    ","tags":["API","Python SDK"]},{"location":"api-ref/rest-api/","title":"REST API","text":"

    The Prefect REST API is used for communicating data from clients to a Prefect server so that orchestration can be performed. This API is consumed by clients such as the Prefect Python SDK or the server dashboard.

    Prefect Cloud and a locally hosted Prefect server each provide a REST API.

    • Prefect Cloud:
      • Interactive Prefect Cloud REST API documentation
      • Finding your Prefect Cloud details
    • A Locally hosted open-source Prefect server:
      • Interactive REST API documentation for a locally hosted open-source Prefect server is available at http://localhost:4200/docs or the /docs endpoint of the PREFECT_API_URL you have configured to access the server. You must have the server running with prefect server start to access the interactive documentation.
      • Prefect REST API documentation
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#interacting-with-the-rest-api","title":"Interacting with the REST API","text":"

    You have many options to interact with the Prefect REST API:

    • Create an instance of PrefectClient
    • Use your favorite Python HTTP library such as Requests or HTTPX
    • Use an HTTP library in your language of choice
    • Use curl from the command line
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#prefectclient-with-a-prefect-server","title":"PrefectClient with a Prefect server","text":"

    This example uses PrefectClient with a locally hosted Prefect server:

    import asyncio\nfrom prefect.client import get_client\n\nasync def get_flows():\n    client = get_client()\n    r = await client.read_flows(limit=5)\n    return r\n\nr = asyncio.run(get_flows())\n\nfor flow in r:\n    print(flow.name, flow.id)\n\nif __name__ == \"__main__\":\n    asyncio.run(get_flows())\n

    Output:

    cat-facts 58ed68b1-0201-4f37-adef-0ea24bd2a022\ndog-facts e7c0403d-44e7-45cf-a6c8-79117b7f3766\nsloth-facts 771c0574-f5bf-4f59-a69d-3be3e061a62d\ncapybara-facts fbadaf8b-584f-48b9-b092-07d351edd424\nlemur-facts 53f710e7-3b0f-4b2f-ab6b-44934111818c\n
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#requests-with-prefect","title":"Requests with Prefect","text":"

    This example uses the Requests library with Prefect Cloud to return the five newest artifacts.

    import requests\n\nPREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/abc-my-cloud-account-id-is-here/workspaces/123-my-workspace-id-is-here\"\nPREFECT_API_KEY=\"123abc_my_api_key_goes_here\"\ndata = {\n    \"sort\": \"CREATED_DESC\",\n    \"limit\": 5,\n    \"artifacts\": {\n        \"key\": {\n            \"exists_\": True\n        }\n    }\n}\n\nheaders = {\"Authorization\": f\"Bearer {PREFECT_API_KEY}\"}\nendpoint = f\"{PREFECT_API_URL}/artifacts/filter\"\n\nresponse = requests.post(endpoint, headers=headers, json=data)\nassert response.status_code == 200\nfor artifact in response.json():\n    print(artifact)\n
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#curl-with-prefect-cloud","title":"curl with Prefect Cloud","text":"

    This example uses curl with Prefect Cloud to create a flow run:

    ACCOUNT_ID=\"abc-my-cloud-account-id-goes-here\"\nWORKSPACE_ID=\"123-my-workspace-id-goes-here\"\nPREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/$ACCOUNT_ID/workspaces/$WORKSPACE_ID\"\nPREFECT_API_KEY=\"123abc_my_api_key_goes_here\"\nDEPLOYMENT_ID=\"my_deployment_id\"\n\ncurl --location --request POST \"$PREFECT_API_URL/deployments/$DEPLOYMENT_ID/create_flow_run\" \\\n  --header \"Content-Type: application/json\" \\\n  --header \"Authorization: Bearer $PREFECT_API_KEY\" \\\n  --header \"X-PREFECT-API-VERSION: 0.8.4\" \\\n  --data-raw \"{}\"\n

    Note that in this example --data-raw \"{}\" is required and is where you can specify other aspects of the flow run such as the state. Windows users substitute ^ for \\ for line multi-line commands.

    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#finding-your-prefect-cloud-details","title":"Finding your Prefect Cloud details","text":"

    When working with the Prefect Cloud REST API you will need your Account ID and often the Workspace ID for the workspace you want to interact with. You can find both IDs for a Prefect profile in the CLI with prefect profile inspect my_profile. This command will also display your Prefect API key, as shown below:

    PREFECT_API_URL='https://api.prefect.cloud/api/accounts/abc-my-account-id-is-here/workspaces/123-my-workspace-id-is-here'\nPREFECT_API_KEY='123abc_my_api_key_is_here'\n

    Alternatively, view your Account ID and Workspace ID in your browser URL. For example: https://app.prefect.cloud/account/abc-my-account-id-is-here/workspaces/123-my-workspace-id-is-here.

    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#rest-guidelines","title":"REST Guidelines","text":"

    The REST APIs adhere to the following guidelines:

    • Collection names are pluralized (for example, /flows or /runs).
    • We indicate variable placeholders with colons: GET /flows/:id.
    • We use snake case for route names: GET /task_runs.
    • We avoid nested resources unless there is no possibility of accessing the child resource outside the parent context. For example, we query /task_runs with a flow run filter instead of accessing /flow_runs/:id/task_runs.
    • The API is hosted with an /api/:version prefix that (optionally) allows versioning in the future. By convention, we treat that as part of the base URL and do not include that in API examples.
    • Filtering, sorting, and pagination parameters are provided in the request body of POST requests where applicable.
      • Pagination parameters are limit and offset.
      • Sorting is specified with a single sort parameter.
      • See more information on filtering below.
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#http-verbs","title":"HTTP verbs","text":"
    • GET, PUT and DELETE requests are always idempotent. POST and PATCH are not guaranteed to be idempotent.
    • GET requests cannot receive information from the request body.
    • POST requests can receive information from the request body.
    • POST /collection creates a new member of the collection.
    • GET /collection lists all members of the collection.
    • GET /collection/:id gets a specific member of the collection by ID.
    • DELETE /collection/:id deletes a specific member of the collection.
    • PUT /collection/:id creates or replaces a specific member of the collection.
    • PATCH /collection/:id partially updates a specific member of the collection.
    • POST /collection/action is how we implement non-CRUD actions. For example, to set a flow run's state, we use POST /flow_runs/:id/set_state.
    • POST /collection/action may also be used for read-only queries. This is to allow us to send complex arguments as body arguments (which often cannot be done via GET). Examples include POST /flow_runs/filter, POST /flow_runs/count, and POST /flow_runs/history.
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#filtering","title":"Filtering","text":"

    Objects can be filtered by providing filter criteria in the body of a POST request. When multiple criteria are specified, logical AND will be applied to the criteria.

    Filter criteria are structured as follows:

    {\n    \"objects\": {\n        \"object_field\": {\n            \"field_operator_\": <field_value>\n        }\n    }\n}\n

    In this example, objects is the name of the collection to filter over (for example, flows). The collection can be either the object being queried for (flows for POST /flows/filter) or a related object (flow_runs for POST /flows/filter).

    object_field is the name of the field over which to filter (name for flows). Note that some objects may have nested object fields, such as {flow_run: {state: {type: {any_: []}}}}.

    field_operator_ is the operator to apply to a field when filtering. Common examples include:

    • any_: return objects where this field matches any of the following values.
    • is_null_: return objects where this field is or is not null.
    • eq_: return objects where this field is equal to the following value.
    • all_: return objects where this field matches all of the following values.
    • before_: return objects where this datetime field is less than or equal to the following value.
    • after_: return objects where this datetime field is greater than or equal to the following value.

    For example, to query for flows with the tag \"database\" and failed flow runs, POST /flows/filter with the following request body:

    {\n    \"flows\": {\n        \"tags\": {\n            \"all_\": [\"database\"]\n        }\n    },\n    \"flow_runs\": {\n        \"state\": {\n            \"type\": {\n              \"any_\": [\"FAILED\"]\n            }\n        }\n    }\n}\n
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#openapi","title":"OpenAPI","text":"

    The Prefect REST API can be fully described with an OpenAPI 3.0 compliant document. OpenAPI is a standard specification for describing REST APIs.

    To generate the Prefect server's complete OpenAPI document, run the following commands in an interactive Python session:

    from prefect.server.api.server import create_app\n\napp = create_app()\nopenapi_doc = app.openapi()\n

    This document allows you to generate your own API client, explore the API using an API inspection tool, or write tests to ensure API compliance.

    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/server/","title":"Server API","text":"

    The Prefect server API is used by the server to work with workflow metadata and enforce orchestration logic. This API is primarily used by Prefect developers.

    Select links in the left navigation menu to explore.

    ","tags":["API","Server API"]},{"location":"api-ref/server/api/admin/","title":"server.api.admin","text":"","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin","title":"prefect.server.api.admin","text":"

    Routes for admin-level interactions with the Prefect REST API.

    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.clear_database","title":"clear_database async","text":"

    Clear all database tables without dropping them.

    Source code in prefect/server/api/admin.py
    @router.post(\"/database/clear\", status_code=status.HTTP_204_NO_CONTENT)\nasync def clear_database(\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    confirm: bool = Body(\n        False,\n        embed=True,\n        description=\"Pass confirm=True to confirm you want to modify the database.\",\n    ),\n    response: Response = None,\n):\n    \"\"\"Clear all database tables without dropping them.\"\"\"\n    if not confirm:\n        response.status_code = status.HTTP_400_BAD_REQUEST\n        return\n    async with db.session_context(begin_transaction=True) as session:\n        # work pool has a circular dependency on pool queue; delete it first\n        await session.execute(db.WorkPool.__table__.delete())\n        for table in reversed(db.Base.metadata.sorted_tables):\n            await session.execute(table.delete())\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.create_database","title":"create_database async","text":"

    Create all database objects.

    Source code in prefect/server/api/admin.py
    @router.post(\"/database/create\", status_code=status.HTTP_204_NO_CONTENT)\nasync def create_database(\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    confirm: bool = Body(\n        False,\n        embed=True,\n        description=\"Pass confirm=True to confirm you want to modify the database.\",\n    ),\n    response: Response = None,\n):\n    \"\"\"Create all database objects.\"\"\"\n    if not confirm:\n        response.status_code = status.HTTP_400_BAD_REQUEST\n        return\n\n    await db.create_db()\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.drop_database","title":"drop_database async","text":"

    Drop all database objects.

    Source code in prefect/server/api/admin.py
    @router.post(\"/database/drop\", status_code=status.HTTP_204_NO_CONTENT)\nasync def drop_database(\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    confirm: bool = Body(\n        False,\n        embed=True,\n        description=\"Pass confirm=True to confirm you want to modify the database.\",\n    ),\n    response: Response = None,\n):\n    \"\"\"Drop all database objects.\"\"\"\n    if not confirm:\n        response.status_code = status.HTTP_400_BAD_REQUEST\n        return\n\n    await db.drop_db()\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.read_settings","title":"read_settings async","text":"

    Get the current Prefect REST API settings.

    Secret setting values will be obfuscated.

    Source code in prefect/server/api/admin.py
    @router.get(\"/settings\")\nasync def read_settings() -> prefect.settings.Settings:\n    \"\"\"\n    Get the current Prefect REST API settings.\n\n    Secret setting values will be obfuscated.\n    \"\"\"\n    return prefect.settings.get_current_settings().with_obfuscated_secrets()\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.read_version","title":"read_version async","text":"

    Returns the Prefect version number

    Source code in prefect/server/api/admin.py
    @router.get(\"/version\")\nasync def read_version() -> str:\n    \"\"\"Returns the Prefect version number\"\"\"\n    return prefect.__version__\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/dependencies/","title":"server.api.dependencies","text":"","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/dependencies/#prefect.server.api.dependencies","title":"prefect.server.api.dependencies","text":"

    Utilities for injecting FastAPI dependencies.

    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/dependencies/#prefect.server.api.dependencies.EnforceMinimumAPIVersion","title":"EnforceMinimumAPIVersion","text":"

    FastAPI Dependency used to check compatibility between the version of the api and a given request.

    Looks for the header 'X-PREFECT-API-VERSION' in the request and compares it to the api's version. Rejects requests that are lower than the minimum version.

    Source code in prefect/server/api/dependencies.py
    class EnforceMinimumAPIVersion:\n    \"\"\"\n    FastAPI Dependency used to check compatibility between the version of the api\n    and a given request.\n\n    Looks for the header 'X-PREFECT-API-VERSION' in the request and compares it\n    to the api's version. Rejects requests that are lower than the minimum version.\n    \"\"\"\n\n    def __init__(self, minimum_api_version: str, logger: logging.Logger):\n        self.minimum_api_version = minimum_api_version\n        versions = [int(v) for v in minimum_api_version.split(\".\")]\n        self.api_major = versions[0]\n        self.api_minor = versions[1]\n        self.api_patch = versions[2]\n        self.logger = logger\n\n    async def __call__(\n        self,\n        x_prefect_api_version: str = Header(None),\n    ):\n        request_version = x_prefect_api_version\n\n        # if no version header, assume latest and continue\n        if not request_version:\n            return\n\n        # parse version\n        try:\n            major, minor, patch = [int(v) for v in request_version.split(\".\")]\n        except ValueError:\n            await self._notify_of_invalid_value(request_version)\n            raise HTTPException(\n                status_code=status.HTTP_400_BAD_REQUEST,\n                detail=(\n                    \"Invalid X-PREFECT-API-VERSION header format.\"\n                    f\"Expected header in format 'x.y.z' but received {request_version}\"\n                ),\n            )\n\n        if (major, minor, patch) < (self.api_major, self.api_minor, self.api_patch):\n            await self._notify_of_outdated_version(request_version)\n            raise HTTPException(\n                status_code=status.HTTP_400_BAD_REQUEST,\n                detail=(\n                    f\"The request specified API version {request_version} but this \"\n                    f\"server requires version {self.minimum_api_version} or higher.\"\n                ),\n            )\n\n    async def _notify_of_invalid_value(self, request_version: str):\n        self.logger.error(\n            f\"Invalid X-PREFECT-API-VERSION header format: '{request_version}'\"\n        )\n\n    async def _notify_of_outdated_version(self, request_version: str):\n        self.logger.error(\n            f\"X-PREFECT-API-VERSION header specifies version '{request_version}' \"\n            f\"but minimum allowed version is '{self.minimum_api_version}'\"\n        )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/dependencies/#prefect.server.api.dependencies.LimitBody","title":"LimitBody","text":"

    A fastapi.Depends factory for pulling a limit: int parameter from the request body while determining the default from the current settings.

    Source code in prefect/server/api/dependencies.py
    def LimitBody() -> Depends:\n    \"\"\"\n    A `fastapi.Depends` factory for pulling a `limit: int` parameter from the\n    request body while determining the default from the current settings.\n    \"\"\"\n\n    def get_limit(\n        limit: int = Body(\n            None,\n            description=\"Defaults to PREFECT_API_DEFAULT_LIMIT if not provided.\",\n        )\n    ):\n        default_limit = PREFECT_API_DEFAULT_LIMIT.value()\n        limit = limit if limit is not None else default_limit\n        if not limit >= 0:\n            raise HTTPException(\n                status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,\n                detail=\"Invalid limit: must be greater than or equal to 0.\",\n            )\n        if limit > default_limit:\n            raise HTTPException(\n                status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,\n                detail=f\"Invalid limit: must be less than or equal to {default_limit}.\",\n            )\n        return limit\n\n    return Depends(get_limit)\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/deployments/","title":"server.api.deployments","text":"","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments","title":"prefect.server.api.deployments","text":"

    Routes for interacting with Deployment objects.

    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.count_deployments","title":"count_deployments async","text":"

    Count deployments.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/count\")\nasync def count_deployments(\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_pool_queues: schemas.filters.WorkQueueFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> int:\n    \"\"\"\n    Count deployments.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.deployments.count_deployments(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n        )\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.create_deployment","title":"create_deployment async","text":"

    Gracefully creates a new deployment from the provided schema. If a deployment with the same name and flow_id already exists, the deployment is updated.

    If the deployment has an active schedule, flow runs will be scheduled. When upserting, any scheduled runs from the existing deployment will be deleted.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/\")\nasync def create_deployment(\n    deployment: schemas.actions.DeploymentCreate,\n    response: Response,\n    worker_lookups: WorkerLookups = Depends(WorkerLookups),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.responses.DeploymentResponse:\n    \"\"\"\n    Gracefully creates a new deployment from the provided schema. If a deployment with\n    the same name and flow_id already exists, the deployment is updated.\n\n    If the deployment has an active schedule, flow runs will be scheduled.\n    When upserting, any scheduled runs from the existing deployment will be deleted.\n    \"\"\"\n\n    async with db.session_context(begin_transaction=True) as session:\n        if (\n            deployment.work_pool_name\n            and deployment.work_pool_name != DEFAULT_AGENT_WORK_POOL_NAME\n        ):\n            # Make sure that deployment is valid before beginning creation process\n            work_pool = await models.workers.read_work_pool_by_name(\n                session=session, work_pool_name=deployment.work_pool_name\n            )\n            if work_pool is None:\n                raise HTTPException(\n                    status_code=status.HTTP_404_NOT_FOUND,\n                    detail=f'Work pool \"{deployment.work_pool_name}\" not found.',\n                )\n            try:\n                deployment.check_valid_configuration(work_pool.base_job_template)\n            except (MissingVariableError, jsonschema.exceptions.ValidationError) as exc:\n                raise HTTPException(\n                    status_code=status.HTTP_409_CONFLICT,\n                    detail=f\"Error creating deployment: {exc!r}\",\n                )\n\n        # hydrate the input model into a full model\n        deployment_dict = deployment.dict(exclude={\"work_pool_name\"})\n        if deployment.work_pool_name and deployment.work_queue_name:\n            # If a specific pool name/queue name combination was provided, get the\n            # ID for that work pool queue.\n            deployment_dict[\"work_queue_id\"] = (\n                await worker_lookups._get_work_queue_id_from_name(\n                    session=session,\n                    work_pool_name=deployment.work_pool_name,\n                    work_queue_name=deployment.work_queue_name,\n                    create_queue_if_not_found=True,\n                )\n            )\n        elif deployment.work_pool_name:\n            # If just a pool name was provided, get the ID for its default\n            # work pool queue.\n            deployment_dict[\"work_queue_id\"] = (\n                await worker_lookups._get_default_work_queue_id_from_work_pool_name(\n                    session=session,\n                    work_pool_name=deployment.work_pool_name,\n                )\n            )\n        elif deployment.work_queue_name:\n            # If just a queue name was provided, ensure that the queue exists and\n            # get its ID.\n            work_queue = await models.work_queues._ensure_work_queue_exists(\n                session=session, name=deployment.work_queue_name\n            )\n            deployment_dict[\"work_queue_id\"] = work_queue.id\n\n        deployment = schemas.core.Deployment(**deployment_dict)\n        # check to see if relevant blocks exist, allowing us throw a useful error message\n        # for debugging\n        if deployment.infrastructure_document_id is not None:\n            infrastructure_block = (\n                await models.block_documents.read_block_document_by_id(\n                    session=session,\n                    block_document_id=deployment.infrastructure_document_id,\n                )\n            )\n            if not infrastructure_block:\n                raise HTTPException(\n                    status_code=status.HTTP_409_CONFLICT,\n                    detail=(\n                        \"Error creating deployment. Could not find infrastructure\"\n                        f\" block with id: {deployment.infrastructure_document_id}. This\"\n                        \" usually occurs when applying a deployment specification that\"\n                        \" was built against a different Prefect database / workspace.\"\n                    ),\n                )\n\n        if deployment.storage_document_id is not None:\n            infrastructure_block = (\n                await models.block_documents.read_block_document_by_id(\n                    session=session,\n                    block_document_id=deployment.storage_document_id,\n                )\n            )\n            if not infrastructure_block:\n                raise HTTPException(\n                    status_code=status.HTTP_409_CONFLICT,\n                    detail=(\n                        \"Error creating deployment. Could not find storage block with\"\n                        f\" id: {deployment.storage_document_id}. This usually occurs\"\n                        \" when applying a deployment specification that was built\"\n                        \" against a different Prefect database / workspace.\"\n                    ),\n                )\n\n        now = pendulum.now(\"UTC\")\n        model = await models.deployments.create_deployment(\n            session=session, deployment=deployment\n        )\n\n        if model.created >= now:\n            response.status_code = status.HTTP_201_CREATED\n\n        return schemas.responses.DeploymentResponse.from_orm(model)\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.create_flow_run_from_deployment","title":"create_flow_run_from_deployment async","text":"

    Create a flow run from a deployment.

    Any parameters not provided will be inferred from the deployment's parameters. If tags are not provided, the deployment's tags will be used.

    If no state is provided, the flow run will be created in a SCHEDULED state.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/{id}/create_flow_run\")\nasync def create_flow_run_from_deployment(\n    flow_run: schemas.actions.DeploymentFlowRunCreate,\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    worker_lookups: WorkerLookups = Depends(WorkerLookups),\n    response: Response = None,\n) -> schemas.responses.FlowRunResponse:\n    \"\"\"\n    Create a flow run from a deployment.\n\n    Any parameters not provided will be inferred from the deployment's parameters.\n    If tags are not provided, the deployment's tags will be used.\n\n    If no state is provided, the flow run will be created in a SCHEDULED state.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        # get relevant info from the deployment\n        deployment = await models.deployments.read_deployment(\n            session=session, deployment_id=deployment_id\n        )\n\n        if not deployment:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n\n        parameters = deployment.parameters\n        parameters.update(flow_run.parameters or {})\n\n        if deployment.enforce_parameter_schema:\n            if not isinstance(deployment.parameter_openapi_schema, dict):\n                raise HTTPException(\n                    status.HTTP_409_CONFLICT,\n                    detail=(\n                        \"Error updating deployment: Cannot update parameters because\"\n                        \" parameter schema enforcement is enabled and the deployment\"\n                        \" does not have a valid parameter schema.\"\n                    ),\n                )\n            try:\n                validate_values_conform_to_schema(\n                    parameters, deployment.parameter_openapi_schema\n                )\n            except ValueError as exc:\n                raise HTTPException(\n                    status.HTTP_409_CONFLICT, detail=f\"Error creating flow run: {exc}\"\n                )\n\n        work_queue_name = deployment.work_queue_name\n        work_queue_id = deployment.work_queue_id\n\n        if flow_run.work_queue_name:\n            # can't mutate the ORM model or else it will commit the changes back\n            work_queue_id = await worker_lookups._get_work_queue_id_from_name(\n                session=session,\n                work_pool_name=deployment.work_queue.work_pool.name,\n                work_queue_name=flow_run.work_queue_name,\n                create_queue_if_not_found=True,\n            )\n            work_queue_name = flow_run.work_queue_name\n\n        # hydrate the input model into a full flow run / state model\n        flow_run = schemas.core.FlowRun(\n            **flow_run.dict(\n                exclude={\n                    \"parameters\",\n                    \"tags\",\n                    \"infrastructure_document_id\",\n                    \"work_queue_name\",\n                }\n            ),\n            flow_id=deployment.flow_id,\n            deployment_id=deployment.id,\n            parameters=parameters,\n            tags=set(deployment.tags).union(flow_run.tags),\n            infrastructure_document_id=(\n                flow_run.infrastructure_document_id\n                or deployment.infrastructure_document_id\n            ),\n            work_queue_name=work_queue_name,\n            work_queue_id=work_queue_id,\n        )\n\n        if not flow_run.state:\n            flow_run.state = schemas.states.Scheduled()\n\n        now = pendulum.now(\"UTC\")\n        model = await models.flow_runs.create_flow_run(\n            session=session, flow_run=flow_run\n        )\n        if model.created >= now:\n            response.status_code = status.HTTP_201_CREATED\n        return schemas.responses.FlowRunResponse.from_orm(model)\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.delete_deployment","title":"delete_deployment async","text":"

    Delete a deployment by id.

    Source code in prefect/server/api/deployments.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_deployment(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a deployment by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.deployments.delete_deployment(\n            session=session, deployment_id=deployment_id\n        )\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n        )\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.get_scheduled_flow_runs_for_deployments","title":"get_scheduled_flow_runs_for_deployments async","text":"

    Get scheduled runs for a set of deployments. Used by a runner to poll for work.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/get_scheduled_flow_runs\")\nasync def get_scheduled_flow_runs_for_deployments(\n    deployment_ids: List[UUID] = Body(\n        default=..., description=\"The deployment IDs to get scheduled runs for\"\n    ),\n    scheduled_before: DateTimeTZ = Body(\n        None, description=\"The maximum time to look for scheduled flow runs\"\n    ),\n    limit: int = dependencies.LimitBody(),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.FlowRunResponse]:\n    \"\"\"\n    Get scheduled runs for a set of deployments. Used by a runner to poll for work.\n    \"\"\"\n    async with db.session_context() as session:\n        orm_flow_runs = await models.flow_runs.read_flow_runs(\n            session=session,\n            limit=limit,\n            deployment_filter=schemas.filters.DeploymentFilter(\n                id=schemas.filters.DeploymentFilterId(any_=deployment_ids),\n            ),\n            flow_run_filter=schemas.filters.FlowRunFilter(\n                next_scheduled_start_time=schemas.filters.FlowRunFilterNextScheduledStartTime(\n                    before_=scheduled_before\n                ),\n                state=schemas.filters.FlowRunFilterState(\n                    type=schemas.filters.FlowRunFilterStateType(\n                        any_=[schemas.states.StateType.SCHEDULED]\n                    )\n                ),\n            ),\n            sort=schemas.sorting.FlowRunSort.NEXT_SCHEDULED_START_TIME_ASC,\n        )\n\n        flow_run_responses = [\n            schemas.responses.FlowRunResponse.from_orm(orm_flow_run=orm_flow_run)\n            for orm_flow_run in orm_flow_runs\n        ]\n\n    async with db.session_context(\n        begin_transaction=True, with_for_update=True\n    ) as session:\n        await models.deployments._update_deployment_last_polled(\n            session=session, deployment_ids=deployment_ids\n        )\n\n    return flow_run_responses\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.read_deployment","title":"read_deployment async","text":"

    Get a deployment by id.

    Source code in prefect/server/api/deployments.py
    @router.get(\"/{id}\")\nasync def read_deployment(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.responses.DeploymentResponse:\n    \"\"\"\n    Get a deployment by id.\n    \"\"\"\n    async with db.session_context() as session:\n        deployment = await models.deployments.read_deployment(\n            session=session, deployment_id=deployment_id\n        )\n        if not deployment:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n        return schemas.responses.DeploymentResponse.from_orm(deployment)\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.read_deployment_by_name","title":"read_deployment_by_name async","text":"

    Get a deployment using the name of the flow and the deployment.

    Source code in prefect/server/api/deployments.py
    @router.get(\"/name/{flow_name}/{deployment_name}\")\nasync def read_deployment_by_name(\n    flow_name: str = Path(..., description=\"The name of the flow\"),\n    deployment_name: str = Path(..., description=\"The name of the deployment\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.responses.DeploymentResponse:\n    \"\"\"\n    Get a deployment using the name of the flow and the deployment.\n    \"\"\"\n    async with db.session_context() as session:\n        deployment = await models.deployments.read_deployment_by_name(\n            session=session, name=deployment_name, flow_name=flow_name\n        )\n        if not deployment:\n            raise HTTPException(\n                status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n        return schemas.responses.DeploymentResponse.from_orm(deployment)\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.read_deployments","title":"read_deployments async","text":"

    Query for deployments.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/filter\")\nasync def read_deployments(\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_pool_queues: schemas.filters.WorkQueueFilter = None,\n    sort: schemas.sorting.DeploymentSort = Body(\n        schemas.sorting.DeploymentSort.NAME_ASC\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.DeploymentResponse]:\n    \"\"\"\n    Query for deployments.\n    \"\"\"\n    async with db.session_context() as session:\n        response = await models.deployments.read_deployments(\n            session=session,\n            offset=offset,\n            sort=sort,\n            limit=limit,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n        )\n        return [\n            schemas.responses.DeploymentResponse.from_orm(orm_deployment=deployment)\n            for deployment in response\n        ]\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.schedule_deployment","title":"schedule_deployment async","text":"

    Schedule runs for a deployment. For backfills, provide start/end times in the past.

    This function will generate the minimum number of runs that satisfy the min and max times, and the min and max counts. Specifically, the following order will be respected.

    - Runs will be generated starting on or after the `start_time`\n- No more than `max_runs` runs will be generated\n- No runs will be generated after `end_time` is reached\n- At least `min_runs` runs will be generated\n- Runs will be generated until at least `start_time + min_time` is reached\n
    Source code in prefect/server/api/deployments.py
    @router.post(\"/{id}/schedule\")\nasync def schedule_deployment(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    start_time: DateTimeTZ = Body(None, description=\"The earliest date to schedule\"),\n    end_time: DateTimeTZ = Body(None, description=\"The latest date to schedule\"),\n    min_time: datetime.timedelta = Body(\n        None,\n        description=(\n            \"Runs will be scheduled until at least this long after the `start_time`\"\n        ),\n    ),\n    min_runs: int = Body(None, description=\"The minimum number of runs to schedule\"),\n    max_runs: int = Body(None, description=\"The maximum number of runs to schedule\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> None:\n    \"\"\"\n    Schedule runs for a deployment. For backfills, provide start/end times in the past.\n\n    This function will generate the minimum number of runs that satisfy the min\n    and max times, and the min and max counts. Specifically, the following order\n    will be respected.\n\n        - Runs will be generated starting on or after the `start_time`\n        - No more than `max_runs` runs will be generated\n        - No runs will be generated after `end_time` is reached\n        - At least `min_runs` runs will be generated\n        - Runs will be generated until at least `start_time + min_time` is reached\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        await models.deployments.schedule_runs(\n            session=session,\n            deployment_id=deployment_id,\n            start_time=start_time,\n            min_time=min_time,\n            end_time=end_time,\n            min_runs=min_runs,\n            max_runs=max_runs,\n        )\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.set_schedule_active","title":"set_schedule_active async","text":"

    Set a deployment schedule to active. Runs will be scheduled immediately.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/{id}/set_schedule_active\")\nasync def set_schedule_active(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> None:\n    \"\"\"\n    Set a deployment schedule to active. Runs will be scheduled immediately.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        deployment = await models.deployments.read_deployment(\n            session=session, deployment_id=deployment_id\n        )\n        if not deployment:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n        deployment.is_schedule_active = True\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.set_schedule_inactive","title":"set_schedule_inactive async","text":"

    Set a deployment schedule to inactive. Any auto-scheduled runs still in a Scheduled state will be deleted.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/{id}/set_schedule_inactive\")\nasync def set_schedule_inactive(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> None:\n    \"\"\"\n    Set a deployment schedule to inactive. Any auto-scheduled runs still in a Scheduled\n    state will be deleted.\n    \"\"\"\n    async with db.session_context(begin_transaction=False) as session:\n        deployment = await models.deployments.read_deployment(\n            session=session, deployment_id=deployment_id\n        )\n        if not deployment:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n        deployment.is_schedule_active = False\n        # commit here to make the inactive schedule \"visible\" to the scheduler service\n        await session.commit()\n\n        # delete any auto scheduled runs\n        await models.deployments._delete_scheduled_runs(\n            session=session,\n            deployment_id=deployment_id,\n            db=db,\n            auto_scheduled_only=True,\n        )\n\n        await session.commit()\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.work_queue_check_for_deployment","title":"work_queue_check_for_deployment async","text":"

    Get list of work-queues that are able to pick up the specified deployment.

    This endpoint is intended to be used by the UI to provide users warnings about deployments that are unable to be executed because there are no work queues that will pick up their runs, based on existing filter criteria. It may be deprecated in the future because there is not a strict relationship between work queues and deployments.

    Source code in prefect/server/api/deployments.py
    @router.get(\"/{id}/work_queue_check\", deprecated=True)\nasync def work_queue_check_for_deployment(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.WorkQueue]:\n    \"\"\"\n    Get list of work-queues that are able to pick up the specified deployment.\n\n    This endpoint is intended to be used by the UI to provide users warnings\n    about deployments that are unable to be executed because there are no work\n    queues that will pick up their runs, based on existing filter criteria. It\n    may be deprecated in the future because there is not a strict relationship\n    between work queues and deployments.\n    \"\"\"\n    try:\n        async with db.session_context() as session:\n            work_queues = await models.deployments.check_work_queues_for_deployment(\n                session=session, deployment_id=deployment_id\n            )\n    except ObjectNotFoundError:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n        )\n    return work_queues\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/flow_run_states/","title":"server.api.flow_run_states","text":"","tags":["Prefect API","flow runs","states"]},{"location":"api-ref/server/api/flow_run_states/#prefect.server.api.flow_run_states","title":"prefect.server.api.flow_run_states","text":"

    Routes for interacting with flow run state objects.

    ","tags":["Prefect API","flow runs","states"]},{"location":"api-ref/server/api/flow_run_states/#prefect.server.api.flow_run_states.read_flow_run_state","title":"read_flow_run_state async","text":"

    Get a flow run state by id.

    Source code in prefect/server/api/flow_run_states.py
    @router.get(\"/{id}\")\nasync def read_flow_run_state(\n    flow_run_state_id: UUID = Path(\n        ..., description=\"The flow run state id\", alias=\"id\"\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.states.State:\n    \"\"\"\n    Get a flow run state by id.\n    \"\"\"\n    async with db.session_context() as session:\n        flow_run_state = await models.flow_run_states.read_flow_run_state(\n            session=session, flow_run_state_id=flow_run_state_id\n        )\n    if not flow_run_state:\n        raise HTTPException(\n            status.HTTP_404_NOT_FOUND, detail=\"Flow run state not found\"\n        )\n    return flow_run_state\n
    ","tags":["Prefect API","flow runs","states"]},{"location":"api-ref/server/api/flow_run_states/#prefect.server.api.flow_run_states.read_flow_run_states","title":"read_flow_run_states async","text":"

    Get states associated with a flow run.

    Source code in prefect/server/api/flow_run_states.py
    @router.get(\"/\")\nasync def read_flow_run_states(\n    flow_run_id: UUID,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.states.State]:\n    \"\"\"\n    Get states associated with a flow run.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flow_run_states.read_flow_run_states(\n            session=session, flow_run_id=flow_run_id\n        )\n
    ","tags":["Prefect API","flow runs","states"]},{"location":"api-ref/server/api/flow_runs/","title":"server.api.flow_runs","text":"","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs","title":"prefect.server.api.flow_runs","text":"

    Routes for interacting with flow run objects.

    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.average_flow_run_lateness","title":"average_flow_run_lateness async","text":"

    Query for average flow-run lateness in seconds.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/lateness\")\nasync def average_flow_run_lateness(\n    flows: Optional[schemas.filters.FlowFilter] = None,\n    flow_runs: Optional[schemas.filters.FlowRunFilter] = None,\n    task_runs: Optional[schemas.filters.TaskRunFilter] = None,\n    deployments: Optional[schemas.filters.DeploymentFilter] = None,\n    work_pools: Optional[schemas.filters.WorkPoolFilter] = None,\n    work_pool_queues: Optional[schemas.filters.WorkQueueFilter] = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> Optional[float]:\n    \"\"\"\n    Query for average flow-run lateness in seconds.\n    \"\"\"\n    async with db.session_context() as session:\n        if db.dialect.name == \"sqlite\":\n            # Since we want an _average_ of the lateness we're unable to use\n            # the existing FlowRun.expected_start_time_delta property as it\n            # returns a timedelta and SQLite is unable to properly deal with it\n            # and always returns 1970.0 as the average. This copies the same\n            # logic but ensures that it returns the number of seconds instead\n            # so it's compatible with SQLite.\n            base_query = sa.case(\n                (\n                    db.FlowRun.start_time > db.FlowRun.expected_start_time,\n                    sa.func.strftime(\"%s\", db.FlowRun.start_time)\n                    - sa.func.strftime(\"%s\", db.FlowRun.expected_start_time),\n                ),\n                (\n                    db.FlowRun.start_time.is_(None)\n                    & db.FlowRun.state_type.notin_(schemas.states.TERMINAL_STATES)\n                    & (db.FlowRun.expected_start_time < sa.func.datetime(\"now\")),\n                    sa.func.strftime(\"%s\", sa.func.datetime(\"now\"))\n                    - sa.func.strftime(\"%s\", db.FlowRun.expected_start_time),\n                ),\n                else_=0,\n            )\n        else:\n            base_query = db.FlowRun.estimated_start_time_delta\n\n        query = await models.flow_runs._apply_flow_run_filters(\n            sa.select(sa.func.avg(base_query)),\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n        )\n        result = await session.execute(query)\n\n        avg_lateness = result.scalar()\n\n        if avg_lateness is None:\n            return None\n        elif isinstance(avg_lateness, datetime.timedelta):\n            return avg_lateness.total_seconds()\n        else:\n            return avg_lateness\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.count_flow_runs","title":"count_flow_runs async","text":"

    Query for flow runs.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/count\")\nasync def count_flow_runs(\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_pool_queues: schemas.filters.WorkQueueFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> int:\n    \"\"\"\n    Query for flow runs.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flow_runs.count_flow_runs(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.create_flow_run","title":"create_flow_run async","text":"

    Create a flow run. If a flow run with the same flow_id and idempotency key already exists, the existing flow run will be returned.

    If no state is provided, the flow run will be created in a PENDING state.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/\")\nasync def create_flow_run(\n    flow_run: schemas.actions.FlowRunCreate,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    response: Response = None,\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_flow_orchestration_parameters\n    ),\n    api_version=Depends(dependencies.provide_request_api_version),\n) -> schemas.responses.FlowRunResponse:\n    \"\"\"\n    Create a flow run. If a flow run with the same flow_id and\n    idempotency key already exists, the existing flow run will be returned.\n\n    If no state is provided, the flow run will be created in a PENDING state.\n    \"\"\"\n    # hydrate the input model into a full flow run / state model\n    flow_run = schemas.core.FlowRun(**flow_run.dict())\n\n    # pass the request version to the orchestration engine to support compatibility code\n    orchestration_parameters.update({\"api-version\": api_version})\n\n    if not flow_run.state:\n        flow_run.state = schemas.states.Pending()\n\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        model = await models.flow_runs.create_flow_run(\n            session=session,\n            flow_run=flow_run,\n            orchestration_parameters=orchestration_parameters,\n        )\n        if model.created >= now:\n            response.status_code = status.HTTP_201_CREATED\n\n        return schemas.responses.FlowRunResponse.from_orm(model)\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.create_flow_run_input","title":"create_flow_run_input async","text":"

    Create a key/value input for a flow run.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/{id}/input\", status_code=status.HTTP_201_CREATED)\nasync def create_flow_run_input(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    key: str = Body(..., description=\"The input key\"),\n    value: bytes = Body(..., description=\"The value of the input\"),\n    sender: Optional[str] = Body(None, description=\"The sender of the input\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Create a key/value input for a flow run.\n    \"\"\"\n    async with db.session_context() as session:\n        try:\n            await models.flow_run_input.create_flow_run_input(\n                session=session,\n                flow_run_input=schemas.core.FlowRunInput(\n                    flow_run_id=flow_run_id,\n                    key=key,\n                    sender=sender,\n                    value=value.decode(),\n                ),\n            )\n            await session.commit()\n\n        except IntegrityError as exc:\n            if \"unique constraint\" in str(exc).lower():\n                raise HTTPException(\n                    status_code=status.HTTP_409_CONFLICT,\n                    detail=\"A flow run input with this key already exists.\",\n                )\n            else:\n                raise HTTPException(\n                    status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run not found\"\n                )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.delete_flow_run","title":"delete_flow_run async","text":"

    Delete a flow run by id.

    Source code in prefect/server/api/flow_runs.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_flow_run(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a flow run by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.flow_runs.delete_flow_run(\n            session=session, flow_run_id=flow_run_id\n        )\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run not found\"\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.delete_flow_run_input","title":"delete_flow_run_input async","text":"

    Delete a flow run input

    Source code in prefect/server/api/flow_runs.py
    @router.delete(\"/{id}/input/{key}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_flow_run_input(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    key: str = Path(..., description=\"The input key\", alias=\"key\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a flow run input\n    \"\"\"\n\n    async with db.session_context() as session:\n        deleted = await models.flow_run_input.delete_flow_run_input(\n            session=session, flow_run_id=flow_run_id, key=key\n        )\n        await session.commit()\n\n        if not deleted:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run input not found\"\n            )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.filter_flow_run_input","title":"filter_flow_run_input async","text":"

    Filter flow run inputs by key prefix

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/{id}/input/filter\")\nasync def filter_flow_run_input(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    prefix: str = Body(..., description=\"The input key prefix\", embed=True),\n    limit: int = Body(\n        1, description=\"The maximum number of results to return\", embed=True\n    ),\n    exclude_keys: List[str] = Body(\n        [], description=\"Exclude inputs with these keys\", embed=True\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.FlowRunInput]:\n    \"\"\"\n    Filter flow run inputs by key prefix\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flow_run_input.filter_flow_run_input(\n            session=session,\n            flow_run_id=flow_run_id,\n            prefix=prefix,\n            limit=limit,\n            exclude_keys=exclude_keys,\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.flow_run_history","title":"flow_run_history async","text":"

    Query for flow run history data across a given range and interval.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/history\")\nasync def flow_run_history(\n    history_start: DateTimeTZ = Body(..., description=\"The history's start time.\"),\n    history_end: DateTimeTZ = Body(..., description=\"The history's end time.\"),\n    history_interval: datetime.timedelta = Body(\n        ...,\n        description=(\n            \"The size of each history interval, in seconds. Must be at least 1 second.\"\n        ),\n        alias=\"history_interval_seconds\",\n    ),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_queues: schemas.filters.WorkQueueFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.HistoryResponse]:\n    \"\"\"\n    Query for flow run history data across a given range and interval.\n    \"\"\"\n    if history_interval < datetime.timedelta(seconds=1):\n        raise HTTPException(\n            status.HTTP_422_UNPROCESSABLE_ENTITY,\n            detail=\"History interval must not be less than 1 second.\",\n        )\n\n    async with db.session_context() as session:\n        return await run_history(\n            session=session,\n            run_type=\"flow_run\",\n            history_start=history_start,\n            history_end=history_end,\n            history_interval=history_interval,\n            flows=flows,\n            flow_runs=flow_runs,\n            task_runs=task_runs,\n            deployments=deployments,\n            work_pools=work_pools,\n            work_queues=work_queues,\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_run","title":"read_flow_run async","text":"

    Get a flow run by id.

    Source code in prefect/server/api/flow_runs.py
    @router.get(\"/{id}\")\nasync def read_flow_run(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.responses.FlowRunResponse:\n    \"\"\"\n    Get a flow run by id.\n    \"\"\"\n    async with db.session_context() as session:\n        flow_run = await models.flow_runs.read_flow_run(\n            session=session, flow_run_id=flow_run_id\n        )\n        if not flow_run:\n            raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Flow run not found\")\n        return schemas.responses.FlowRunResponse.from_orm(flow_run)\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_run_graph_v1","title":"read_flow_run_graph_v1 async","text":"

    Get a task run dependency map for a given flow run.

    Source code in prefect/server/api/flow_runs.py
    @router.get(\"/{id}/graph\")\nasync def read_flow_run_graph_v1(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[DependencyResult]:\n    \"\"\"\n    Get a task run dependency map for a given flow run.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flow_runs.read_task_run_dependencies(\n            session=session, flow_run_id=flow_run_id\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_run_graph_v2","title":"read_flow_run_graph_v2 async","text":"

    Get a graph of the tasks and subflow runs for the given flow run

    Source code in prefect/server/api/flow_runs.py
    @router.get(\"/{id:uuid}/graph-v2\")\nasync def read_flow_run_graph_v2(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    since: datetime.datetime = Query(\n        datetime.datetime.min,\n        description=\"Only include runs that start or end after this time.\",\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> Graph:\n    \"\"\"\n    Get a graph of the tasks and subflow runs for the given flow run\n    \"\"\"\n    async with db.session_context() as session:\n        try:\n            return await read_flow_run_graph(\n                session=session,\n                flow_run_id=flow_run_id,\n                since=since,\n            )\n        except FlowRunGraphTooLarge as e:\n            raise HTTPException(\n                status_code=status.HTTP_400_BAD_REQUEST,\n                detail=str(e),\n            )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_run_input","title":"read_flow_run_input async","text":"

    Create a value from a flow run input

    Source code in prefect/server/api/flow_runs.py
    @router.get(\"/{id}/input/{key}\")\nasync def read_flow_run_input(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    key: str = Path(..., description=\"The input key\", alias=\"key\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> PlainTextResponse:\n    \"\"\"\n    Create a value from a flow run input\n    \"\"\"\n\n    async with db.session_context() as session:\n        flow_run_input = await models.flow_run_input.read_flow_run_input(\n            session=session, flow_run_id=flow_run_id, key=key\n        )\n\n    if flow_run_input:\n        return PlainTextResponse(flow_run_input.value)\n    else:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run input not found\"\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_runs","title":"read_flow_runs async","text":"

    Query for flow runs.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/filter\", response_class=ORJSONResponse)\nasync def read_flow_runs(\n    sort: schemas.sorting.FlowRunSort = Body(schemas.sorting.FlowRunSort.ID_DESC),\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_pool_queues: schemas.filters.WorkQueueFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.FlowRunResponse]:\n    \"\"\"\n    Query for flow runs.\n    \"\"\"\n    async with db.session_context() as session:\n        db_flow_runs = await models.flow_runs.read_flow_runs(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n            offset=offset,\n            limit=limit,\n            sort=sort,\n        )\n\n        # Instead of relying on fastapi.encoders.jsonable_encoder to convert the\n        # response to JSON, we do so more efficiently ourselves.\n        # In particular, the FastAPI encoder is very slow for large, nested objects.\n        # See: https://github.com/tiangolo/fastapi/issues/1224\n        encoded = [\n            schemas.responses.FlowRunResponse.from_orm(fr).dict(json_compatible=True)\n            for fr in db_flow_runs\n        ]\n        return ORJSONResponse(content=encoded)\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.resume_flow_run","title":"resume_flow_run async","text":"

    Resume a paused flow run.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/{id}/resume\")\nasync def resume_flow_run(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    run_input: Optional[Dict] = Body(default=None, embed=True),\n    response: Response = None,\n    flow_policy: BaseOrchestrationPolicy = Depends(\n        orchestration_dependencies.provide_flow_policy\n    ),\n    task_policy: BaseOrchestrationPolicy = Depends(\n        orchestration_dependencies.provide_task_policy\n    ),\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_flow_orchestration_parameters\n    ),\n    api_version=Depends(dependencies.provide_request_api_version),\n) -> OrchestrationResult:\n    \"\"\"\n    Resume a paused flow run.\n    \"\"\"\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        flow_run = await models.flow_runs.read_flow_run(session, flow_run_id)\n        state = flow_run.state\n\n        if state is None or state.type != schemas.states.StateType.PAUSED:\n            result = OrchestrationResult(\n                state=None,\n                status=schemas.responses.SetStateStatus.ABORT,\n                details=schemas.responses.StateAbortDetails(\n                    reason=\"Cannot resume a flow run that is not paused.\"\n                ),\n            )\n            return result\n\n        orchestration_parameters.update({\"api-version\": api_version})\n\n        keyset = state.state_details.run_input_keyset\n        if keyset and not run_input:\n            return OrchestrationResult(\n                state=state,\n                status=schemas.responses.SetStateStatus.REJECT,\n                details=schemas.responses.StateAbortDetails(\n                    reason=\"Flow run was expecting input but none was provided.\"\n                ),\n            )\n        elif keyset and run_input:\n            schema_json = await models.flow_run_input.read_flow_run_input(\n                session=session, flow_run_id=flow_run.id, key=keyset[\"schema\"]\n            )\n\n            if schema_json is None:\n                return OrchestrationResult(\n                    state=state,\n                    status=schemas.responses.SetStateStatus.REJECT,\n                    details=schemas.responses.StateAbortDetails(\n                        reason=\"Run input schema not found.\"\n                    ),\n                )\n\n            try:\n                schema = orjson.loads(schema_json.value)\n            except orjson.JSONDecodeError:\n                return OrchestrationResult(\n                    state=state,\n                    status=schemas.responses.SetStateStatus.REJECT,\n                    details=schemas.responses.StateAbortDetails(\n                        reason=\"Run input schema is not valid JSON.\"\n                    ),\n                )\n\n            try:\n                jsonschema.validate(run_input, schema)\n            except (jsonschema.ValidationError, jsonschema.SchemaError) as exc:\n                return OrchestrationResult(\n                    state=state,\n                    status=schemas.responses.SetStateStatus.REJECT,\n                    details=schemas.responses.StateAbortDetails(\n                        reason=f\"Run input validation failed: {exc.message}\"\n                    ),\n                )\n\n        if state.state_details.pause_reschedule:\n            orchestration_result = await models.flow_runs.set_flow_run_state(\n                session=session,\n                flow_run_id=flow_run_id,\n                state=schemas.states.Scheduled(\n                    name=\"Resuming\", scheduled_time=pendulum.now(\"UTC\")\n                ),\n                flow_policy=flow_policy,\n                orchestration_parameters=orchestration_parameters,\n            )\n        else:\n            orchestration_result = await models.flow_runs.set_flow_run_state(\n                session=session,\n                flow_run_id=flow_run_id,\n                state=schemas.states.Running(),\n                flow_policy=flow_policy,\n                orchestration_parameters=orchestration_parameters,\n            )\n\n        if (\n            keyset\n            and run_input\n            and orchestration_result.status == schemas.responses.SetStateStatus.ACCEPT\n        ):\n            # The state change is accepted, go ahead and store the validated\n            # run input.\n            await models.flow_run_input.create_flow_run_input(\n                session=session,\n                flow_run_input=schemas.core.FlowRunInput(\n                    flow_run_id=flow_run_id,\n                    key=keyset[\"response\"],\n                    value=orjson.dumps(run_input).decode(\"utf-8\"),\n                ),\n            )\n\n        # set the 201 if a new state was created\n        if orchestration_result.state and orchestration_result.state.timestamp >= now:\n            response.status_code = status.HTTP_201_CREATED\n        else:\n            response.status_code = status.HTTP_200_OK\n\n        return orchestration_result\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.set_flow_run_state","title":"set_flow_run_state async","text":"

    Set a flow run state, invoking any orchestration rules.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/{id}/set_state\")\nasync def set_flow_run_state(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    state: schemas.actions.StateCreate = Body(..., description=\"The intended state.\"),\n    force: bool = Body(\n        False,\n        description=(\n            \"If false, orchestration rules will be applied that may alter or prevent\"\n            \" the state transition. If True, orchestration rules are not applied.\"\n        ),\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    response: Response = None,\n    flow_policy: BaseOrchestrationPolicy = Depends(\n        orchestration_dependencies.provide_flow_policy\n    ),\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_flow_orchestration_parameters\n    ),\n    api_version=Depends(dependencies.provide_request_api_version),\n) -> OrchestrationResult:\n    \"\"\"Set a flow run state, invoking any orchestration rules.\"\"\"\n\n    # pass the request version to the orchestration engine to support compatibility code\n    orchestration_parameters.update({\"api-version\": api_version})\n\n    now = pendulum.now(\"UTC\")\n\n    # create the state\n    async with db.session_context(\n        begin_transaction=True, with_for_update=True\n    ) as session:\n        orchestration_result = await models.flow_runs.set_flow_run_state(\n            session=session,\n            flow_run_id=flow_run_id,\n            # convert to a full State object\n            state=schemas.states.State.parse_obj(state),\n            force=force,\n            flow_policy=flow_policy,\n            orchestration_parameters=orchestration_parameters,\n        )\n\n    # set the 201 if a new state was created\n    if orchestration_result.state and orchestration_result.state.timestamp >= now:\n        response.status_code = status.HTTP_201_CREATED\n    else:\n        response.status_code = status.HTTP_200_OK\n\n    return orchestration_result\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.update_flow_run","title":"update_flow_run async","text":"

    Updates a flow run.

    Source code in prefect/server/api/flow_runs.py
    @router.patch(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def update_flow_run(\n    flow_run: schemas.actions.FlowRunUpdate,\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Updates a flow run.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.flow_runs.update_flow_run(\n            session=session, flow_run=flow_run, flow_run_id=flow_run_id\n        )\n    if not result:\n        raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Flow run not found\")\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flows/","title":"server.api.flows","text":"","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows","title":"prefect.server.api.flows","text":"

    Routes for interacting with flow objects.

    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.count_flows","title":"count_flows async","text":"

    Count flows.

    Source code in prefect/server/api/flows.py
    @router.post(\"/count\")\nasync def count_flows(\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> int:\n    \"\"\"\n    Count flows.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flows.count_flows(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n        )\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.create_flow","title":"create_flow async","text":"

    Gracefully creates a new flow from the provided schema. If a flow with the same name already exists, the existing flow is returned.

    Source code in prefect/server/api/flows.py
    @router.post(\"/\")\nasync def create_flow(\n    flow: schemas.actions.FlowCreate,\n    response: Response,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.Flow:\n    \"\"\"Gracefully creates a new flow from the provided schema. If a flow with the\n    same name already exists, the existing flow is returned.\n    \"\"\"\n    # hydrate the input model into a full flow model\n    flow = schemas.core.Flow(**flow.dict())\n\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        model = await models.flows.create_flow(session=session, flow=flow)\n\n    if model.created >= now:\n        response.status_code = status.HTTP_201_CREATED\n    return model\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.delete_flow","title":"delete_flow async","text":"

    Delete a flow by id.

    Source code in prefect/server/api/flows.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_flow(\n    flow_id: UUID = Path(..., description=\"The flow id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a flow by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.flows.delete_flow(session=session, flow_id=flow_id)\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow not found\"\n        )\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.read_flow","title":"read_flow async","text":"

    Get a flow by id.

    Source code in prefect/server/api/flows.py
    @router.get(\"/{id}\")\nasync def read_flow(\n    flow_id: UUID = Path(..., description=\"The flow id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.Flow:\n    \"\"\"\n    Get a flow by id.\n    \"\"\"\n    async with db.session_context() as session:\n        flow = await models.flows.read_flow(session=session, flow_id=flow_id)\n    if not flow:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow not found\"\n        )\n    return flow\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.read_flow_by_name","title":"read_flow_by_name async","text":"

    Get a flow by name.

    Source code in prefect/server/api/flows.py
    @router.get(\"/name/{name}\")\nasync def read_flow_by_name(\n    name: str = Path(..., description=\"The name of the flow\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.Flow:\n    \"\"\"\n    Get a flow by name.\n    \"\"\"\n    async with db.session_context() as session:\n        flow = await models.flows.read_flow_by_name(session=session, name=name)\n    if not flow:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow not found\"\n        )\n    return flow\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.read_flows","title":"read_flows async","text":"

    Query for flows.

    Source code in prefect/server/api/flows.py
    @router.post(\"/filter\")\nasync def read_flows(\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    sort: schemas.sorting.FlowSort = Body(schemas.sorting.FlowSort.NAME_ASC),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.Flow]:\n    \"\"\"\n    Query for flows.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flows.read_flows(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            sort=sort,\n            offset=offset,\n            limit=limit,\n        )\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.update_flow","title":"update_flow async","text":"

    Updates a flow.

    Source code in prefect/server/api/flows.py
    @router.patch(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def update_flow(\n    flow: schemas.actions.FlowUpdate,\n    flow_id: UUID = Path(..., description=\"The flow id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Updates a flow.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.flows.update_flow(\n            session=session, flow=flow, flow_id=flow_id\n        )\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow not found\"\n        )\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/run_history/","title":"server.api.run_history","text":"","tags":["Prefect API","flow runs","task runs","observability"]},{"location":"api-ref/server/api/run_history/#prefect.server.api.run_history","title":"prefect.server.api.run_history","text":"

    Utilities for querying flow and task run history.

    ","tags":["Prefect API","flow runs","task runs","observability"]},{"location":"api-ref/server/api/run_history/#prefect.server.api.run_history.run_history","title":"run_history async","text":"

    Produce a history of runs aggregated by interval and state

    Source code in prefect/server/api/run_history.py
    @inject_db\nasync def run_history(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    run_type: Literal[\"flow_run\", \"task_run\"],\n    history_start: DateTimeTZ,\n    history_end: DateTimeTZ,\n    history_interval: datetime.timedelta,\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_queues: schemas.filters.WorkQueueFilter = None,\n) -> List[schemas.responses.HistoryResponse]:\n    \"\"\"\n    Produce a history of runs aggregated by interval and state\n    \"\"\"\n\n    # SQLite has issues with very small intervals\n    # (by 0.001 seconds it stops incrementing the interval)\n    if history_interval < datetime.timedelta(seconds=1):\n        raise ValueError(\"History interval must not be less than 1 second.\")\n\n    # prepare run-specific models\n    if run_type == \"flow_run\":\n        run_model = db.FlowRun\n        run_filter_function = models.flow_runs._apply_flow_run_filters\n    elif run_type == \"task_run\":\n        run_model = db.TaskRun\n        run_filter_function = models.task_runs._apply_task_run_filters\n    else:\n        raise ValueError(\n            f\"Unknown run type {run_type!r}. Expected 'flow_run' or 'task_run'.\"\n        )\n\n    # create a CTE for timestamp intervals\n    intervals = db.make_timestamp_intervals(\n        history_start,\n        history_end,\n        history_interval,\n    ).cte(\"intervals\")\n\n    # apply filters to the flow runs (and related states)\n    runs = (\n        await run_filter_function(\n            sa.select(\n                run_model.id,\n                run_model.expected_start_time,\n                run_model.estimated_run_time,\n                run_model.estimated_start_time_delta,\n                run_model.state_type,\n                run_model.state_name,\n            ).select_from(run_model),\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_queues,\n        )\n    ).alias(\"runs\")\n    # outer join intervals to the filtered runs to create a dataset composed of\n    # every interval and the aggregate of all its runs. The runs aggregate is represented\n    # by a descriptive JSON object\n    counts = (\n        sa.select(\n            intervals.c.interval_start,\n            intervals.c.interval_end,\n            # build a JSON object, ignoring the case where the count of runs is 0\n            sa.case(\n                (sa.func.count(runs.c.id) == 0, None),\n                else_=db.build_json_object(\n                    \"state_type\",\n                    runs.c.state_type,\n                    \"state_name\",\n                    runs.c.state_name,\n                    \"count_runs\",\n                    sa.func.count(runs.c.id),\n                    # estimated run times only includes positive run times (to avoid any unexpected corner cases)\n                    \"sum_estimated_run_time\",\n                    sa.func.sum(\n                        db.greatest(0, sa.extract(\"epoch\", runs.c.estimated_run_time))\n                    ),\n                    # estimated lateness is the sum of any positive start time deltas\n                    \"sum_estimated_lateness\",\n                    sa.func.sum(\n                        db.greatest(\n                            0, sa.extract(\"epoch\", runs.c.estimated_start_time_delta)\n                        )\n                    ),\n                ),\n            ).label(\"state_agg\"),\n        )\n        .select_from(intervals)\n        .join(\n            runs,\n            sa.and_(\n                runs.c.expected_start_time >= intervals.c.interval_start,\n                runs.c.expected_start_time < intervals.c.interval_end,\n            ),\n            isouter=True,\n        )\n        .group_by(\n            intervals.c.interval_start,\n            intervals.c.interval_end,\n            runs.c.state_type,\n            runs.c.state_name,\n        )\n    ).alias(\"counts\")\n\n    # aggregate all state-aggregate objects into a single array for each interval,\n    # ensuring that intervals with no runs have an empty array\n    query = (\n        sa.select(\n            counts.c.interval_start,\n            counts.c.interval_end,\n            sa.func.coalesce(\n                db.json_arr_agg(db.cast_to_json(counts.c.state_agg)).filter(\n                    counts.c.state_agg.is_not(None)\n                ),\n                sa.text(\"'[]'\"),\n            ).label(\"states\"),\n        )\n        .group_by(counts.c.interval_start, counts.c.interval_end)\n        .order_by(counts.c.interval_start)\n        # return no more than 500 bars\n        .limit(500)\n    )\n\n    # issue the query\n    result = await session.execute(query)\n    records = result.mappings()\n\n    # load and parse the record if the database returns JSON as strings\n    if db.uses_json_strings:\n        records = [dict(r) for r in records]\n        for r in records:\n            r[\"states\"] = json.loads(r[\"states\"])\n\n    return pydantic.parse_obj_as(List[schemas.responses.HistoryResponse], list(records))\n
    ","tags":["Prefect API","flow runs","task runs","observability"]},{"location":"api-ref/server/api/saved_searches/","title":"server.api.saved_searches","text":"","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches","title":"prefect.server.api.saved_searches","text":"

    Routes for interacting with saved search objects.

    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches.create_saved_search","title":"create_saved_search async","text":"

    Gracefully creates a new saved search from the provided schema.

    If a saved search with the same name already exists, the saved search's fields are replaced.

    Source code in prefect/server/api/saved_searches.py
    @router.put(\"/\")\nasync def create_saved_search(\n    saved_search: schemas.actions.SavedSearchCreate,\n    response: Response,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.SavedSearch:\n    \"\"\"Gracefully creates a new saved search from the provided schema.\n\n    If a saved search with the same name already exists, the saved search's fields are\n    replaced.\n    \"\"\"\n\n    # hydrate the input model into a full model\n    saved_search = schemas.core.SavedSearch(**saved_search.dict())\n\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        model = await models.saved_searches.create_saved_search(\n            session=session, saved_search=saved_search\n        )\n\n    if model.created >= now:\n        response.status_code = status.HTTP_201_CREATED\n\n    return model\n
    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches.delete_saved_search","title":"delete_saved_search async","text":"

    Delete a saved search by id.

    Source code in prefect/server/api/saved_searches.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_saved_search(\n    saved_search_id: UUID = Path(..., description=\"The saved search id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a saved search by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.saved_searches.delete_saved_search(\n            session=session, saved_search_id=saved_search_id\n        )\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Saved search not found\"\n        )\n
    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches.read_saved_search","title":"read_saved_search async","text":"

    Get a saved search by id.

    Source code in prefect/server/api/saved_searches.py
    @router.get(\"/{id}\")\nasync def read_saved_search(\n    saved_search_id: UUID = Path(..., description=\"The saved search id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.SavedSearch:\n    \"\"\"\n    Get a saved search by id.\n    \"\"\"\n    async with db.session_context() as session:\n        saved_search = await models.saved_searches.read_saved_search(\n            session=session, saved_search_id=saved_search_id\n        )\n    if not saved_search:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Saved search not found\"\n        )\n    return saved_search\n
    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches.read_saved_searches","title":"read_saved_searches async","text":"

    Query for saved searches.

    Source code in prefect/server/api/saved_searches.py
    @router.post(\"/filter\")\nasync def read_saved_searches(\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.SavedSearch]:\n    \"\"\"\n    Query for saved searches.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.saved_searches.read_saved_searches(\n            session=session,\n            offset=offset,\n            limit=limit,\n        )\n
    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/server/","title":"server.api.server","text":"","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server","title":"prefect.server.api.server","text":"

    Defines the Prefect REST API FastAPI app.

    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.RequestLimitMiddleware","title":"RequestLimitMiddleware","text":"

    A middleware that limits the number of concurrent requests handled by the API.

    This is a blunt tool for limiting SQLite concurrent writes which will cause failures at high volume. Ideally, we would only apply the limit to routes that perform writes.

    Source code in prefect/server/api/server.py
    class RequestLimitMiddleware:\n    \"\"\"\n    A middleware that limits the number of concurrent requests handled by the API.\n\n    This is a blunt tool for limiting SQLite concurrent writes which will cause failures\n    at high volume. Ideally, we would only apply the limit to routes that perform\n    writes.\n    \"\"\"\n\n    def __init__(self, app, limit: float):\n        self.app = app\n        self._limiter = anyio.CapacityLimiter(limit)\n\n    async def __call__(self, scope, receive, send) -> None:\n        async with self._limiter:\n            await self.app(scope, receive, send)\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.SPAStaticFiles","title":"SPAStaticFiles","text":"

    Bases: StaticFiles

    Implementation of StaticFiles for serving single page applications.

    Adds get_response handling to ensure that when a resource isn't found the application still returns the index.

    Source code in prefect/server/api/server.py
    class SPAStaticFiles(StaticFiles):\n    \"\"\"\n    Implementation of `StaticFiles` for serving single page applications.\n\n    Adds `get_response` handling to ensure that when a resource isn't found the\n    application still returns the index.\n    \"\"\"\n\n    async def get_response(self, path: str, scope):\n        try:\n            return await super().get_response(path, scope)\n        except HTTPException:\n            return await super().get_response(\"./index.html\", scope)\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.create_api_app","title":"create_api_app","text":"

    Create a FastAPI app that includes the Prefect REST API

    Parameters:

    Name Type Description Default router_prefix Optional[str]

    a prefix to apply to all included routers

    '' dependencies Optional[List[Depends]]

    a list of global dependencies to add to each Prefect REST API router

    None health_check_path str

    the health check route path

    '/health' fast_api_app_kwargs dict

    kwargs to pass to the FastAPI constructor

    None router_overrides Mapping[str, Optional[APIRouter]]

    a mapping of route prefixes (i.e. \"/admin\") to new routers allowing the caller to override the default routers. If None is provided as a value, the default router will be dropped from the application.

    None

    Returns:

    Type Description FastAPI

    a FastAPI app that serves the Prefect REST API

    Source code in prefect/server/api/server.py
    def create_api_app(\n    router_prefix: Optional[str] = \"\",\n    dependencies: Optional[List[Depends]] = None,\n    health_check_path: str = \"/health\",\n    version_check_path: str = \"/version\",\n    fast_api_app_kwargs: dict = None,\n    router_overrides: Mapping[str, Optional[APIRouter]] = None,\n) -> FastAPI:\n    \"\"\"\n    Create a FastAPI app that includes the Prefect REST API\n\n    Args:\n        router_prefix: a prefix to apply to all included routers\n        dependencies: a list of global dependencies to add to each Prefect REST API router\n        health_check_path: the health check route path\n        fast_api_app_kwargs: kwargs to pass to the FastAPI constructor\n        router_overrides: a mapping of route prefixes (i.e. \"/admin\") to new routers\n            allowing the caller to override the default routers. If `None` is provided\n            as a value, the default router will be dropped from the application.\n\n    Returns:\n        a FastAPI app that serves the Prefect REST API\n    \"\"\"\n    fast_api_app_kwargs = fast_api_app_kwargs or {}\n    api_app = FastAPI(title=API_TITLE, **fast_api_app_kwargs)\n    api_app.add_middleware(GZipMiddleware)\n\n    @api_app.get(health_check_path, tags=[\"Root\"])\n    async def health_check():\n        return True\n\n    @api_app.get(version_check_path, tags=[\"Root\"])\n    async def orion_info():\n        return SERVER_API_VERSION\n\n    # always include version checking\n    if dependencies is None:\n        dependencies = [Depends(enforce_minimum_version)]\n    else:\n        dependencies.append(Depends(enforce_minimum_version))\n\n    routers = {router.prefix: router for router in API_ROUTERS}\n\n    if router_overrides:\n        for prefix, router in router_overrides.items():\n            # We may want to allow this behavior in the future to inject new routes, but\n            # for now this will be treated an as an exception\n            if prefix not in routers:\n                raise KeyError(\n                    \"Router override provided for prefix that does not exist:\"\n                    f\" {prefix!r}\"\n                )\n\n            # Drop the existing router\n            existing_router = routers.pop(prefix)\n\n            # Replace it with a new router if provided\n            if router is not None:\n                if prefix != router.prefix:\n                    # We may want to allow this behavior in the future, but it will\n                    # break expectations without additional routing and is banned for\n                    # now\n                    raise ValueError(\n                        f\"Router override for {prefix!r} defines a different prefix \"\n                        f\"{router.prefix!r}.\"\n                    )\n\n                existing_paths = method_paths_from_routes(existing_router.routes)\n                new_paths = method_paths_from_routes(router.routes)\n                if not existing_paths.issubset(new_paths):\n                    raise ValueError(\n                        f\"Router override for {prefix!r} is missing paths defined by \"\n                        f\"the original router: {existing_paths.difference(new_paths)}\"\n                    )\n\n                routers[prefix] = router\n\n    for router in routers.values():\n        api_app.include_router(router, prefix=router_prefix, dependencies=dependencies)\n\n    return api_app\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.create_app","title":"create_app","text":"

    Create an FastAPI app that includes the Prefect REST API and UI

    Parameters:

    Name Type Description Default settings Settings

    The settings to use to create the app. If not set, settings are pulled from the context.

    None ignore_cache bool

    If set, a new application will be created even if the settings match. Otherwise, an application is returned from the cache.

    False ephemeral bool

    If set, the application will be treated as ephemeral. The UI and services will be disabled.

    False Source code in prefect/server/api/server.py
    def create_app(\n    settings: prefect.settings.Settings = None,\n    ephemeral: bool = False,\n    ignore_cache: bool = False,\n) -> FastAPI:\n    \"\"\"\n    Create an FastAPI app that includes the Prefect REST API and UI\n\n    Args:\n        settings: The settings to use to create the app. If not set, settings are pulled\n            from the context.\n        ignore_cache: If set, a new application will be created even if the settings\n            match. Otherwise, an application is returned from the cache.\n        ephemeral: If set, the application will be treated as ephemeral. The UI\n            and services will be disabled.\n    \"\"\"\n    settings = settings or prefect.settings.get_current_settings()\n    cache_key = (settings, ephemeral)\n\n    if cache_key in APP_CACHE and not ignore_cache:\n        return APP_CACHE[cache_key]\n\n    # TODO: Move these startup functions out of this closure into the top-level or\n    #       another dedicated location\n    async def run_migrations():\n        \"\"\"Ensure the database is created and up to date with the current migrations\"\"\"\n        if prefect.settings.PREFECT_API_DATABASE_MIGRATE_ON_START:\n            from prefect.server.database.dependencies import provide_database_interface\n\n            db = provide_database_interface()\n            await db.create_db()\n\n    @_memoize_block_auto_registration\n    async def add_block_types():\n        \"\"\"Add all registered blocks to the database\"\"\"\n        if not prefect.settings.PREFECT_API_BLOCKS_REGISTER_ON_START:\n            return\n\n        from prefect.server.database.dependencies import provide_database_interface\n        from prefect.server.models.block_registration import run_block_auto_registration\n\n        db = provide_database_interface()\n        session = await db.session()\n\n        async with session:\n            await run_block_auto_registration(session=session)\n\n    async def start_services():\n        \"\"\"Start additional services when the Prefect REST API starts up.\"\"\"\n\n        if ephemeral:\n            app.state.services = None\n            return\n\n        service_instances = []\n\n        if prefect.settings.PREFECT_API_SERVICES_SCHEDULER_ENABLED.value():\n            service_instances.append(services.scheduler.Scheduler())\n            service_instances.append(services.scheduler.RecentDeploymentsScheduler())\n\n        if prefect.settings.PREFECT_API_SERVICES_LATE_RUNS_ENABLED.value():\n            service_instances.append(services.late_runs.MarkLateRuns())\n\n        if prefect.settings.PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_ENABLED.value():\n            service_instances.append(services.pause_expirations.FailExpiredPauses())\n\n        if prefect.settings.PREFECT_API_SERVICES_CANCELLATION_CLEANUP_ENABLED.value():\n            service_instances.append(\n                services.cancellation_cleanup.CancellationCleanup()\n            )\n\n        if prefect.settings.PREFECT_SERVER_ANALYTICS_ENABLED.value():\n            service_instances.append(services.telemetry.Telemetry())\n\n        if prefect.settings.PREFECT_API_SERVICES_FLOW_RUN_NOTIFICATIONS_ENABLED.value():\n            service_instances.append(\n                services.flow_run_notifications.FlowRunNotifications()\n            )\n\n        loop = asyncio.get_running_loop()\n\n        app.state.services = {\n            service: loop.create_task(service.start()) for service in service_instances\n        }\n\n        for service, task in app.state.services.items():\n            logger.info(f\"{service.name} service scheduled to start in-app\")\n            task.add_done_callback(partial(on_service_exit, service))\n\n    async def stop_services():\n        \"\"\"Ensure services are stopped before the Prefect REST API shuts down.\"\"\"\n        if hasattr(app.state, \"services\") and app.state.services:\n            await asyncio.gather(*[service.stop() for service in app.state.services])\n            try:\n                await asyncio.gather(\n                    *[task.stop() for task in app.state.services.values()]\n                )\n            except Exception:\n                # `on_service_exit` should handle logging exceptions on exit\n                pass\n\n    @asynccontextmanager\n    async def lifespan(app):\n        try:\n            await run_migrations()\n            await add_block_types()\n            await start_services()\n            yield\n        finally:\n            await stop_services()\n\n    def on_service_exit(service, task):\n        \"\"\"\n        Added as a callback for completion of services to log exit\n        \"\"\"\n        try:\n            # Retrieving the result will raise the exception\n            task.result()\n        except Exception:\n            logger.error(f\"{service.name} service failed!\", exc_info=True)\n        else:\n            logger.info(f\"{service.name} service stopped!\")\n\n    app = FastAPI(\n        title=TITLE,\n        version=API_VERSION,\n        lifespan=lifespan,\n    )\n    api_app = create_api_app(\n        fast_api_app_kwargs={\n            \"exception_handlers\": {\n                # NOTE: FastAPI special cases the generic `Exception` handler and\n                #       registers it as a separate middleware from the others\n                Exception: custom_internal_exception_handler,\n                RequestValidationError: validation_exception_handler,\n                sa.exc.IntegrityError: integrity_exception_handler,\n                ObjectNotFoundError: prefect_object_not_found_exception_handler,\n            }\n        },\n    )\n    ui_app = create_ui_app(ephemeral)\n\n    # middleware\n    app.add_middleware(\n        CORSMiddleware,\n        allow_origins=[\"*\"],\n        allow_methods=[\"*\"],\n        allow_headers=[\"*\"],\n    )\n\n    # Limit the number of concurrent requests when using a SQLite database to reduce\n    # chance of errors where the database cannot be opened due to a high number of\n    # concurrent writes\n    if (\n        get_dialect(prefect.settings.PREFECT_API_DATABASE_CONNECTION_URL.value()).name\n        == \"sqlite\"\n    ):\n        app.add_middleware(RequestLimitMiddleware, limit=100)\n\n    api_app.mount(\n        \"/static\",\n        StaticFiles(\n            directory=os.path.join(\n                os.path.dirname(os.path.realpath(__file__)), \"static\"\n            )\n        ),\n        name=\"static\",\n    )\n    app.api_app = api_app\n    app.mount(\"/api\", app=api_app, name=\"api\")\n    app.mount(\"/\", app=ui_app, name=\"ui\")\n\n    def openapi():\n        \"\"\"\n        Convenience method for extracting the user facing OpenAPI schema from the API app.\n\n        This method is attached to the global public app for easy access.\n        \"\"\"\n        partial_schema = get_openapi(\n            title=API_TITLE,\n            version=API_VERSION,\n            routes=api_app.routes,\n        )\n        new_schema = partial_schema.copy()\n        new_schema[\"paths\"] = {}\n        for path, value in partial_schema[\"paths\"].items():\n            new_schema[\"paths\"][f\"/api{path}\"] = value\n\n        new_schema[\"info\"][\"x-logo\"] = {\"url\": \"static/prefect-logo-mark-gradient.png\"}\n        return new_schema\n\n    app.openapi = openapi\n\n    APP_CACHE[cache_key] = app\n    return app\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.custom_internal_exception_handler","title":"custom_internal_exception_handler async","text":"

    Log a detailed exception for internal server errors before returning.

    Send 503 for errors clients can retry on.

    Source code in prefect/server/api/server.py
    async def custom_internal_exception_handler(request: Request, exc: Exception):\n    \"\"\"\n    Log a detailed exception for internal server errors before returning.\n\n    Send 503 for errors clients can retry on.\n    \"\"\"\n    logger.error(\"Encountered exception in request:\", exc_info=True)\n\n    if is_client_retryable_exception(exc):\n        return JSONResponse(\n            content={\"exception_message\": \"Service Unavailable\"},\n            status_code=status.HTTP_503_SERVICE_UNAVAILABLE,\n        )\n\n    return JSONResponse(\n        content={\"exception_message\": \"Internal Server Error\"},\n        status_code=status.HTTP_500_INTERNAL_SERVER_ERROR,\n    )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.integrity_exception_handler","title":"integrity_exception_handler async","text":"

    Capture database integrity errors.

    Source code in prefect/server/api/server.py
    async def integrity_exception_handler(request: Request, exc: Exception):\n    \"\"\"Capture database integrity errors.\"\"\"\n    logger.error(\"Encountered exception in request:\", exc_info=True)\n    return JSONResponse(\n        content={\n            \"detail\": (\n                \"Data integrity conflict. This usually means a \"\n                \"unique or foreign key constraint was violated. \"\n                \"See server logs for details.\"\n            )\n        },\n        status_code=status.HTTP_409_CONFLICT,\n    )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.prefect_object_not_found_exception_handler","title":"prefect_object_not_found_exception_handler async","text":"

    Return 404 status code on object not found exceptions.

    Source code in prefect/server/api/server.py
    async def prefect_object_not_found_exception_handler(\n    request: Request, exc: ObjectNotFoundError\n):\n    \"\"\"Return 404 status code on object not found exceptions.\"\"\"\n    return JSONResponse(\n        content={\"exception_message\": str(exc)}, status_code=status.HTTP_404_NOT_FOUND\n    )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.validation_exception_handler","title":"validation_exception_handler async","text":"

    Provide a detailed message for request validation errors.

    Source code in prefect/server/api/server.py
    async def validation_exception_handler(request: Request, exc: RequestValidationError):\n    \"\"\"Provide a detailed message for request validation errors.\"\"\"\n    return JSONResponse(\n        status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,\n        content=jsonable_encoder(\n            {\n                \"exception_message\": \"Invalid request received.\",\n                \"exception_detail\": exc.errors(),\n                \"request_body\": exc.body,\n            }\n        ),\n    )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/task_run_states/","title":"server.api.task_run_states","text":"","tags":["Prefect API","task runs","states"]},{"location":"api-ref/server/api/task_run_states/#prefect.server.api.task_run_states","title":"prefect.server.api.task_run_states","text":"

    Routes for interacting with task run state objects.

    ","tags":["Prefect API","task runs","states"]},{"location":"api-ref/server/api/task_run_states/#prefect.server.api.task_run_states.read_task_run_state","title":"read_task_run_state async","text":"

    Get a task run state by id.

    Source code in prefect/server/api/task_run_states.py
    @router.get(\"/{id}\")\nasync def read_task_run_state(\n    task_run_state_id: UUID = Path(\n        ..., description=\"The task run state id\", alias=\"id\"\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.states.State:\n    \"\"\"\n    Get a task run state by id.\n    \"\"\"\n    async with db.session_context() as session:\n        task_run_state = await models.task_run_states.read_task_run_state(\n            session=session, task_run_state_id=task_run_state_id\n        )\n    if not task_run_state:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run state not found\"\n        )\n    return task_run_state\n
    ","tags":["Prefect API","task runs","states"]},{"location":"api-ref/server/api/task_run_states/#prefect.server.api.task_run_states.read_task_run_states","title":"read_task_run_states async","text":"

    Get states associated with a task run.

    Source code in prefect/server/api/task_run_states.py
    @router.get(\"/\")\nasync def read_task_run_states(\n    task_run_id: UUID,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.states.State]:\n    \"\"\"\n    Get states associated with a task run.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.task_run_states.read_task_run_states(\n            session=session, task_run_id=task_run_id\n        )\n
    ","tags":["Prefect API","task runs","states"]},{"location":"api-ref/server/api/task_runs/","title":"server.api.task_runs","text":"","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs","title":"prefect.server.api.task_runs","text":"

    Routes for interacting with task run objects.

    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.count_task_runs","title":"count_task_runs async","text":"

    Count task runs.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/count\")\nasync def count_task_runs(\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n) -> int:\n    \"\"\"\n    Count task runs.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.task_runs.count_task_runs(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n        )\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.create_task_run","title":"create_task_run async","text":"

    Create a task run. If a task run with the same flow_run_id, task_key, and dynamic_key already exists, the existing task run will be returned.

    If no state is provided, the task run will be created in a PENDING state.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/\")\nasync def create_task_run(\n    task_run: schemas.actions.TaskRunCreate,\n    response: Response,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_task_orchestration_parameters\n    ),\n) -> schemas.core.TaskRun:\n    \"\"\"\n    Create a task run. If a task run with the same flow_run_id,\n    task_key, and dynamic_key already exists, the existing task\n    run will be returned.\n\n    If no state is provided, the task run will be created in a PENDING state.\n    \"\"\"\n    # hydrate the input model into a full task run / state model\n    task_run = schemas.core.TaskRun(**task_run.dict())\n\n    if not task_run.state:\n        task_run.state = schemas.states.Pending()\n\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        model = await models.task_runs.create_task_run(\n            session=session,\n            task_run=task_run,\n            orchestration_parameters=orchestration_parameters,\n        )\n\n    if model.created >= now:\n        response.status_code = status.HTTP_201_CREATED\n    return model\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.delete_task_run","title":"delete_task_run async","text":"

    Delete a task run by id.

    Source code in prefect/server/api/task_runs.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_task_run(\n    task_run_id: UUID = Path(..., description=\"The task run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a task run by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.task_runs.delete_task_run(\n            session=session, task_run_id=task_run_id\n        )\n    if not result:\n        raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Task not found\")\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.read_task_run","title":"read_task_run async","text":"

    Get a task run by id.

    Source code in prefect/server/api/task_runs.py
    @router.get(\"/{id}\")\nasync def read_task_run(\n    task_run_id: UUID = Path(..., description=\"The task run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.TaskRun:\n    \"\"\"\n    Get a task run by id.\n    \"\"\"\n    async with db.session_context() as session:\n        task_run = await models.task_runs.read_task_run(\n            session=session, task_run_id=task_run_id\n        )\n    if not task_run:\n        raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Task not found\")\n    return task_run\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.read_task_runs","title":"read_task_runs async","text":"

    Query for task runs.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/filter\")\nasync def read_task_runs(\n    sort: schemas.sorting.TaskRunSort = Body(schemas.sorting.TaskRunSort.ID_DESC),\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.TaskRun]:\n    \"\"\"\n    Query for task runs.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.task_runs.read_task_runs(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            offset=offset,\n            limit=limit,\n            sort=sort,\n        )\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.set_task_run_state","title":"set_task_run_state async","text":"

    Set a task run state, invoking any orchestration rules.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/{id}/set_state\")\nasync def set_task_run_state(\n    task_run_id: UUID = Path(..., description=\"The task run id\", alias=\"id\"),\n    state: schemas.actions.StateCreate = Body(..., description=\"The intended state.\"),\n    force: bool = Body(\n        False,\n        description=(\n            \"If false, orchestration rules will be applied that may alter or prevent\"\n            \" the state transition. If True, orchestration rules are not applied.\"\n        ),\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    response: Response = None,\n    task_policy: BaseOrchestrationPolicy = Depends(\n        orchestration_dependencies.provide_task_policy\n    ),\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_task_orchestration_parameters\n    ),\n) -> OrchestrationResult:\n    \"\"\"Set a task run state, invoking any orchestration rules.\"\"\"\n\n    now = pendulum.now(\"UTC\")\n\n    # create the state\n    async with db.session_context(\n        begin_transaction=True, with_for_update=True\n    ) as session:\n        orchestration_result = await models.task_runs.set_task_run_state(\n            session=session,\n            task_run_id=task_run_id,\n            state=schemas.states.State.parse_obj(\n                state\n            ),  # convert to a full State object\n            force=force,\n            task_policy=task_policy,\n            orchestration_parameters=orchestration_parameters,\n        )\n\n    # set the 201 if a new state was created\n    if orchestration_result.state and orchestration_result.state.timestamp >= now:\n        response.status_code = status.HTTP_201_CREATED\n    else:\n        response.status_code = status.HTTP_200_OK\n\n    return orchestration_result\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.task_run_history","title":"task_run_history async","text":"

    Query for task run history data across a given range and interval.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/history\")\nasync def task_run_history(\n    history_start: DateTimeTZ = Body(..., description=\"The history's start time.\"),\n    history_end: DateTimeTZ = Body(..., description=\"The history's end time.\"),\n    history_interval: datetime.timedelta = Body(\n        ...,\n        description=(\n            \"The size of each history interval, in seconds. Must be at least 1 second.\"\n        ),\n        alias=\"history_interval_seconds\",\n    ),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.HistoryResponse]:\n    \"\"\"\n    Query for task run history data across a given range and interval.\n    \"\"\"\n    if history_interval < datetime.timedelta(seconds=1):\n        raise HTTPException(\n            status.HTTP_422_UNPROCESSABLE_ENTITY,\n            detail=\"History interval must not be less than 1 second.\",\n        )\n\n    async with db.session_context() as session:\n        return await run_history(\n            session=session,\n            run_type=\"task_run\",\n            history_start=history_start,\n            history_end=history_end,\n            history_interval=history_interval,\n            flows=flows,\n            flow_runs=flow_runs,\n            task_runs=task_runs,\n            deployments=deployments,\n        )\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.update_task_run","title":"update_task_run async","text":"

    Updates a task run.

    Source code in prefect/server/api/task_runs.py
    @router.patch(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def update_task_run(\n    task_run: schemas.actions.TaskRunUpdate,\n    task_run_id: UUID = Path(..., description=\"The task run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Updates a task run.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.task_runs.update_task_run(\n            session=session, task_run=task_run, task_run_id=task_run_id\n        )\n    if not result:\n        raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Task run not found\")\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/models/deployments/","title":"server.models.deployments","text":""},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments","title":"prefect.server.models.deployments","text":"

    Functions for interacting with deployment ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.check_work_queues_for_deployment","title":"check_work_queues_for_deployment async","text":"

    Get work queues that can pick up the specified deployment.

    Work queues will pick up a deployment when all of the following are met.

    • The deployment has ALL tags that the work queue has (i.e. the work queue's tags must be a subset of the deployment's tags).
    • The work queue's specified deployment IDs match the deployment's ID, or the work queue does NOT have specified deployment IDs.
    • The work queue's specified flow runners match the deployment's flow runner or the work queue does NOT have a specified flow runner.

    Notes on the query:

    • Our database currently allows either \"null\" and empty lists as null values in filters, so we need to catch both cases with \"or\".
    • json_contains(A, B) should be interpreted as \"True if A contains B\".

    Returns:

    Type Description List[WorkQueue]

    List[db.WorkQueue]: WorkQueues

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def check_work_queues_for_deployment(\n    db: PrefectDBInterface, session: sa.orm.Session, deployment_id: UUID\n) -> List[schemas.core.WorkQueue]:\n    \"\"\"\n    Get work queues that can pick up the specified deployment.\n\n    Work queues will pick up a deployment when all of the following are met.\n\n    - The deployment has ALL tags that the work queue has (i.e. the work\n    queue's tags must be a subset of the deployment's tags).\n    - The work queue's specified deployment IDs match the deployment's ID,\n    or the work queue does NOT have specified deployment IDs.\n    - The work queue's specified flow runners match the deployment's flow\n    runner or the work queue does NOT have a specified flow runner.\n\n    Notes on the query:\n\n    - Our database currently allows either \"null\" and empty lists as\n    null values in filters, so we need to catch both cases with \"or\".\n    - `json_contains(A, B)` should be interpreted as \"True if A\n    contains B\".\n\n    Returns:\n        List[db.WorkQueue]: WorkQueues\n    \"\"\"\n    deployment = await session.get(db.Deployment, deployment_id)\n    if not deployment:\n        raise ObjectNotFoundError(f\"Deployment with id {deployment_id} not found\")\n\n    query = (\n        select(db.WorkQueue)\n        # work queue tags are a subset of deployment tags\n        .filter(\n            or_(\n                json_contains(deployment.tags, db.WorkQueue.filter[\"tags\"]),\n                json_contains([], db.WorkQueue.filter[\"tags\"]),\n                json_contains(None, db.WorkQueue.filter[\"tags\"]),\n            )\n        )\n        # deployment_ids is null or contains the deployment's ID\n        .filter(\n            or_(\n                json_contains(\n                    db.WorkQueue.filter[\"deployment_ids\"],\n                    str(deployment.id),\n                ),\n                json_contains(None, db.WorkQueue.filter[\"deployment_ids\"]),\n                json_contains([], db.WorkQueue.filter[\"deployment_ids\"]),\n            )\n        )\n    )\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.count_deployments","title":"count_deployments async","text":"

    Count deployments.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_filter FlowFilter

    only count deployments whose flows match these criteria

    None flow_run_filter FlowRunFilter

    only count deployments whose flow runs match these criteria

    None task_run_filter TaskRunFilter

    only count deployments whose task runs match these criteria

    None deployment_filter DeploymentFilter

    only count deployment that match these filters

    None work_pool_filter WorkPoolFilter

    only count deployments that match these work pool filters

    None work_queue_filter WorkQueueFilter

    only count deployments that match these work pool queue filters

    None

    Returns:

    Name Type Description int int

    the number of deployments matching filters

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def count_deployments(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    work_queue_filter: schemas.filters.WorkQueueFilter = None,\n) -> int:\n    \"\"\"\n    Count deployments.\n\n    Args:\n        session: A database session\n        flow_filter: only count deployments whose flows match these criteria\n        flow_run_filter: only count deployments whose flow runs match these criteria\n        task_run_filter: only count deployments whose task runs match these criteria\n        deployment_filter: only count deployment that match these filters\n        work_pool_filter: only count deployments that match these work pool filters\n        work_queue_filter: only count deployments that match these work pool queue filters\n\n    Returns:\n        int: the number of deployments matching filters\n    \"\"\"\n\n    query = select(sa.func.count(sa.text(\"*\"))).select_from(db.Deployment)\n\n    query = await _apply_deployment_filters(\n        query=query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        work_queue_filter=work_queue_filter,\n        db=db,\n    )\n\n    result = await session.execute(query)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.create_deployment","title":"create_deployment async","text":"

    Upserts a deployment.

    Parameters:

    Name Type Description Default session Session

    a database session

    required deployment Deployment

    a deployment model

    required

    Returns:

    Type Description

    db.Deployment: the newly-created or updated deployment

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def create_deployment(\n    session: sa.orm.Session, deployment: schemas.core.Deployment, db: PrefectDBInterface\n):\n    \"\"\"Upserts a deployment.\n\n    Args:\n        session: a database session\n        deployment: a deployment model\n\n    Returns:\n        db.Deployment: the newly-created or updated deployment\n\n    \"\"\"\n\n    # set `updated` manually\n    # known limitation of `on_conflict_do_update`, will not use `Column.onupdate`\n    # https://docs.sqlalchemy.org/en/14/dialects/sqlite.html#the-set-clause\n    deployment.updated = pendulum.now(\"UTC\")\n\n    insert_values = deployment.dict(shallow=True, exclude_unset=True)\n\n    insert_stmt = (\n        (await db.insert(db.Deployment))\n        .values(**insert_values)\n        .on_conflict_do_update(\n            index_elements=db.deployment_unique_upsert_columns,\n            set_={\n                **deployment.dict(\n                    shallow=True,\n                    exclude_unset=True,\n                    exclude={\"id\", \"created\", \"created_by\"},\n                ),\n            },\n        )\n    )\n\n    await session.execute(insert_stmt)\n\n    query = (\n        sa.select(db.Deployment)\n        .where(\n            sa.and_(\n                db.Deployment.flow_id == deployment.flow_id,\n                db.Deployment.name == deployment.name,\n            )\n        )\n        .execution_options(populate_existing=True)\n    )\n    result = await session.execute(query)\n    model = result.scalar()\n\n    # because this could upsert a different schedule, delete any runs from the old\n    # deployment\n    await _delete_scheduled_runs(\n        session=session, deployment_id=model.id, db=db, auto_scheduled_only=True\n    )\n\n    return model\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.delete_deployment","title":"delete_deployment async","text":"

    Delete a deployment by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required deployment_id UUID

    a deployment id

    required

    Returns:

    Name Type Description bool bool

    whether or not the deployment was deleted

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def delete_deployment(\n    session: sa.orm.Session, deployment_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a deployment by id.\n\n    Args:\n        session: A database session\n        deployment_id: a deployment id\n\n    Returns:\n        bool: whether or not the deployment was deleted\n    \"\"\"\n\n    # delete scheduled runs, both auto- and user- created.\n    await _delete_scheduled_runs(\n        session=session, deployment_id=deployment_id, auto_scheduled_only=False\n    )\n\n    result = await session.execute(\n        delete(db.Deployment).where(db.Deployment.id == deployment_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.read_deployment","title":"read_deployment async","text":"

    Reads a deployment by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required deployment_id UUID

    a deployment id

    required

    Returns:

    Type Description

    db.Deployment: the deployment

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def read_deployment(\n    session: sa.orm.Session, deployment_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"Reads a deployment by id.\n\n    Args:\n        session: A database session\n        deployment_id: a deployment id\n\n    Returns:\n        db.Deployment: the deployment\n    \"\"\"\n\n    return await session.get(db.Deployment, deployment_id)\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.read_deployment_by_name","title":"read_deployment_by_name async","text":"

    Reads a deployment by name.

    Parameters:

    Name Type Description Default session Session

    A database session

    required name str

    a deployment name

    required flow_name str

    the name of the flow the deployment belongs to

    required

    Returns:

    Type Description

    db.Deployment: the deployment

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def read_deployment_by_name(\n    session: sa.orm.Session, name: str, flow_name: str, db: PrefectDBInterface\n):\n    \"\"\"Reads a deployment by name.\n\n    Args:\n        session: A database session\n        name: a deployment name\n        flow_name: the name of the flow the deployment belongs to\n\n    Returns:\n        db.Deployment: the deployment\n    \"\"\"\n\n    result = await session.execute(\n        select(db.Deployment)\n        .join(db.Flow, db.Deployment.flow_id == db.Flow.id)\n        .where(\n            sa.and_(\n                db.Flow.name == flow_name,\n                db.Deployment.name == name,\n            )\n        )\n        .limit(1)\n    )\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.read_deployments","title":"read_deployments async","text":"

    Read deployments.

    Parameters:

    Name Type Description Default session Session

    A database session

    required offset int

    Query offset

    None limit int

    Query limit

    None flow_filter FlowFilter

    only select deployments whose flows match these criteria

    None flow_run_filter FlowRunFilter

    only select deployments whose flow runs match these criteria

    None task_run_filter TaskRunFilter

    only select deployments whose task runs match these criteria

    None deployment_filter DeploymentFilter

    only select deployment that match these filters

    None work_pool_filter WorkPoolFilter

    only select deployments whose work pools match these criteria

    None work_queue_filter WorkQueueFilter

    only select deployments whose work pool queues match these criteria

    None sort DeploymentSort

    the sort criteria for selected deployments. Defaults to name ASC.

    NAME_ASC

    Returns:

    Type Description

    List[db.Deployment]: deployments

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def read_deployments(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    offset: int = None,\n    limit: int = None,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    work_queue_filter: schemas.filters.WorkQueueFilter = None,\n    sort: schemas.sorting.DeploymentSort = schemas.sorting.DeploymentSort.NAME_ASC,\n):\n    \"\"\"\n    Read deployments.\n\n    Args:\n        session: A database session\n        offset: Query offset\n        limit: Query limit\n        flow_filter: only select deployments whose flows match these criteria\n        flow_run_filter: only select deployments whose flow runs match these criteria\n        task_run_filter: only select deployments whose task runs match these criteria\n        deployment_filter: only select deployment that match these filters\n        work_pool_filter: only select deployments whose work pools match these criteria\n        work_queue_filter: only select deployments whose work pool queues match these criteria\n        sort: the sort criteria for selected deployments. Defaults to `name` ASC.\n\n    Returns:\n        List[db.Deployment]: deployments\n    \"\"\"\n\n    query = select(db.Deployment).order_by(sort.as_sql_sort(db=db))\n\n    query = await _apply_deployment_filters(\n        query=query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        work_queue_filter=work_queue_filter,\n        db=db,\n    )\n\n    if offset is not None:\n        query = query.offset(offset)\n    if limit is not None:\n        query = query.limit(limit)\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.schedule_runs","title":"schedule_runs async","text":"

    Schedule flow runs for a deployment

    Parameters:

    Name Type Description Default session Session

    a database session

    required deployment_id UUID

    the id of the deployment to schedule

    required start_time datetime

    the time from which to start scheduling runs

    None end_time datetime

    runs will be scheduled until at most this time

    None min_time timedelta

    runs will be scheduled until at least this far in the future

    None min_runs int

    a minimum amount of runs to schedule

    None max_runs int

    a maximum amount of runs to schedule

    None

    This function will generate the minimum number of runs that satisfy the min and max times, and the min and max counts. Specifically, the following order will be respected.

    - Runs will be generated starting on or after the `start_time`\n- No more than `max_runs` runs will be generated\n- No runs will be generated after `end_time` is reached\n- At least `min_runs` runs will be generated\n- Runs will be generated until at least `start_time` + `min_time` is reached\n

    Returns:

    Type Description List[UUID]

    a list of flow run ids scheduled for the deployment

    Source code in prefect/server/models/deployments.py
    async def schedule_runs(\n    session: sa.orm.Session,\n    deployment_id: UUID,\n    start_time: datetime.datetime = None,\n    end_time: datetime.datetime = None,\n    min_time: datetime.timedelta = None,\n    min_runs: int = None,\n    max_runs: int = None,\n    auto_scheduled: bool = True,\n) -> List[UUID]:\n    \"\"\"\n    Schedule flow runs for a deployment\n\n    Args:\n        session: a database session\n        deployment_id: the id of the deployment to schedule\n        start_time: the time from which to start scheduling runs\n        end_time: runs will be scheduled until at most this time\n        min_time: runs will be scheduled until at least this far in the future\n        min_runs: a minimum amount of runs to schedule\n        max_runs: a maximum amount of runs to schedule\n\n    This function will generate the minimum number of runs that satisfy the min\n    and max times, and the min and max counts. Specifically, the following order\n    will be respected.\n\n        - Runs will be generated starting on or after the `start_time`\n        - No more than `max_runs` runs will be generated\n        - No runs will be generated after `end_time` is reached\n        - At least `min_runs` runs will be generated\n        - Runs will be generated until at least `start_time` + `min_time` is reached\n\n    Returns:\n        a list of flow run ids scheduled for the deployment\n    \"\"\"\n    if min_runs is None:\n        min_runs = PREFECT_API_SERVICES_SCHEDULER_MIN_RUNS.value()\n    if max_runs is None:\n        max_runs = PREFECT_API_SERVICES_SCHEDULER_MAX_RUNS.value()\n    if start_time is None:\n        start_time = pendulum.now(\"UTC\")\n    if end_time is None:\n        end_time = start_time + (\n            PREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME.value()\n        )\n    if min_time is None:\n        min_time = PREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME.value()\n\n    start_time = pendulum.instance(start_time)\n    end_time = pendulum.instance(end_time)\n\n    runs = await _generate_scheduled_flow_runs(\n        session=session,\n        deployment_id=deployment_id,\n        start_time=start_time,\n        end_time=end_time,\n        min_time=min_time,\n        min_runs=min_runs,\n        max_runs=max_runs,\n        auto_scheduled=auto_scheduled,\n    )\n    return await _insert_scheduled_flow_runs(session=session, runs=runs)\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.update_deployment","title":"update_deployment async","text":"

    Updates a deployment.

    Parameters:

    Name Type Description Default session Session

    a database session

    required deployment_id UUID

    the ID of the deployment to modify

    required deployment DeploymentUpdate

    changes to a deployment model

    required

    Returns:

    Name Type Description bool bool

    whether the deployment was updated

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def update_deployment(\n    session: sa.orm.Session,\n    deployment_id: UUID,\n    deployment: schemas.actions.DeploymentUpdate,\n    db: PrefectDBInterface,\n) -> bool:\n    \"\"\"Updates a deployment.\n\n    Args:\n        session: a database session\n        deployment_id: the ID of the deployment to modify\n        deployment: changes to a deployment model\n\n    Returns:\n        bool: whether the deployment was updated\n\n    \"\"\"\n\n    # exclude_unset=True allows us to only update values provided by\n    # the user, ignoring any defaults on the model\n    update_data = deployment.dict(\n        shallow=True,\n        exclude_unset=True,\n        exclude={\"work_pool_name\"},\n    )\n    if deployment.work_pool_name and deployment.work_queue_name:\n        # If a specific pool name/queue name combination was provided, get the\n        # ID for that work pool queue.\n        update_data[\n            \"work_queue_id\"\n        ] = await WorkerLookups()._get_work_queue_id_from_name(\n            session=session,\n            work_pool_name=deployment.work_pool_name,\n            work_queue_name=deployment.work_queue_name,\n            create_queue_if_not_found=True,\n        )\n    elif deployment.work_pool_name:\n        # If just a pool name was provided, get the ID for its default\n        # work pool queue.\n        update_data[\n            \"work_queue_id\"\n        ] = await WorkerLookups()._get_default_work_queue_id_from_work_pool_name(\n            session=session,\n            work_pool_name=deployment.work_pool_name,\n        )\n    elif deployment.work_queue_name:\n        # If just a queue name was provided, ensure the queue exists and\n        # get its ID.\n        work_queue = await models.work_queues._ensure_work_queue_exists(\n            session=session, name=update_data[\"work_queue_name\"], db=db\n        )\n        update_data[\"work_queue_id\"] = work_queue.id\n\n    update_stmt = (\n        sa.update(db.Deployment)\n        .where(db.Deployment.id == deployment_id)\n        .values(**update_data)\n    )\n    result = await session.execute(update_stmt)\n\n    # delete any auto scheduled runs that would have reflected the old deployment config\n    await _delete_scheduled_runs(\n        session=session, deployment_id=deployment_id, db=db, auto_scheduled_only=True\n    )\n\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flow_run_states/","title":"server.models.flow_run_states","text":""},{"location":"api-ref/server/models/flow_run_states/#prefect.server.models.flow_run_states","title":"prefect.server.models.flow_run_states","text":"

    Functions for interacting with flow run state ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/flow_run_states/#prefect.server.models.flow_run_states.delete_flow_run_state","title":"delete_flow_run_state async","text":"

    Delete a flow run state by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_run_state_id UUID

    a flow run state id

    required

    Returns:

    Name Type Description bool bool

    whether or not the flow run state was deleted

    Source code in prefect/server/models/flow_run_states.py
    @inject_db\nasync def delete_flow_run_state(\n    session: sa.orm.Session, flow_run_state_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a flow run state by id.\n\n    Args:\n        session: A database session\n        flow_run_state_id: a flow run state id\n\n    Returns:\n        bool: whether or not the flow run state was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.FlowRunState).where(db.FlowRunState.id == flow_run_state_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flow_run_states/#prefect.server.models.flow_run_states.read_flow_run_state","title":"read_flow_run_state async","text":"

    Reads a flow run state by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_run_state_id UUID

    a flow run state id

    required

    Returns:

    Type Description

    db.FlowRunState: the flow state

    Source code in prefect/server/models/flow_run_states.py
    @inject_db\nasync def read_flow_run_state(\n    session: sa.orm.Session, flow_run_state_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads a flow run state by id.\n\n    Args:\n        session: A database session\n        flow_run_state_id: a flow run state id\n\n    Returns:\n        db.FlowRunState: the flow state\n    \"\"\"\n\n    return await session.get(db.FlowRunState, flow_run_state_id)\n
    "},{"location":"api-ref/server/models/flow_run_states/#prefect.server.models.flow_run_states.read_flow_run_states","title":"read_flow_run_states async","text":"

    Reads flow runs states for a flow run.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_run_id UUID

    the flow run id

    required

    Returns:

    Type Description

    List[db.FlowRunState]: the flow run states

    Source code in prefect/server/models/flow_run_states.py
    @inject_db\nasync def read_flow_run_states(\n    session: sa.orm.Session, flow_run_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads flow runs states for a flow run.\n\n    Args:\n        session: A database session\n        flow_run_id: the flow run id\n\n    Returns:\n        List[db.FlowRunState]: the flow run states\n    \"\"\"\n\n    query = (\n        select(db.FlowRunState)\n        .filter_by(flow_run_id=flow_run_id)\n        .order_by(db.FlowRunState.timestamp)\n    )\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/flow_runs/","title":"server.models.flow_runs","text":""},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs","title":"prefect.server.models.flow_runs","text":"

    Functions for interacting with flow run ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.count_flow_runs","title":"count_flow_runs async","text":"

    Count flow runs.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required flow_filter FlowFilter

    only count flow runs whose flows match these filters

    None flow_run_filter FlowRunFilter

    only count flow runs that match these filters

    None task_run_filter TaskRunFilter

    only count flow runs whose task runs match these filters

    None deployment_filter DeploymentFilter

    only count flow runs whose deployments match these filters

    None

    Returns:

    Name Type Description int int

    count of flow runs

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def count_flow_runs(\n    session: AsyncSession,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    work_queue_filter: schemas.filters.WorkQueueFilter = None,\n) -> int:\n    \"\"\"\n    Count flow runs.\n\n    Args:\n        session: a database session\n        flow_filter: only count flow runs whose flows match these filters\n        flow_run_filter: only count flow runs that match these filters\n        task_run_filter: only count flow runs whose task runs match these filters\n        deployment_filter: only count flow runs whose deployments match these filters\n\n    Returns:\n        int: count of flow runs\n    \"\"\"\n\n    query = select(sa.func.count(sa.text(\"*\"))).select_from(db.FlowRun)\n\n    query = await _apply_flow_run_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        work_queue_filter=work_queue_filter,\n        db=db,\n    )\n\n    result = await session.execute(query)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.create_flow_run","title":"create_flow_run async","text":"

    Creates a new flow run.

    If the provided flow run has a state attached, it will also be created.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required flow_run FlowRun

    a flow run model

    required

    Returns:

    Type Description

    db.FlowRun: the newly-created flow run

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def create_flow_run(\n    session: AsyncSession,\n    flow_run: schemas.core.FlowRun,\n    db: PrefectDBInterface,\n    orchestration_parameters: Optional[dict] = None,\n):\n    \"\"\"Creates a new flow run.\n\n    If the provided flow run has a state attached, it will also be created.\n\n    Args:\n        session: a database session\n        flow_run: a flow run model\n\n    Returns:\n        db.FlowRun: the newly-created flow run\n    \"\"\"\n    now = pendulum.now(\"UTC\")\n\n    flow_run_dict = dict(\n        **flow_run.dict(\n            shallow=True,\n            exclude={\n                \"created\",\n                \"state\",\n                \"estimated_run_time\",\n                \"estimated_start_time_delta\",\n            },\n            exclude_unset=True,\n        ),\n        created=now,\n    )\n\n    # if no idempotency key was provided, create the run directly\n    if not flow_run.idempotency_key:\n        model = db.FlowRun(**flow_run_dict)\n        session.add(model)\n        await session.flush()\n\n    # otherwise let the database take care of enforcing idempotency\n    else:\n        insert_stmt = (\n            (await db.insert(db.FlowRun))\n            .values(**flow_run_dict)\n            .on_conflict_do_nothing(\n                index_elements=db.flow_run_unique_upsert_columns,\n            )\n        )\n        await session.execute(insert_stmt)\n\n        # read the run to see if idempotency was applied or not\n        query = (\n            sa.select(db.FlowRun)\n            .where(\n                sa.and_(\n                    db.FlowRun.flow_id == flow_run.flow_id,\n                    db.FlowRun.idempotency_key == flow_run.idempotency_key,\n                )\n            )\n            .limit(1)\n            .execution_options(populate_existing=True)\n            .options(\n                selectinload(db.FlowRun.work_queue).selectinload(db.WorkQueue.work_pool)\n            )\n        )\n        result = await session.execute(query)\n        model = result.scalar()\n\n    # if the flow run was created in this function call then we need to set the\n    # state. If it was created idempotently, the created time won't match.\n    if model.created == now and flow_run.state:\n        await models.flow_runs.set_flow_run_state(\n            session=session,\n            flow_run_id=model.id,\n            state=flow_run.state,\n            force=True,\n            orchestration_parameters=orchestration_parameters,\n        )\n    return model\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.delete_flow_run","title":"delete_flow_run async","text":"

    Delete a flow run by flow_run_id.

    Parameters:

    Name Type Description Default session AsyncSession

    A database session

    required flow_run_id UUID

    a flow run id

    required

    Returns:

    Name Type Description bool bool

    whether or not the flow run was deleted

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def delete_flow_run(\n    session: AsyncSession, flow_run_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a flow run by flow_run_id.\n\n    Args:\n        session: A database session\n        flow_run_id: a flow run id\n\n    Returns:\n        bool: whether or not the flow run was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.FlowRun).where(db.FlowRun.id == flow_run_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.read_flow_run","title":"read_flow_run async","text":"

    Reads a flow run by id.

    Parameters:

    Name Type Description Default session AsyncSession

    A database session

    required flow_run_id UUID

    a flow run id

    required

    Returns:

    Type Description

    db.FlowRun: the flow run

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def read_flow_run(\n    session: AsyncSession,\n    flow_run_id: UUID,\n    db: PrefectDBInterface,\n    for_update: bool = False,\n):\n    \"\"\"\n    Reads a flow run by id.\n\n    Args:\n        session: A database session\n        flow_run_id: a flow run id\n\n    Returns:\n        db.FlowRun: the flow run\n    \"\"\"\n    select = (\n        sa.select(db.FlowRun)\n        .where(db.FlowRun.id == flow_run_id)\n        .options(\n            selectinload(db.FlowRun.work_queue).selectinload(db.WorkQueue.work_pool)\n        )\n    )\n\n    if for_update:\n        select = select.with_for_update()\n\n    result = await session.execute(select)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.read_flow_run_graph","title":"read_flow_run_graph async","text":"

    Given a flow run, return the graph of it's task and subflow runs. If a since datetime is provided, only return items that may have changed since that time.

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def read_flow_run_graph(\n    db: PrefectDBInterface,\n    session: AsyncSession,\n    flow_run_id: UUID,\n    since: datetime.datetime = datetime.datetime.min,\n) -> Graph:\n    \"\"\"Given a flow run, return the graph of it's task and subflow runs. If a `since`\n    datetime is provided, only return items that may have changed since that time.\"\"\"\n    return await db.queries.flow_run_graph_v2(\n        db=db,\n        session=session,\n        flow_run_id=flow_run_id,\n        since=since,\n        max_nodes=PREFECT_API_MAX_FLOW_RUN_GRAPH_NODES.value(),\n    )\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.read_flow_runs","title":"read_flow_runs async","text":"

    Read flow runs.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required columns List

    a list of the flow run ORM columns to load, for performance

    None flow_filter FlowFilter

    only select flow runs whose flows match these filters

    None flow_run_filter FlowRunFilter

    only select flow runs match these filters

    None task_run_filter TaskRunFilter

    only select flow runs whose task runs match these filters

    None deployment_filter DeploymentFilter

    only select flow runs whose deployments match these filters

    None offset int

    Query offset

    None limit int

    Query limit

    None sort FlowRunSort

    Query sort

    ID_DESC

    Returns:

    Type Description

    List[db.FlowRun]: flow runs

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def read_flow_runs(\n    session: AsyncSession,\n    db: PrefectDBInterface,\n    columns: List = None,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    work_queue_filter: schemas.filters.WorkQueueFilter = None,\n    offset: int = None,\n    limit: int = None,\n    sort: schemas.sorting.FlowRunSort = schemas.sorting.FlowRunSort.ID_DESC,\n):\n    \"\"\"\n    Read flow runs.\n\n    Args:\n        session: a database session\n        columns: a list of the flow run ORM columns to load, for performance\n        flow_filter: only select flow runs whose flows match these filters\n        flow_run_filter: only select flow runs match these filters\n        task_run_filter: only select flow runs whose task runs match these filters\n        deployment_filter: only select flow runs whose deployments match these filters\n        offset: Query offset\n        limit: Query limit\n        sort: Query sort\n\n    Returns:\n        List[db.FlowRun]: flow runs\n    \"\"\"\n    query = (\n        select(db.FlowRun)\n        .order_by(sort.as_sql_sort(db))\n        .options(\n            selectinload(db.FlowRun.work_queue).selectinload(db.WorkQueue.work_pool)\n        )\n    )\n\n    if columns:\n        query = query.options(load_only(*columns))\n\n    query = await _apply_flow_run_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        work_queue_filter=work_queue_filter,\n        db=db,\n    )\n\n    if offset is not None:\n        query = query.offset(offset)\n\n    if limit is not None:\n        query = query.limit(limit)\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.read_task_run_dependencies","title":"read_task_run_dependencies async","text":"

    Get a task run dependency map for a given flow run.

    Source code in prefect/server/models/flow_runs.py
    async def read_task_run_dependencies(\n    session: AsyncSession,\n    flow_run_id: UUID,\n) -> List[DependencyResult]:\n    \"\"\"\n    Get a task run dependency map for a given flow run.\n    \"\"\"\n    flow_run = await models.flow_runs.read_flow_run(\n        session=session, flow_run_id=flow_run_id\n    )\n    if not flow_run:\n        raise ObjectNotFoundError(f\"Flow run with id {flow_run_id} not found\")\n\n    task_runs = await models.task_runs.read_task_runs(\n        session=session,\n        flow_run_filter=schemas.filters.FlowRunFilter(\n            id=schemas.filters.FlowRunFilterId(any_=[flow_run_id])\n        ),\n    )\n\n    dependency_graph = []\n\n    for task_run in task_runs:\n        inputs = list(set(chain(*task_run.task_inputs.values())))\n        untrackable_result_status = (\n            False\n            if task_run.state is None\n            else task_run.state.state_details.untrackable_result\n        )\n        dependency_graph.append(\n            {\n                \"id\": task_run.id,\n                \"upstream_dependencies\": inputs,\n                \"state\": task_run.state,\n                \"expected_start_time\": task_run.expected_start_time,\n                \"name\": task_run.name,\n                \"start_time\": task_run.start_time,\n                \"end_time\": task_run.end_time,\n                \"total_run_time\": task_run.total_run_time,\n                \"estimated_run_time\": task_run.estimated_run_time,\n                \"untrackable_result\": untrackable_result_status,\n            }\n        )\n\n    return dependency_graph\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.set_flow_run_state","title":"set_flow_run_state async","text":"

    Creates a new orchestrated flow run state.

    Setting a new state on a run is the one of the principal actions that is governed by Prefect's orchestration logic. Setting a new run state will not guarantee creation, but instead trigger orchestration rules to govern the proposed state input. If the state is considered valid, it will be written to the database. Otherwise, a it's possible a different state, or no state, will be created. A force flag is supplied to bypass a subset of orchestration logic.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required flow_run_id UUID

    the flow run id

    required state State

    a flow run state model

    required force bool

    if False, orchestration rules will be applied that may alter or prevent the state transition. If True, orchestration rules are not applied.

    False

    Returns:

    Type Description OrchestrationResult

    OrchestrationResult object

    Source code in prefect/server/models/flow_runs.py
    async def set_flow_run_state(\n    session: AsyncSession,\n    flow_run_id: UUID,\n    state: schemas.states.State,\n    force: bool = False,\n    flow_policy: BaseOrchestrationPolicy = None,\n    orchestration_parameters: dict = None,\n) -> OrchestrationResult:\n    \"\"\"\n    Creates a new orchestrated flow run state.\n\n    Setting a new state on a run is the one of the principal actions that is governed by\n    Prefect's orchestration logic. Setting a new run state will not guarantee creation,\n    but instead trigger orchestration rules to govern the proposed `state` input. If\n    the state is considered valid, it will be written to the database. Otherwise, a\n    it's possible a different state, or no state, will be created. A `force` flag is\n    supplied to bypass a subset of orchestration logic.\n\n    Args:\n        session: a database session\n        flow_run_id: the flow run id\n        state: a flow run state model\n        force: if False, orchestration rules will be applied that may alter or prevent\n            the state transition. If True, orchestration rules are not applied.\n\n    Returns:\n        OrchestrationResult object\n    \"\"\"\n\n    # load the flow run\n    run = await models.flow_runs.read_flow_run(\n        session=session,\n        flow_run_id=flow_run_id,\n        # Lock the row to prevent orchestration race conditions\n        for_update=True,\n    )\n\n    if not run:\n        raise ObjectNotFoundError(f\"Flow run with id {flow_run_id} not found\")\n\n    initial_state = run.state.as_state() if run.state else None\n    initial_state_type = initial_state.type if initial_state else None\n    proposed_state_type = state.type if state else None\n    intended_transition = (initial_state_type, proposed_state_type)\n\n    if force or flow_policy is None:\n        flow_policy = MinimalFlowPolicy\n\n    orchestration_rules = flow_policy.compile_transition_rules(*intended_transition)\n    global_rules = GlobalFlowPolicy.compile_transition_rules(*intended_transition)\n\n    context = FlowOrchestrationContext(\n        session=session,\n        run=run,\n        initial_state=initial_state,\n        proposed_state=state,\n    )\n\n    if orchestration_parameters is not None:\n        context.parameters = orchestration_parameters\n\n    # apply orchestration rules and create the new flow run state\n    async with contextlib.AsyncExitStack() as stack:\n        for rule in orchestration_rules:\n            context = await stack.enter_async_context(\n                rule(context, *intended_transition)\n            )\n\n        for rule in global_rules:\n            context = await stack.enter_async_context(\n                rule(context, *intended_transition)\n            )\n\n        await context.validate_proposed_state()\n\n    if context.orchestration_error is not None:\n        raise context.orchestration_error\n\n    result = OrchestrationResult(\n        state=context.validated_state,\n        status=context.response_status,\n        details=context.response_details,\n    )\n\n    # if a new state is being set (either ACCEPTED from user or REJECTED\n    # and set by the server), check for any notification policies\n    if result.status in (SetStateStatus.ACCEPT, SetStateStatus.REJECT):\n        await models.flow_run_notification_policies.queue_flow_run_notifications(\n            session=session, flow_run=run\n        )\n\n    return result\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.update_flow_run","title":"update_flow_run async","text":"

    Updates a flow run.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required flow_run_id UUID

    the flow run id to update

    required flow_run FlowRunUpdate

    a flow run model

    required

    Returns:

    Name Type Description bool bool

    whether or not matching rows were found to update

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def update_flow_run(\n    session: AsyncSession,\n    flow_run_id: UUID,\n    flow_run: schemas.actions.FlowRunUpdate,\n    db: PrefectDBInterface,\n) -> bool:\n    \"\"\"\n    Updates a flow run.\n\n    Args:\n        session: a database session\n        flow_run_id: the flow run id to update\n        flow_run: a flow run model\n\n    Returns:\n        bool: whether or not matching rows were found to update\n    \"\"\"\n    update_stmt = (\n        sa.update(db.FlowRun).where(db.FlowRun.id == flow_run_id)\n        # exclude_unset=True allows us to only update values provided by\n        # the user, ignoring any defaults on the model\n        .values(**flow_run.dict(shallow=True, exclude_unset=True))\n    )\n    result = await session.execute(update_stmt)\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flows/","title":"server.models.flows","text":""},{"location":"api-ref/server/models/flows/#prefect.server.models.flows","title":"prefect.server.models.flows","text":"

    Functions for interacting with flow ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.count_flows","title":"count_flows async","text":"

    Count flows.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_filter FlowFilter

    only count flows that match these filters

    None flow_run_filter FlowRunFilter

    only count flows whose flow runs match these filters

    None task_run_filter TaskRunFilter

    only count flows whose task runs match these filters

    None deployment_filter DeploymentFilter

    only count flows whose deployments match these filters

    None work_pool_filter WorkPoolFilter

    only count flows whose work pools match these filters

    None

    Returns:

    Name Type Description int int

    count of flows

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def count_flows(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n) -> int:\n    \"\"\"\n    Count flows.\n\n    Args:\n        session: A database session\n        flow_filter: only count flows that match these filters\n        flow_run_filter: only count flows whose flow runs match these filters\n        task_run_filter: only count flows whose task runs match these filters\n        deployment_filter: only count flows whose deployments match these filters\n        work_pool_filter: only count flows whose work pools match these filters\n\n    Returns:\n        int: count of flows\n    \"\"\"\n\n    query = select(sa.func.count(sa.text(\"*\"))).select_from(db.Flow)\n\n    query = await _apply_flow_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        db=db,\n    )\n\n    result = await session.execute(query)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.create_flow","title":"create_flow async","text":"

    Creates a new flow.

    If a flow with the same name already exists, the existing flow is returned.

    Parameters:

    Name Type Description Default session Session

    a database session

    required flow Flow

    a flow model

    required

    Returns:

    Type Description

    db.Flow: the newly-created or existing flow

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def create_flow(\n    session: sa.orm.Session, flow: schemas.core.Flow, db: PrefectDBInterface\n):\n    \"\"\"\n    Creates a new flow.\n\n    If a flow with the same name already exists, the existing flow is returned.\n\n    Args:\n        session: a database session\n        flow: a flow model\n\n    Returns:\n        db.Flow: the newly-created or existing flow\n    \"\"\"\n\n    insert_stmt = (\n        (await db.insert(db.Flow))\n        .values(**flow.dict(shallow=True, exclude_unset=True))\n        .on_conflict_do_nothing(\n            index_elements=db.flow_unique_upsert_columns,\n        )\n    )\n    await session.execute(insert_stmt)\n\n    query = (\n        sa.select(db.Flow)\n        .where(\n            db.Flow.name == flow.name,\n        )\n        .limit(1)\n        .execution_options(populate_existing=True)\n    )\n    result = await session.execute(query)\n    model = result.scalar()\n    return model\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.delete_flow","title":"delete_flow async","text":"

    Delete a flow by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_id UUID

    a flow id

    required

    Returns:

    Name Type Description bool bool

    whether or not the flow was deleted

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def delete_flow(\n    session: sa.orm.Session, flow_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a flow by id.\n\n    Args:\n        session: A database session\n        flow_id: a flow id\n\n    Returns:\n        bool: whether or not the flow was deleted\n    \"\"\"\n\n    result = await session.execute(delete(db.Flow).where(db.Flow.id == flow_id))\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.read_flow","title":"read_flow async","text":"

    Reads a flow by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_id UUID

    a flow id

    required

    Returns:

    Type Description

    db.Flow: the flow

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def read_flow(session: sa.orm.Session, flow_id: UUID, db: PrefectDBInterface):\n    \"\"\"\n    Reads a flow by id.\n\n    Args:\n        session: A database session\n        flow_id: a flow id\n\n    Returns:\n        db.Flow: the flow\n    \"\"\"\n    return await session.get(db.Flow, flow_id)\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.read_flow_by_name","title":"read_flow_by_name async","text":"

    Reads a flow by name.

    Parameters:

    Name Type Description Default session Session

    A database session

    required name str

    a flow name

    required

    Returns:

    Type Description

    db.Flow: the flow

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def read_flow_by_name(session: sa.orm.Session, name: str, db: PrefectDBInterface):\n    \"\"\"\n    Reads a flow by name.\n\n    Args:\n        session: A database session\n        name: a flow name\n\n    Returns:\n        db.Flow: the flow\n    \"\"\"\n\n    result = await session.execute(select(db.Flow).filter_by(name=name))\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.read_flows","title":"read_flows async","text":"

    Read multiple flows.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_filter FlowFilter

    only select flows that match these filters

    None flow_run_filter FlowRunFilter

    only select flows whose flow runs match these filters

    None task_run_filter TaskRunFilter

    only select flows whose task runs match these filters

    None deployment_filter DeploymentFilter

    only select flows whose deployments match these filters

    None work_pool_filter WorkPoolFilter

    only select flows whose work pools match these filters

    None offset int

    Query offset

    None limit int

    Query limit

    None

    Returns:

    Type Description

    List[db.Flow]: flows

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def read_flows(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    sort: schemas.sorting.FlowSort = schemas.sorting.FlowSort.NAME_ASC,\n    offset: int = None,\n    limit: int = None,\n):\n    \"\"\"\n    Read multiple flows.\n\n    Args:\n        session: A database session\n        flow_filter: only select flows that match these filters\n        flow_run_filter: only select flows whose flow runs match these filters\n        task_run_filter: only select flows whose task runs match these filters\n        deployment_filter: only select flows whose deployments match these filters\n        work_pool_filter: only select flows whose work pools match these filters\n        offset: Query offset\n        limit: Query limit\n\n    Returns:\n        List[db.Flow]: flows\n    \"\"\"\n\n    query = select(db.Flow).order_by(sort.as_sql_sort(db=db))\n\n    query = await _apply_flow_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        db=db,\n    )\n\n    if offset is not None:\n        query = query.offset(offset)\n\n    if limit is not None:\n        query = query.limit(limit)\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.update_flow","title":"update_flow async","text":"

    Updates a flow.

    Parameters:

    Name Type Description Default session Session

    a database session

    required flow_id UUID

    the flow id to update

    required flow FlowUpdate

    a flow update model

    required

    Returns:

    Name Type Description bool

    whether or not matching rows were found to update

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def update_flow(\n    session: sa.orm.Session,\n    flow_id: UUID,\n    flow: schemas.actions.FlowUpdate,\n    db: PrefectDBInterface,\n):\n    \"\"\"\n    Updates a flow.\n\n    Args:\n        session: a database session\n        flow_id: the flow id to update\n        flow: a flow update model\n\n    Returns:\n        bool: whether or not matching rows were found to update\n    \"\"\"\n    update_stmt = (\n        sa.update(db.Flow).where(db.Flow.id == flow_id)\n        # exclude_unset=True allows us to only update values provided by\n        # the user, ignoring any defaults on the model\n        .values(**flow.dict(shallow=True, exclude_unset=True))\n    )\n    result = await session.execute(update_stmt)\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/saved_searches/","title":"server.models.saved_searches","text":""},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches","title":"prefect.server.models.saved_searches","text":"

    Functions for interacting with saved search ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.create_saved_search","title":"create_saved_search async","text":"

    Upserts a SavedSearch.

    If a SavedSearch with the same name exists, all properties will be updated.

    Parameters:

    Name Type Description Default session Session

    a database session

    required saved_search SavedSearch

    a SavedSearch model

    required

    Returns:

    Type Description

    db.SavedSearch: the newly-created or updated SavedSearch

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def create_saved_search(\n    session: sa.orm.Session,\n    saved_search: schemas.core.SavedSearch,\n    db: PrefectDBInterface,\n):\n    \"\"\"\n    Upserts a SavedSearch.\n\n    If a SavedSearch with the same name exists, all properties will be updated.\n\n    Args:\n        session (sa.orm.Session): a database session\n        saved_search (schemas.core.SavedSearch): a SavedSearch model\n\n    Returns:\n        db.SavedSearch: the newly-created or updated SavedSearch\n\n    \"\"\"\n\n    insert_stmt = (\n        (await db.insert(db.SavedSearch))\n        .values(**saved_search.dict(shallow=True, exclude_unset=True))\n        .on_conflict_do_update(\n            index_elements=db.saved_search_unique_upsert_columns,\n            set_=saved_search.dict(shallow=True, include={\"filters\"}),\n        )\n    )\n\n    await session.execute(insert_stmt)\n\n    query = (\n        sa.select(db.SavedSearch)\n        .where(\n            db.SavedSearch.name == saved_search.name,\n        )\n        .execution_options(populate_existing=True)\n    )\n    result = await session.execute(query)\n    model = result.scalar()\n\n    return model\n
    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.delete_saved_search","title":"delete_saved_search async","text":"

    Delete a SavedSearch by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required saved_search_id str

    a SavedSearch id

    required

    Returns:

    Name Type Description bool bool

    whether or not the SavedSearch was deleted

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def delete_saved_search(\n    session: sa.orm.Session, saved_search_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a SavedSearch by id.\n\n    Args:\n        session (sa.orm.Session): A database session\n        saved_search_id (str): a SavedSearch id\n\n    Returns:\n        bool: whether or not the SavedSearch was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.SavedSearch).where(db.SavedSearch.id == saved_search_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.read_saved_search","title":"read_saved_search async","text":"

    Reads a SavedSearch by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required saved_search_id str

    a SavedSearch id

    required

    Returns:

    Type Description

    db.SavedSearch: the SavedSearch

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def read_saved_search(\n    session: sa.orm.Session, saved_search_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads a SavedSearch by id.\n\n    Args:\n        session (sa.orm.Session): A database session\n        saved_search_id (str): a SavedSearch id\n\n    Returns:\n        db.SavedSearch: the SavedSearch\n    \"\"\"\n\n    return await session.get(db.SavedSearch, saved_search_id)\n
    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.read_saved_search_by_name","title":"read_saved_search_by_name async","text":"

    Reads a SavedSearch by name.

    Parameters:

    Name Type Description Default session Session

    A database session

    required name str

    a SavedSearch name

    required

    Returns:

    Type Description

    db.SavedSearch: the SavedSearch

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def read_saved_search_by_name(\n    session: sa.orm.Session, name: str, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads a SavedSearch by name.\n\n    Args:\n        session (sa.orm.Session): A database session\n        name (str): a SavedSearch name\n\n    Returns:\n        db.SavedSearch: the SavedSearch\n    \"\"\"\n    result = await session.execute(\n        select(db.SavedSearch).where(db.SavedSearch.name == name).limit(1)\n    )\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.read_saved_searches","title":"read_saved_searches async","text":"

    Read SavedSearches.

    Parameters:

    Name Type Description Default session Session

    A database session

    required offset int

    Query offset

    None limit(int)

    Query limit

    required

    Returns:

    Type Description

    List[db.SavedSearch]: SavedSearches

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def read_saved_searches(\n    db: PrefectDBInterface,\n    session: sa.orm.Session,\n    offset: int = None,\n    limit: int = None,\n):\n    \"\"\"\n    Read SavedSearches.\n\n    Args:\n        session (sa.orm.Session): A database session\n        offset (int): Query offset\n        limit(int): Query limit\n\n    Returns:\n        List[db.SavedSearch]: SavedSearches\n    \"\"\"\n\n    query = select(db.SavedSearch).order_by(db.SavedSearch.name)\n\n    if offset is not None:\n        query = query.offset(offset)\n    if limit is not None:\n        query = query.limit(limit)\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/task_run_states/","title":"server.models.task_run_states","text":""},{"location":"api-ref/server/models/task_run_states/#prefect.server.models.task_run_states","title":"prefect.server.models.task_run_states","text":"

    Functions for interacting with task run state ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/task_run_states/#prefect.server.models.task_run_states.delete_task_run_state","title":"delete_task_run_state async","text":"

    Delete a task run state by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required task_run_state_id UUID

    a task run state id

    required

    Returns:

    Name Type Description bool bool

    whether or not the task run state was deleted

    Source code in prefect/server/models/task_run_states.py
    @inject_db\nasync def delete_task_run_state(\n    session: sa.orm.Session, task_run_state_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a task run state by id.\n\n    Args:\n        session: A database session\n        task_run_state_id: a task run state id\n\n    Returns:\n        bool: whether or not the task run state was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.TaskRunState).where(db.TaskRunState.id == task_run_state_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/task_run_states/#prefect.server.models.task_run_states.read_task_run_state","title":"read_task_run_state async","text":"

    Reads a task run state by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required task_run_state_id UUID

    a task run state id

    required

    Returns:

    Type Description

    db.TaskRunState: the task state

    Source code in prefect/server/models/task_run_states.py
    @inject_db\nasync def read_task_run_state(\n    session: sa.orm.Session, task_run_state_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads a task run state by id.\n\n    Args:\n        session: A database session\n        task_run_state_id: a task run state id\n\n    Returns:\n        db.TaskRunState: the task state\n    \"\"\"\n\n    return await session.get(db.TaskRunState, task_run_state_id)\n
    "},{"location":"api-ref/server/models/task_run_states/#prefect.server.models.task_run_states.read_task_run_states","title":"read_task_run_states async","text":"

    Reads task runs states for a task run.

    Parameters:

    Name Type Description Default session Session

    A database session

    required task_run_id UUID

    the task run id

    required

    Returns:

    Type Description

    List[db.TaskRunState]: the task run states

    Source code in prefect/server/models/task_run_states.py
    @inject_db\nasync def read_task_run_states(\n    session: sa.orm.Session, task_run_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads task runs states for a task run.\n\n    Args:\n        session: A database session\n        task_run_id: the task run id\n\n    Returns:\n        List[db.TaskRunState]: the task run states\n    \"\"\"\n\n    query = (\n        select(db.TaskRunState)\n        .filter_by(task_run_id=task_run_id)\n        .order_by(db.TaskRunState.timestamp)\n    )\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/task_runs/","title":"server.models.task_runs","text":""},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs","title":"prefect.server.models.task_runs","text":"

    Functions for interacting with task run ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.count_task_runs","title":"count_task_runs async","text":"

    Count task runs.

    Parameters:

    Name Type Description Default session Session

    a database session

    required flow_filter FlowFilter

    only count task runs whose flows match these filters

    None flow_run_filter FlowRunFilter

    only count task runs whose flow runs match these filters

    None task_run_filter TaskRunFilter

    only count task runs that match these filters

    None deployment_filter DeploymentFilter

    only count task runs whose deployments match these filters

    None Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def count_task_runs(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n) -> int:\n    \"\"\"\n    Count task runs.\n\n    Args:\n        session: a database session\n        flow_filter: only count task runs whose flows match these filters\n        flow_run_filter: only count task runs whose flow runs match these filters\n        task_run_filter: only count task runs that match these filters\n        deployment_filter: only count task runs whose deployments match these filters\n    Returns:\n        int: count of task runs\n    \"\"\"\n\n    query = select(sa.func.count(sa.text(\"*\"))).select_from(db.TaskRun)\n\n    query = await _apply_task_run_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        db=db,\n    )\n\n    result = await session.execute(query)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.create_task_run","title":"create_task_run async","text":"

    Creates a new task run.

    If a task run with the same flow_run_id, task_key, and dynamic_key already exists, the existing task run will be returned. If the provided task run has a state attached, it will also be created.

    Parameters:

    Name Type Description Default session Session

    a database session

    required task_run TaskRun

    a task run model

    required

    Returns:

    Type Description

    db.TaskRun: the newly-created or existing task run

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def create_task_run(\n    session: sa.orm.Session,\n    task_run: schemas.core.TaskRun,\n    db: PrefectDBInterface,\n    orchestration_parameters: dict = None,\n):\n    \"\"\"\n    Creates a new task run.\n\n    If a task run with the same flow_run_id, task_key, and dynamic_key already exists,\n    the existing task run will be returned. If the provided task run has a state\n    attached, it will also be created.\n\n    Args:\n        session: a database session\n        task_run: a task run model\n\n    Returns:\n        db.TaskRun: the newly-created or existing task run\n    \"\"\"\n\n    now = pendulum.now(\"UTC\")\n\n    # if a dynamic key exists, we need to guard against conflicts\n    if task_run.flow_run_id:\n        insert_stmt = (\n            (await db.insert(db.TaskRun))\n            .values(\n                created=now,\n                **task_run.dict(\n                    shallow=True, exclude={\"state\", \"created\"}, exclude_unset=True\n                ),\n            )\n            .on_conflict_do_nothing(\n                index_elements=db.task_run_unique_upsert_columns,\n            )\n        )\n        await session.execute(insert_stmt)\n\n        query = (\n            sa.select(db.TaskRun)\n            .where(\n                sa.and_(\n                    db.TaskRun.flow_run_id == task_run.flow_run_id,\n                    db.TaskRun.task_key == task_run.task_key,\n                    db.TaskRun.dynamic_key == task_run.dynamic_key,\n                )\n            )\n            .limit(1)\n            .execution_options(populate_existing=True)\n        )\n        result = await session.execute(query)\n        model = result.scalar()\n    else:\n        # Upsert on (task_key, dynamic_key) application logic.\n        query = (\n            sa.select(db.TaskRun)\n            .where(\n                sa.and_(\n                    db.TaskRun.flow_run_id.is_(None),\n                    db.TaskRun.task_key == task_run.task_key,\n                    db.TaskRun.dynamic_key == task_run.dynamic_key,\n                )\n            )\n            .limit(1)\n            .execution_options(populate_existing=True)\n        )\n\n        result = await session.execute(query)\n        model = result.scalar()\n\n        if model is None:\n            model = db.TaskRun(\n                created=now,\n                **task_run.dict(\n                    shallow=True, exclude={\"state\", \"created\"}, exclude_unset=True\n                ),\n                state=None,\n            )\n            session.add(model)\n            await session.flush()\n\n    if model.created == now and task_run.state:\n        await models.task_runs.set_task_run_state(\n            session=session,\n            task_run_id=model.id,\n            state=task_run.state,\n            force=True,\n            orchestration_parameters=orchestration_parameters,\n        )\n    return model\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.delete_task_run","title":"delete_task_run async","text":"

    Delete a task run by id.

    Parameters:

    Name Type Description Default session Session

    a database session

    required task_run_id UUID

    the task run id to delete

    required

    Returns:

    Name Type Description bool bool

    whether or not the task run was deleted

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def delete_task_run(\n    session: sa.orm.Session, task_run_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a task run by id.\n\n    Args:\n        session: a database session\n        task_run_id: the task run id to delete\n\n    Returns:\n        bool: whether or not the task run was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.TaskRun).where(db.TaskRun.id == task_run_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.read_task_run","title":"read_task_run async","text":"

    Read a task run by id.

    Parameters:

    Name Type Description Default session Session

    a database session

    required task_run_id UUID

    the task run id

    required

    Returns:

    Type Description

    db.TaskRun: the task run

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def read_task_run(\n    session: sa.orm.Session, task_run_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Read a task run by id.\n\n    Args:\n        session: a database session\n        task_run_id: the task run id\n\n    Returns:\n        db.TaskRun: the task run\n    \"\"\"\n\n    model = await session.get(db.TaskRun, task_run_id)\n    return model\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.read_task_runs","title":"read_task_runs async","text":"

    Read task runs.

    Parameters:

    Name Type Description Default session Session

    a database session

    required flow_filter FlowFilter

    only select task runs whose flows match these filters

    None flow_run_filter FlowRunFilter

    only select task runs whose flow runs match these filters

    None task_run_filter TaskRunFilter

    only select task runs that match these filters

    None deployment_filter DeploymentFilter

    only select task runs whose deployments match these filters

    None offset int

    Query offset

    None limit int

    Query limit

    None sort TaskRunSort

    Query sort

    ID_DESC

    Returns:

    Type Description

    List[db.TaskRun]: the task runs

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def read_task_runs(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    offset: int = None,\n    limit: int = None,\n    sort: schemas.sorting.TaskRunSort = schemas.sorting.TaskRunSort.ID_DESC,\n):\n    \"\"\"\n    Read task runs.\n\n    Args:\n        session: a database session\n        flow_filter: only select task runs whose flows match these filters\n        flow_run_filter: only select task runs whose flow runs match these filters\n        task_run_filter: only select task runs that match these filters\n        deployment_filter: only select task runs whose deployments match these filters\n        offset: Query offset\n        limit: Query limit\n        sort: Query sort\n\n    Returns:\n        List[db.TaskRun]: the task runs\n    \"\"\"\n\n    query = select(db.TaskRun).order_by(sort.as_sql_sort(db))\n\n    query = await _apply_task_run_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        db=db,\n    )\n\n    if offset is not None:\n        query = query.offset(offset)\n\n    if limit is not None:\n        query = query.limit(limit)\n\n    logger.debug(f\"In read_task_runs, query generated is:\\n{query}\")\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.set_task_run_state","title":"set_task_run_state async","text":"

    Creates a new orchestrated task run state.

    Setting a new state on a run is the one of the principal actions that is governed by Prefect's orchestration logic. Setting a new run state will not guarantee creation, but instead trigger orchestration rules to govern the proposed state input. If the state is considered valid, it will be written to the database. Otherwise, a it's possible a different state, or no state, will be created. A force flag is supplied to bypass a subset of orchestration logic.

    Parameters:

    Name Type Description Default session Session

    a database session

    required task_run_id UUID

    the task run id

    required state State

    a task run state model

    required force bool

    if False, orchestration rules will be applied that may alter or prevent the state transition. If True, orchestration rules are not applied.

    False

    Returns:

    Type Description OrchestrationResult

    OrchestrationResult object

    Source code in prefect/server/models/task_runs.py
    async def set_task_run_state(\n    session: sa.orm.Session,\n    task_run_id: UUID,\n    state: schemas.states.State,\n    force: bool = False,\n    task_policy: BaseOrchestrationPolicy = None,\n    orchestration_parameters: dict = None,\n) -> OrchestrationResult:\n    \"\"\"\n    Creates a new orchestrated task run state.\n\n    Setting a new state on a run is the one of the principal actions that is governed by\n    Prefect's orchestration logic. Setting a new run state will not guarantee creation,\n    but instead trigger orchestration rules to govern the proposed `state` input. If\n    the state is considered valid, it will be written to the database. Otherwise, a\n    it's possible a different state, or no state, will be created. A `force` flag is\n    supplied to bypass a subset of orchestration logic.\n\n    Args:\n        session: a database session\n        task_run_id: the task run id\n        state: a task run state model\n        force: if False, orchestration rules will be applied that may alter or prevent\n            the state transition. If True, orchestration rules are not applied.\n\n    Returns:\n        OrchestrationResult object\n    \"\"\"\n\n    # load the task run\n    run = await models.task_runs.read_task_run(session=session, task_run_id=task_run_id)\n\n    if not run:\n        raise ObjectNotFoundError(f\"Task run with id {task_run_id} not found\")\n\n    initial_state = run.state.as_state() if run.state else None\n    initial_state_type = initial_state.type if initial_state else None\n    proposed_state_type = state.type if state else None\n    intended_transition = (initial_state_type, proposed_state_type)\n\n    if force or task_policy is None:\n        task_policy = MinimalTaskPolicy\n\n    orchestration_rules = task_policy.compile_transition_rules(*intended_transition)\n    global_rules = GlobalTaskPolicy.compile_transition_rules(*intended_transition)\n\n    context = TaskOrchestrationContext(\n        session=session,\n        run=run,\n        initial_state=initial_state,\n        proposed_state=state,\n    )\n\n    if orchestration_parameters is not None:\n        context.parameters = orchestration_parameters\n\n    # apply orchestration rules and create the new task run state\n    async with contextlib.AsyncExitStack() as stack:\n        for rule in orchestration_rules:\n            context = await stack.enter_async_context(\n                rule(context, *intended_transition)\n            )\n\n        for rule in global_rules:\n            context = await stack.enter_async_context(\n                rule(context, *intended_transition)\n            )\n\n        await context.validate_proposed_state()\n\n    if context.orchestration_error is not None:\n        raise context.orchestration_error\n\n    result = OrchestrationResult(\n        state=context.validated_state,\n        status=context.response_status,\n        details=context.response_details,\n    )\n\n    return result\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.update_task_run","title":"update_task_run async","text":"

    Updates a task run.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required task_run_id UUID

    the task run id to update

    required task_run TaskRunUpdate

    a task run model

    required

    Returns:

    Name Type Description bool bool

    whether or not matching rows were found to update

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def update_task_run(\n    session: AsyncSession,\n    task_run_id: UUID,\n    task_run: schemas.actions.TaskRunUpdate,\n    db: PrefectDBInterface,\n) -> bool:\n    \"\"\"\n    Updates a task run.\n\n    Args:\n        session: a database session\n        task_run_id: the task run id to update\n        task_run: a task run model\n\n    Returns:\n        bool: whether or not matching rows were found to update\n    \"\"\"\n    update_stmt = (\n        sa.update(db.TaskRun).where(db.TaskRun.id == task_run_id)\n        # exclude_unset=True allows us to only update values provided by\n        # the user, ignoring any defaults on the model\n        .values(**task_run.dict(shallow=True, exclude_unset=True))\n    )\n    result = await session.execute(update_stmt)\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/orchestration/core_policy/","title":"server.orchestration.core_policy","text":""},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy","title":"prefect.server.orchestration.core_policy","text":"

    Orchestration logic that fires on state transitions.

    CoreFlowPolicy and CoreTaskPolicy contain all default orchestration rules that Prefect enforces on a state transition.

    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CoreFlowPolicy","title":"CoreFlowPolicy","text":"

    Bases: BaseOrchestrationPolicy

    Orchestration rules that run against flow-run-state transitions in priority order.

    Source code in prefect/server/orchestration/core_policy.py
    class CoreFlowPolicy(BaseOrchestrationPolicy):\n    \"\"\"\n    Orchestration rules that run against flow-run-state transitions in priority order.\n    \"\"\"\n\n    def priority():\n        return [\n            HandleFlowTerminalStateTransitions,\n            EnforceCancellingToCancelledTransition,\n            BypassCancellingScheduledFlowRuns,\n            PreventPendingTransitions,\n            HandlePausingFlows,\n            HandleResumingPausedFlows,\n            CopyScheduledTime,\n            WaitForScheduledTime,\n            RetryFailedFlows,\n        ]\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CoreTaskPolicy","title":"CoreTaskPolicy","text":"

    Bases: BaseOrchestrationPolicy

    Orchestration rules that run against task-run-state transitions in priority order.

    Source code in prefect/server/orchestration/core_policy.py
    class CoreTaskPolicy(BaseOrchestrationPolicy):\n    \"\"\"\n    Orchestration rules that run against task-run-state transitions in priority order.\n    \"\"\"\n\n    def priority():\n        return [\n            CacheRetrieval,\n            HandleTaskTerminalStateTransitions,\n            PreventRunningTasksFromStoppedFlows,\n            SecureTaskConcurrencySlots,  # retrieve cached states even if slots are full\n            CopyScheduledTime,\n            WaitForScheduledTime,\n            RetryFailedTasks,\n            RenameReruns,\n            UpdateFlowRunTrackerOnTasks,\n            CacheInsertion,\n            ReleaseTaskConcurrencySlots,\n        ]\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.SecureTaskConcurrencySlots","title":"SecureTaskConcurrencySlots","text":"

    Bases: BaseOrchestrationRule

    Checks relevant concurrency slots are available before entering a Running state.

    This rule checks if concurrency limits have been set on the tags associated with a TaskRun. If so, a concurrency slot will be secured against each concurrency limit before being allowed to transition into a running state. If a concurrency limit has been reached, the client will be instructed to delay the transition for the duration specified by the \"PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS\" setting before trying again. If the concurrency limit set on a tag is 0, the transition will be aborted to prevent deadlocks.

    Source code in prefect/server/orchestration/core_policy.py
    class SecureTaskConcurrencySlots(BaseOrchestrationRule):\n    \"\"\"\n    Checks relevant concurrency slots are available before entering a Running state.\n\n    This rule checks if concurrency limits have been set on the tags associated with a\n    TaskRun. If so, a concurrency slot will be secured against each concurrency limit\n    before being allowed to transition into a running state. If a concurrency limit has\n    been reached, the client will be instructed to delay the transition for the duration\n    specified by the \"PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS\" setting\n    before trying again. If the concurrency limit set on a tag is 0, the transition will\n    be aborted to prevent deadlocks.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        self._applied_limits = []\n        filtered_limits = (\n            await concurrency_limits.filter_concurrency_limits_for_orchestration(\n                context.session, tags=context.run.tags\n            )\n        )\n        run_limits = {limit.tag: limit for limit in filtered_limits}\n        for tag, cl in run_limits.items():\n            limit = cl.concurrency_limit\n            if limit == 0:\n                # limits of 0 will deadlock, and the transition needs to abort\n                for stale_tag in self._applied_limits:\n                    stale_limit = run_limits.get(stale_tag, None)\n                    active_slots = set(stale_limit.active_slots)\n                    active_slots.discard(str(context.run.id))\n                    stale_limit.active_slots = list(active_slots)\n\n                await self.abort_transition(\n                    reason=(\n                        f'The concurrency limit on tag \"{tag}\" is 0 and will deadlock'\n                        \" if the task tries to run again.\"\n                    ),\n                )\n            elif len(cl.active_slots) >= limit:\n                # if the limit has already been reached, delay the transition\n                for stale_tag in self._applied_limits:\n                    stale_limit = run_limits.get(stale_tag, None)\n                    active_slots = set(stale_limit.active_slots)\n                    active_slots.discard(str(context.run.id))\n                    stale_limit.active_slots = list(active_slots)\n\n                await self.delay_transition(\n                    PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS.value(),\n                    f\"Concurrency limit for the {tag} tag has been reached\",\n                )\n            else:\n                # log the TaskRun ID to active_slots\n                self._applied_limits.append(tag)\n                active_slots = set(cl.active_slots)\n                active_slots.add(str(context.run.id))\n                cl.active_slots = list(active_slots)\n\n    async def cleanup(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        for tag in self._applied_limits:\n            cl = await concurrency_limits.read_concurrency_limit_by_tag(\n                context.session, tag\n            )\n            active_slots = set(cl.active_slots)\n            active_slots.discard(str(context.run.id))\n            cl.active_slots = list(active_slots)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.ReleaseTaskConcurrencySlots","title":"ReleaseTaskConcurrencySlots","text":"

    Bases: BaseUniversalTransform

    Releases any concurrency slots held by a run upon exiting a Running or Cancelling state.

    Source code in prefect/server/orchestration/core_policy.py
    class ReleaseTaskConcurrencySlots(BaseUniversalTransform):\n    \"\"\"\n    Releases any concurrency slots held by a run upon exiting a Running or\n    Cancelling state.\n    \"\"\"\n\n    async def after_transition(\n        self,\n        context: OrchestrationContext,\n    ):\n        if self.nullified_transition():\n            return\n\n        if context.validated_state and context.validated_state.type not in [\n            states.StateType.RUNNING,\n            states.StateType.CANCELLING,\n        ]:\n            filtered_limits = (\n                await concurrency_limits.filter_concurrency_limits_for_orchestration(\n                    context.session, tags=context.run.tags\n                )\n            )\n            run_limits = {limit.tag: limit for limit in filtered_limits}\n            for tag, cl in run_limits.items():\n                active_slots = set(cl.active_slots)\n                active_slots.discard(str(context.run.id))\n                cl.active_slots = list(active_slots)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CacheInsertion","title":"CacheInsertion","text":"

    Bases: BaseOrchestrationRule

    Caches completed states with cache keys after they are validated.

    Source code in prefect/server/orchestration/core_policy.py
    class CacheInsertion(BaseOrchestrationRule):\n    \"\"\"\n    Caches completed states with cache keys after they are validated.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.COMPLETED]\n\n    @inject_db\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n        db: PrefectDBInterface,\n    ) -> None:\n        if not validated_state or not context.session:\n            return\n\n        cache_key = validated_state.state_details.cache_key\n        if cache_key:\n            new_cache_item = db.TaskRunStateCache(\n                cache_key=cache_key,\n                cache_expiration=validated_state.state_details.cache_expiration,\n                task_run_state_id=validated_state.id,\n            )\n            context.session.add(new_cache_item)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CacheRetrieval","title":"CacheRetrieval","text":"

    Bases: BaseOrchestrationRule

    Rejects running states if a completed state has been cached.

    This rule rejects transitions into a running state with a cache key if the key has already been associated with a completed state in the cache table. The client will be instructed to transition into the cached completed state instead.

    Source code in prefect/server/orchestration/core_policy.py
    class CacheRetrieval(BaseOrchestrationRule):\n    \"\"\"\n    Rejects running states if a completed state has been cached.\n\n    This rule rejects transitions into a running state with a cache key if the key\n    has already been associated with a completed state in the cache table. The client\n    will be instructed to transition into the cached completed state instead.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    @inject_db\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n        db: PrefectDBInterface,\n    ) -> None:\n        cache_key = proposed_state.state_details.cache_key\n        if cache_key and not proposed_state.state_details.refresh_cache:\n            # Check for cached states matching the cache key\n            cached_state_id = (\n                select(db.TaskRunStateCache.task_run_state_id)\n                .where(\n                    sa.and_(\n                        db.TaskRunStateCache.cache_key == cache_key,\n                        sa.or_(\n                            db.TaskRunStateCache.cache_expiration.is_(None),\n                            db.TaskRunStateCache.cache_expiration > pendulum.now(\"utc\"),\n                        ),\n                    ),\n                )\n                .order_by(db.TaskRunStateCache.created.desc())\n                .limit(1)\n            ).scalar_subquery()\n            query = select(db.TaskRunState).where(db.TaskRunState.id == cached_state_id)\n            cached_state = (await context.session.execute(query)).scalar()\n            if cached_state:\n                new_state = cached_state.as_state().copy(reset_fields=True)\n                new_state.name = \"Cached\"\n                await self.reject_transition(\n                    state=new_state, reason=\"Retrieved state from cache\"\n                )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.RetryFailedFlows","title":"RetryFailedFlows","text":"

    Bases: BaseOrchestrationRule

    Rejects failed states and schedules a retry if the retry limit has not been reached.

    This rule rejects transitions into a failed state if retries has been set and the run count has not reached the specified limit. The client will be instructed to transition into a scheduled state to retry flow execution.

    Source code in prefect/server/orchestration/core_policy.py
    class RetryFailedFlows(BaseOrchestrationRule):\n    \"\"\"\n    Rejects failed states and schedules a retry if the retry limit has not been reached.\n\n    This rule rejects transitions into a failed state if `retries` has been\n    set and the run count has not reached the specified limit. The client will be\n    instructed to transition into a scheduled state to retry flow execution.\n    \"\"\"\n\n    FROM_STATES = [StateType.RUNNING]\n    TO_STATES = [StateType.FAILED]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: FlowOrchestrationContext,\n    ) -> None:\n        run_settings = context.run_settings\n        run_count = context.run.run_count\n\n        if run_settings.retries is None or run_count > run_settings.retries:\n            return  # Retry count exceeded, allow transition to failed\n\n        scheduled_start_time = pendulum.now(\"UTC\").add(\n            seconds=run_settings.retry_delay or 0\n        )\n\n        # support old-style flow run retries for older clients\n        # older flow retries require us to loop over failed tasks to update their state\n        # this is not required after API version 0.8.3\n        api_version = context.parameters.get(\"api-version\", None)\n        if api_version and api_version < Version(\"0.8.3\"):\n            failed_task_runs = await models.task_runs.read_task_runs(\n                context.session,\n                flow_run_filter=filters.FlowRunFilter(id={\"any_\": [context.run.id]}),\n                task_run_filter=filters.TaskRunFilter(\n                    state={\"type\": {\"any_\": [\"FAILED\"]}}\n                ),\n            )\n            for run in failed_task_runs:\n                await models.task_runs.set_task_run_state(\n                    context.session,\n                    run.id,\n                    state=states.AwaitingRetry(scheduled_time=scheduled_start_time),\n                    force=True,\n                )\n                # Reset the run count so that the task run retries still work correctly\n                run.run_count = 0\n\n        # Reset pause metadata on retry\n        # Pauses as a concept only exist after API version 0.8.4\n        api_version = context.parameters.get(\"api-version\", None)\n        if api_version is None or api_version >= Version(\"0.8.4\"):\n            updated_policy = context.run.empirical_policy.dict()\n            updated_policy[\"resuming\"] = False\n            updated_policy[\"pause_keys\"] = set()\n            context.run.empirical_policy = core.FlowRunPolicy(**updated_policy)\n\n        # Generate a new state for the flow\n        retry_state = states.AwaitingRetry(\n            scheduled_time=scheduled_start_time,\n            message=proposed_state.message,\n            data=proposed_state.data,\n        )\n        await self.reject_transition(state=retry_state, reason=\"Retrying\")\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.RetryFailedTasks","title":"RetryFailedTasks","text":"

    Bases: BaseOrchestrationRule

    Rejects failed states and schedules a retry if the retry limit has not been reached.

    This rule rejects transitions into a failed state if retries has been set, the run count has not reached the specified limit, and the client asserts it is a retriable task run. The client will be instructed to transition into a scheduled state to retry task execution.

    Source code in prefect/server/orchestration/core_policy.py
    class RetryFailedTasks(BaseOrchestrationRule):\n    \"\"\"\n    Rejects failed states and schedules a retry if the retry limit has not been reached.\n\n    This rule rejects transitions into a failed state if `retries` has been\n    set, the run count has not reached the specified limit, and the client\n    asserts it is a retriable task run. The client will be instructed to\n    transition into a scheduled state to retry task execution.\n    \"\"\"\n\n    FROM_STATES = [StateType.RUNNING]\n    TO_STATES = [StateType.FAILED]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        run_settings = context.run_settings\n        run_count = context.run.run_count\n        delay = run_settings.retry_delay\n\n        if isinstance(delay, list):\n            base_delay = delay[min(run_count - 1, len(delay) - 1)]\n        else:\n            base_delay = run_settings.retry_delay or 0\n\n        # guard against negative relative jitter inputs\n        if run_settings.retry_jitter_factor:\n            delay = clamped_poisson_interval(\n                base_delay, clamping_factor=run_settings.retry_jitter_factor\n            )\n        else:\n            delay = base_delay\n\n        # set by user to conditionally retry a task using @task(retry_condition_fn=...)\n        if getattr(proposed_state.state_details, \"retriable\", True) is False:\n            return\n\n        if run_settings.retries is not None and run_count <= run_settings.retries:\n            retry_state = states.AwaitingRetry(\n                scheduled_time=pendulum.now(\"UTC\").add(seconds=delay),\n                message=proposed_state.message,\n                data=proposed_state.data,\n            )\n            await self.reject_transition(state=retry_state, reason=\"Retrying\")\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.RenameReruns","title":"RenameReruns","text":"

    Bases: BaseOrchestrationRule

    Name the states if they have run more than once.

    In the special case where the initial state is an \"AwaitingRetry\" scheduled state, the proposed state will be renamed to \"Retrying\" instead.

    Source code in prefect/server/orchestration/core_policy.py
    class RenameReruns(BaseOrchestrationRule):\n    \"\"\"\n    Name the states if they have run more than once.\n\n    In the special case where the initial state is an \"AwaitingRetry\" scheduled state,\n    the proposed state will be renamed to \"Retrying\" instead.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        run_count = context.run.run_count\n        if run_count > 0:\n            if initial_state.name == \"AwaitingRetry\":\n                await self.rename_state(\"Retrying\")\n            else:\n                await self.rename_state(\"Rerunning\")\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CopyScheduledTime","title":"CopyScheduledTime","text":"

    Bases: BaseOrchestrationRule

    Ensures scheduled time is copied from scheduled states to pending states.

    If a new scheduled time has been proposed on the pending state, the scheduled time on the scheduled state will be ignored.

    Source code in prefect/server/orchestration/core_policy.py
    class CopyScheduledTime(BaseOrchestrationRule):\n    \"\"\"\n    Ensures scheduled time is copied from scheduled states to pending states.\n\n    If a new scheduled time has been proposed on the pending state, the scheduled time\n    on the scheduled state will be ignored.\n    \"\"\"\n\n    FROM_STATES = [StateType.SCHEDULED]\n    TO_STATES = [StateType.PENDING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        if not proposed_state.state_details.scheduled_time:\n            proposed_state.state_details.scheduled_time = (\n                initial_state.state_details.scheduled_time\n            )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.WaitForScheduledTime","title":"WaitForScheduledTime","text":"

    Bases: BaseOrchestrationRule

    Prevents transitions to running states from happening to early.

    This rule enforces that all scheduled states will only start with the machine clock used by the Prefect REST API instance. This rule will identify transitions from scheduled states that are too early and nullify them. Instead, no state will be written to the database and the client will be sent an instruction to wait for delay_seconds before attempting the transition again.

    Source code in prefect/server/orchestration/core_policy.py
    class WaitForScheduledTime(BaseOrchestrationRule):\n    \"\"\"\n    Prevents transitions to running states from happening to early.\n\n    This rule enforces that all scheduled states will only start with the machine clock\n    used by the Prefect REST API instance. This rule will identify transitions from scheduled\n    states that are too early and nullify them. Instead, no state will be written to the\n    database and the client will be sent an instruction to wait for `delay_seconds`\n    before attempting the transition again.\n    \"\"\"\n\n    FROM_STATES = [StateType.SCHEDULED, StateType.PENDING]\n    TO_STATES = [StateType.RUNNING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        scheduled_time = initial_state.state_details.scheduled_time\n        if not scheduled_time:\n            return\n\n        # At this moment, we round delay to the nearest second as the API schema\n        # specifies an integer return value.\n        delay = scheduled_time - pendulum.now(\"UTC\")\n        delay_seconds = delay.in_seconds()\n        delay_seconds += round(delay.microseconds / 1e6)\n        if delay_seconds > 0:\n            await self.delay_transition(\n                delay_seconds, reason=\"Scheduled time is in the future\"\n            )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.HandlePausingFlows","title":"HandlePausingFlows","text":"

    Bases: BaseOrchestrationRule

    Governs runs attempting to enter a Paused/Suspended state

    Source code in prefect/server/orchestration/core_policy.py
    class HandlePausingFlows(BaseOrchestrationRule):\n    \"\"\"\n    Governs runs attempting to enter a Paused/Suspended state\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.PAUSED]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        verb = \"suspend\" if proposed_state.name == \"Suspended\" else \"pause\"\n\n        if initial_state is None:\n            await self.abort_transition(f\"Cannot {verb} flows with no state.\")\n            return\n\n        if not initial_state.is_running():\n            await self.reject_transition(\n                state=None,\n                reason=f\"Cannot {verb} flows that are not currently running.\",\n            )\n            return\n\n        self.key = proposed_state.state_details.pause_key\n        if self.key is None:\n            # if no pause key is provided, default to a UUID\n            self.key = str(uuid4())\n\n        if self.key in context.run.empirical_policy.pause_keys:\n            await self.reject_transition(\n                state=None, reason=f\"This {verb} has already fired.\"\n            )\n            return\n\n        if proposed_state.state_details.pause_reschedule:\n            if context.run.parent_task_run_id:\n                await self.abort_transition(\n                    reason=f\"Cannot {verb} subflows.\",\n                )\n                return\n\n            if context.run.deployment_id is None:\n                await self.abort_transition(\n                    reason=f\"Cannot {verb} flows without a deployment.\",\n                )\n                return\n\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        updated_policy = context.run.empirical_policy.dict()\n        updated_policy[\"pause_keys\"].add(self.key)\n        context.run.empirical_policy = core.FlowRunPolicy(**updated_policy)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.HandleResumingPausedFlows","title":"HandleResumingPausedFlows","text":"

    Bases: BaseOrchestrationRule

    Governs runs attempting to leave a Paused state

    Source code in prefect/server/orchestration/core_policy.py
    class HandleResumingPausedFlows(BaseOrchestrationRule):\n    \"\"\"\n    Governs runs attempting to leave a Paused state\n    \"\"\"\n\n    FROM_STATES = [StateType.PAUSED]\n    TO_STATES = ALL_ORCHESTRATION_STATES\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        if not (\n            proposed_state.is_running()\n            or proposed_state.is_scheduled()\n            or proposed_state.is_final()\n        ):\n            await self.reject_transition(\n                state=None,\n                reason=(\n                    f\"This run cannot transition to the {proposed_state.type} state\"\n                    f\" from the {initial_state.type} state.\"\n                ),\n            )\n            return\n\n        verb = \"suspend\" if proposed_state.name == \"Suspended\" else \"pause\"\n\n        if initial_state.state_details.pause_reschedule:\n            if not context.run.deployment_id:\n                await self.reject_transition(\n                    state=None,\n                    reason=(\n                        f\"Cannot reschedule a {proposed_state.name.lower()} flow run\"\n                        \" without a deployment.\"\n                    ),\n                )\n                return\n        pause_timeout = initial_state.state_details.pause_timeout\n        if pause_timeout and pause_timeout < pendulum.now(\"UTC\"):\n            pause_timeout_failure = states.Failed(\n                message=(\n                    f\"The flow was {proposed_state.name.lower()} and never resumed.\"\n                ),\n            )\n            await self.reject_transition(\n                state=pause_timeout_failure,\n                reason=f\"The flow run {verb} has timed out and can no longer resume.\",\n            )\n            return\n\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        updated_policy = context.run.empirical_policy.dict()\n        updated_policy[\"resuming\"] = True\n        context.run.empirical_policy = core.FlowRunPolicy(**updated_policy)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.UpdateFlowRunTrackerOnTasks","title":"UpdateFlowRunTrackerOnTasks","text":"

    Bases: BaseOrchestrationRule

    Tracks the flow run attempt a task run state is associated with.

    Source code in prefect/server/orchestration/core_policy.py
    class UpdateFlowRunTrackerOnTasks(BaseOrchestrationRule):\n    \"\"\"\n    Tracks the flow run attempt a task run state is associated with.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        if context.run.flow_run_id is not None:\n            self.flow_run = await context.flow_run()\n            if self.flow_run:\n                context.run.flow_run_run_count = self.flow_run.run_count\n            else:\n                raise ObjectNotFoundError(\n                    (\n                        \"Unable to read flow run associated with task run:\"\n                        f\" {context.run.id}, this flow run might have been deleted\"\n                    ),\n                )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.HandleTaskTerminalStateTransitions","title":"HandleTaskTerminalStateTransitions","text":"

    Bases: BaseOrchestrationRule

    We do not allow tasks to leave terminal states if: - The task is completed and has a persisted result - The task is going to CANCELLING / PAUSED / CRASHED

    We reset the run count when a task leaves a terminal state for a non-terminal state which resets task run retries; this is particularly relevant for flow run retries.

    Source code in prefect/server/orchestration/core_policy.py
    class HandleTaskTerminalStateTransitions(BaseOrchestrationRule):\n    \"\"\"\n    We do not allow tasks to leave terminal states if:\n    - The task is completed and has a persisted result\n    - The task is going to CANCELLING / PAUSED / CRASHED\n\n    We reset the run count when a task leaves a terminal state for a non-terminal state\n    which resets task run retries; this is particularly relevant for flow run retries.\n    \"\"\"\n\n    FROM_STATES = TERMINAL_STATES\n    TO_STATES = ALL_ORCHESTRATION_STATES\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        self.original_run_count = context.run.run_count\n\n        # Do not allow runs to be marked as crashed, paused, or cancelling if already terminal\n        if proposed_state.type in {\n            StateType.CANCELLING,\n            StateType.PAUSED,\n            StateType.CRASHED,\n        }:\n            await self.abort_transition(f\"Run is already {initial_state.type.value}.\")\n            return\n\n        # Only allow departure from a happily completed state if the result is not persisted\n        if (\n            initial_state.is_completed()\n            and initial_state.data\n            and initial_state.data.get(\"type\") != \"unpersisted\"\n        ):\n            await self.reject_transition(None, \"This run is already completed.\")\n            return\n\n        if not proposed_state.is_final():\n            # Reset run count to reset retries\n            context.run.run_count = 0\n\n        # Change the name of the state to retrying if its a flow run retry\n        if proposed_state.is_running() and context.run.flow_run_id is not None:\n            self.flow_run = await context.flow_run()\n            flow_retrying = context.run.flow_run_run_count < self.flow_run.run_count\n            if flow_retrying:\n                await self.rename_state(\"Retrying\")\n\n    async def cleanup(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ):\n        # reset run count\n        context.run.run_count = self.original_run_count\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.HandleFlowTerminalStateTransitions","title":"HandleFlowTerminalStateTransitions","text":"

    Bases: BaseOrchestrationRule

    We do not allow flows to leave terminal states if: - The flow is completed and has a persisted result - The flow is going to CANCELLING / PAUSED / CRASHED - The flow is going to scheduled and has no deployment

    We reset the pause metadata when a flow leaves a terminal state for a non-terminal state. This resets pause behavior during manual flow run retries.

    Source code in prefect/server/orchestration/core_policy.py
    class HandleFlowTerminalStateTransitions(BaseOrchestrationRule):\n    \"\"\"\n    We do not allow flows to leave terminal states if:\n    - The flow is completed and has a persisted result\n    - The flow is going to CANCELLING / PAUSED / CRASHED\n    - The flow is going to scheduled and has no deployment\n\n    We reset the pause metadata when a flow leaves a terminal state for a non-terminal\n    state. This resets pause behavior during manual flow run retries.\n    \"\"\"\n\n    FROM_STATES = TERMINAL_STATES\n    TO_STATES = ALL_ORCHESTRATION_STATES\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: FlowOrchestrationContext,\n    ) -> None:\n        self.original_flow_policy = context.run.empirical_policy.dict()\n\n        # Do not allow runs to be marked as crashed, paused, or cancelling if already terminal\n        if proposed_state.type in {\n            StateType.CANCELLING,\n            StateType.PAUSED,\n            StateType.CRASHED,\n        }:\n            await self.abort_transition(\n                f\"Run is already in terminal state {initial_state.type.value}.\"\n            )\n            return\n\n        # Only allow departure from a happily completed state if the result is not\n        # persisted and the a rerun is being proposed\n        if (\n            initial_state.is_completed()\n            and not proposed_state.is_final()\n            and initial_state.data\n            and initial_state.data.get(\"type\") != \"unpersisted\"\n        ):\n            await self.reject_transition(None, \"Run is already COMPLETED.\")\n            return\n\n        # Do not allows runs to be rescheduled without a deployment\n        if proposed_state.is_scheduled() and not context.run.deployment_id:\n            await self.abort_transition(\n                \"Cannot reschedule a run without an associated deployment.\"\n            )\n            return\n\n        if not proposed_state.is_final():\n            # Reset pause metadata when leaving a terminal state\n            api_version = context.parameters.get(\"api-version\", None)\n            if api_version is None or api_version >= Version(\"0.8.4\"):\n                updated_policy = context.run.empirical_policy.dict()\n                updated_policy[\"resuming\"] = False\n                updated_policy[\"pause_keys\"] = set()\n                context.run.empirical_policy = core.FlowRunPolicy(**updated_policy)\n\n    async def cleanup(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ):\n        context.run.empirical_policy = core.FlowRunPolicy(**self.original_flow_policy)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.PreventPendingTransitions","title":"PreventPendingTransitions","text":"

    Bases: BaseOrchestrationRule

    Prevents transitions to PENDING.

    This rule is only used for flow runs.

    This is intended to prevent race conditions during duplicate submissions of runs. Before a run is submitted to its execution environment, it should be placed in a PENDING state. If two workers attempt to submit the same run, one of them should encounter a PENDING -> PENDING transition and abort orchestration of the run.

    Similarly, if the execution environment starts quickly the run may be in a RUNNING state when the second worker attempts the PENDING transition. We deny these state changes as well to prevent duplicate submission. If a run has transitioned to a RUNNING state a worker should not attempt to submit it again unless it has moved into a terminal state.

    CANCELLING and CANCELLED runs should not be allowed to transition to PENDING. For re-runs of deployed runs, they should transition to SCHEDULED first. For re-runs of ad-hoc runs, they should transition directly to RUNNING.

    Source code in prefect/server/orchestration/core_policy.py
    class PreventPendingTransitions(BaseOrchestrationRule):\n    \"\"\"\n    Prevents transitions to PENDING.\n\n    This rule is only used for flow runs.\n\n    This is intended to prevent race conditions during duplicate submissions of runs.\n    Before a run is submitted to its execution environment, it should be placed in a\n    PENDING state. If two workers attempt to submit the same run, one of them should\n    encounter a PENDING -> PENDING transition and abort orchestration of the run.\n\n    Similarly, if the execution environment starts quickly the run may be in a RUNNING\n    state when the second worker attempts the PENDING transition. We deny these state\n    changes as well to prevent duplicate submission. If a run has transitioned to a\n    RUNNING state a worker should not attempt to submit it again unless it has moved\n    into a terminal state.\n\n    CANCELLING and CANCELLED runs should not be allowed to transition to PENDING.\n    For re-runs of deployed runs, they should transition to SCHEDULED first.\n    For re-runs of ad-hoc runs, they should transition directly to RUNNING.\n    \"\"\"\n\n    FROM_STATES = [\n        StateType.PENDING,\n        StateType.CANCELLING,\n        StateType.RUNNING,\n        StateType.CANCELLED,\n    ]\n    TO_STATES = [StateType.PENDING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        await self.abort_transition(\n            reason=(\n                f\"This run is in a {initial_state.type.name} state and cannot\"\n                \" transition to a PENDING state.\"\n            )\n        )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.PreventRunningTasksFromStoppedFlows","title":"PreventRunningTasksFromStoppedFlows","text":"

    Bases: BaseOrchestrationRule

    Prevents running tasks from stopped flows.

    A running state implies execution, but also the converse. This rule ensures that a flow's tasks cannot be run unless the flow is also running.

    Source code in prefect/server/orchestration/core_policy.py
    class PreventRunningTasksFromStoppedFlows(BaseOrchestrationRule):\n    \"\"\"\n    Prevents running tasks from stopped flows.\n\n    A running state implies execution, but also the converse. This rule ensures that a\n    flow's tasks cannot be run unless the flow is also running.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        flow_run = await context.flow_run()\n        if flow_run is not None:\n            if flow_run.state is None:\n                await self.abort_transition(\n                    reason=\"The enclosing flow must be running to begin task execution.\"\n                )\n            elif flow_run.state.type == StateType.PAUSED:\n                # Use the flow run's Paused state details to preserve data like\n                # timeouts.\n                paused_state = states.Paused(\n                    name=\"NotReady\",\n                    pause_expiration_time=flow_run.state.state_details.pause_timeout,\n                    reschedule=flow_run.state.state_details.pause_reschedule,\n                )\n                await self.reject_transition(\n                    state=paused_state,\n                    reason=(\n                        \"The flow is paused, new tasks can execute after resuming flow\"\n                        f\" run: {flow_run.id}.\"\n                    ),\n                )\n            elif not flow_run.state.type == StateType.RUNNING:\n                # task runners should abort task run execution\n                await self.abort_transition(\n                    reason=(\n                        \"The enclosing flow must be running to begin task execution.\"\n                    ),\n                )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.EnforceCancellingToCancelledTransition","title":"EnforceCancellingToCancelledTransition","text":"

    Bases: BaseOrchestrationRule

    Rejects transitions from Cancelling to any terminal state except for Cancelled.

    Source code in prefect/server/orchestration/core_policy.py
    class EnforceCancellingToCancelledTransition(BaseOrchestrationRule):\n    \"\"\"\n    Rejects transitions from Cancelling to any terminal state except for Cancelled.\n    \"\"\"\n\n    FROM_STATES = {StateType.CANCELLED, StateType.CANCELLING}\n    TO_STATES = ALL_ORCHESTRATION_STATES - {StateType.CANCELLED}\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        await self.reject_transition(\n            state=None,\n            reason=(\n                \"Cannot transition flows that are cancelling to a state other \"\n                \"than Cancelled.\"\n            ),\n        )\n        return\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.BypassCancellingScheduledFlowRuns","title":"BypassCancellingScheduledFlowRuns","text":"

    Bases: BaseOrchestrationRule

    Rejects transitions from Scheduled to Cancelling, and instead sets the state to Cancelled, if the flow run has no associated infrastructure process ID.

    The Cancelling state is used to clean up infrastructure. If there is not infrastructure to clean up, we can transition directly to Cancelled. Runs that are AwaitingRetry are a Scheduled state that may have associated infrastructure.

    Source code in prefect/server/orchestration/core_policy.py
    class BypassCancellingScheduledFlowRuns(BaseOrchestrationRule):\n    \"\"\"Rejects transitions from Scheduled to Cancelling, and instead sets the state to Cancelled,\n    if the flow run has no associated infrastructure process ID.\n\n    The `Cancelling` state is used to clean up infrastructure. If there is not infrastructure\n    to clean up, we can transition directly to `Cancelled`. Runs that are `AwaitingRetry` are\n    a `Scheduled` state that may have associated infrastructure.\n    \"\"\"\n\n    FROM_STATES = {StateType.SCHEDULED}\n    TO_STATES = {StateType.CANCELLING}\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: FlowOrchestrationContext,\n    ) -> None:\n        if not context.run.infrastructure_pid:\n            await self.reject_transition(\n                state=states.Cancelled(),\n                reason=\"Scheduled flow run has no infrastructure to terminate.\",\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/","title":"server.orchestration.global_policy","text":""},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy","title":"prefect.server.orchestration.global_policy","text":"

    Bookkeeping logic that fires on every state transition.

    For clarity, GlobalFlowpolicy and GlobalTaskPolicy contain all transition logic implemented using BaseUniversalTransform. None of these operations modify state, and regardless of what orchestration Prefect REST API might enforce on a transition, the global policies contain Prefect's necessary bookkeeping. Because these transforms record information about the validated state committed to the state database, they should be the most deeply nested contexts in orchestration loop.

    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.GlobalFlowPolicy","title":"GlobalFlowPolicy","text":"

    Bases: BaseOrchestrationPolicy

    Global transforms that run against flow-run-state transitions in priority order.

    These transforms are intended to run immediately before and after a state transition is validated.

    Source code in prefect/server/orchestration/global_policy.py
    class GlobalFlowPolicy(BaseOrchestrationPolicy):\n    \"\"\"\n    Global transforms that run against flow-run-state transitions in priority order.\n\n    These transforms are intended to run immediately before and after a state transition\n    is validated.\n    \"\"\"\n\n    def priority():\n        return COMMON_GLOBAL_TRANSFORMS() + [\n            UpdateSubflowParentTask,\n            UpdateSubflowStateDetails,\n            IncrementFlowRunCount,\n            RemoveResumingIndicator,\n        ]\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.GlobalTaskPolicy","title":"GlobalTaskPolicy","text":"

    Bases: BaseOrchestrationPolicy

    Global transforms that run against task-run-state transitions in priority order.

    These transforms are intended to run immediately before and after a state transition is validated.

    Source code in prefect/server/orchestration/global_policy.py
    class GlobalTaskPolicy(BaseOrchestrationPolicy):\n    \"\"\"\n    Global transforms that run against task-run-state transitions in priority order.\n\n    These transforms are intended to run immediately before and after a state transition\n    is validated.\n    \"\"\"\n\n    def priority():\n        return COMMON_GLOBAL_TRANSFORMS() + [\n            IncrementTaskRunCount,\n        ]\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetRunStateType","title":"SetRunStateType","text":"

    Bases: BaseUniversalTransform

    Updates the state type of a run on a state transition.

    Source code in prefect/server/orchestration/global_policy.py
    class SetRunStateType(BaseUniversalTransform):\n    \"\"\"\n    Updates the state type of a run on a state transition.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # record the new state's type\n        context.run.state_type = context.proposed_state.type\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetRunStateName","title":"SetRunStateName","text":"

    Bases: BaseUniversalTransform

    Updates the state name of a run on a state transition.

    Source code in prefect/server/orchestration/global_policy.py
    class SetRunStateName(BaseUniversalTransform):\n    \"\"\"\n    Updates the state name of a run on a state transition.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # record the new state's name\n        context.run.state_name = context.proposed_state.name\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetStartTime","title":"SetStartTime","text":"

    Bases: BaseUniversalTransform

    Records the time a run enters a running state for the first time.

    Source code in prefect/server/orchestration/global_policy.py
    class SetStartTime(BaseUniversalTransform):\n    \"\"\"\n    Records the time a run enters a running state for the first time.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if entering a running state and no start time is set...\n        if context.proposed_state.is_running() and context.run.start_time is None:\n            # set the start time\n            context.run.start_time = context.proposed_state.timestamp\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetRunStateTimestamp","title":"SetRunStateTimestamp","text":"

    Bases: BaseUniversalTransform

    Records the time a run changes states.

    Source code in prefect/server/orchestration/global_policy.py
    class SetRunStateTimestamp(BaseUniversalTransform):\n    \"\"\"\n    Records the time a run changes states.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # record the new state's timestamp\n        context.run.state_timestamp = context.proposed_state.timestamp\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetEndTime","title":"SetEndTime","text":"

    Bases: BaseUniversalTransform

    Records the time a run enters a terminal state.

    With normal client usage, a run will not transition out of a terminal state. However, it's possible to force these transitions manually via the API. While leaving a terminal state, the end time will be unset.

    Source code in prefect/server/orchestration/global_policy.py
    class SetEndTime(BaseUniversalTransform):\n    \"\"\"\n    Records the time a run enters a terminal state.\n\n    With normal client usage, a run will not transition out of a terminal state.\n    However, it's possible to force these transitions manually via the API. While\n    leaving a terminal state, the end time will be unset.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if exiting a final state for a non-final state...\n        if (\n            context.initial_state\n            and context.initial_state.is_final()\n            and not context.proposed_state.is_final()\n        ):\n            # clear the end time\n            context.run.end_time = None\n\n        # if entering a final state...\n        if context.proposed_state.is_final():\n            # if the run has a start time and no end time, give it one\n            if context.run.start_time and not context.run.end_time:\n                context.run.end_time = context.proposed_state.timestamp\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.IncrementRunTime","title":"IncrementRunTime","text":"

    Bases: BaseUniversalTransform

    Records the amount of time a run spends in the running state.

    Source code in prefect/server/orchestration/global_policy.py
    class IncrementRunTime(BaseUniversalTransform):\n    \"\"\"\n    Records the amount of time a run spends in the running state.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if exiting a running state...\n        if context.initial_state and context.initial_state.is_running():\n            # increment the run time by the time spent in the previous state\n            context.run.total_run_time += (\n                context.proposed_state.timestamp - context.initial_state.timestamp\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.IncrementFlowRunCount","title":"IncrementFlowRunCount","text":"

    Bases: BaseUniversalTransform

    Records the number of times a run enters a running state. For use with retries.

    Source code in prefect/server/orchestration/global_policy.py
    class IncrementFlowRunCount(BaseUniversalTransform):\n    \"\"\"\n    Records the number of times a run enters a running state. For use with retries.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if entering a running state...\n        if context.proposed_state.is_running():\n            # do not increment the run count if resuming a paused flow\n            api_version = context.parameters.get(\"api-version\", None)\n            if api_version is None or api_version >= Version(\"0.8.4\"):\n                if context.run.empirical_policy.resuming:\n                    return\n\n            # increment the run count\n            context.run.run_count += 1\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.RemoveResumingIndicator","title":"RemoveResumingIndicator","text":"

    Bases: BaseUniversalTransform

    Removes the indicator on a flow run that marks it as resuming.

    Source code in prefect/server/orchestration/global_policy.py
    class RemoveResumingIndicator(BaseUniversalTransform):\n    \"\"\"\n    Removes the indicator on a flow run that marks it as resuming.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        proposed_state = context.proposed_state\n\n        api_version = context.parameters.get(\"api-version\", None)\n        if api_version is None or api_version >= Version(\"0.8.4\"):\n            if proposed_state.is_running() or proposed_state.is_final():\n                if context.run.empirical_policy.resuming:\n                    updated_policy = context.run.empirical_policy.dict()\n                    updated_policy[\"resuming\"] = False\n                    context.run.empirical_policy = FlowRunPolicy(**updated_policy)\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.IncrementTaskRunCount","title":"IncrementTaskRunCount","text":"

    Bases: BaseUniversalTransform

    Records the number of times a run enters a running state. For use with retries.

    Source code in prefect/server/orchestration/global_policy.py
    class IncrementTaskRunCount(BaseUniversalTransform):\n    \"\"\"\n    Records the number of times a run enters a running state. For use with retries.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if entering a running state...\n        if context.proposed_state.is_running():\n            # increment the run count\n            context.run.run_count += 1\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetExpectedStartTime","title":"SetExpectedStartTime","text":"

    Bases: BaseUniversalTransform

    Estimates the time a state is expected to start running if not set.

    For scheduled states, this estimate is simply the scheduled time. For other states, this is set to the time the proposed state was created by Prefect.

    Source code in prefect/server/orchestration/global_policy.py
    class SetExpectedStartTime(BaseUniversalTransform):\n    \"\"\"\n    Estimates the time a state is expected to start running if not set.\n\n    For scheduled states, this estimate is simply the scheduled time. For other states,\n    this is set to the time the proposed state was created by Prefect.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # set expected start time if this is the first state\n        if not context.run.expected_start_time:\n            if context.proposed_state.is_scheduled():\n                context.run.expected_start_time = (\n                    context.proposed_state.state_details.scheduled_time\n                )\n            else:\n                context.run.expected_start_time = context.proposed_state.timestamp\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetNextScheduledStartTime","title":"SetNextScheduledStartTime","text":"

    Bases: BaseUniversalTransform

    Records the scheduled time on a run.

    When a run enters a scheduled state, run.next_scheduled_start_time is set to the state's scheduled time. When leaving a scheduled state, run.next_scheduled_start_time is unset.

    Source code in prefect/server/orchestration/global_policy.py
    class SetNextScheduledStartTime(BaseUniversalTransform):\n    \"\"\"\n    Records the scheduled time on a run.\n\n    When a run enters a scheduled state, `run.next_scheduled_start_time` is set to\n    the state's scheduled time. When leaving a scheduled state,\n    `run.next_scheduled_start_time` is unset.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # remove the next scheduled start time if exiting a scheduled state\n        if context.initial_state and context.initial_state.is_scheduled():\n            context.run.next_scheduled_start_time = None\n\n        # set next scheduled start time if entering a scheduled state\n        if context.proposed_state.is_scheduled():\n            context.run.next_scheduled_start_time = (\n                context.proposed_state.state_details.scheduled_time\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.UpdateSubflowParentTask","title":"UpdateSubflowParentTask","text":"

    Bases: BaseUniversalTransform

    Whenever a subflow changes state, it must update its parent task run's state.

    Source code in prefect/server/orchestration/global_policy.py
    class UpdateSubflowParentTask(BaseUniversalTransform):\n    \"\"\"\n    Whenever a subflow changes state, it must update its parent task run's state.\n    \"\"\"\n\n    async def after_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # only applies to flow runs with a parent task run id\n        if context.run.parent_task_run_id is not None:\n            # avoid mutation of the flow run state\n            subflow_parent_task_state = context.validated_state.copy(\n                reset_fields=True,\n                include={\n                    \"type\",\n                    \"timestamp\",\n                    \"name\",\n                    \"message\",\n                    \"state_details\",\n                    \"data\",\n                },\n            )\n\n            # set the task's \"child flow run id\" to be the subflow run id\n            subflow_parent_task_state.state_details.child_flow_run_id = context.run.id\n\n            await models.task_runs.set_task_run_state(\n                session=context.session,\n                task_run_id=context.run.parent_task_run_id,\n                state=subflow_parent_task_state,\n                force=True,\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.UpdateSubflowStateDetails","title":"UpdateSubflowStateDetails","text":"

    Bases: BaseUniversalTransform

    Update a child subflow state's references to a corresponding tracking task run id in the parent flow run

    Source code in prefect/server/orchestration/global_policy.py
    class UpdateSubflowStateDetails(BaseUniversalTransform):\n    \"\"\"\n    Update a child subflow state's references to a corresponding tracking task run id\n    in the parent flow run\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # only applies to flow runs with a parent task run id\n        if context.run.parent_task_run_id is not None:\n            context.proposed_state.state_details.task_run_id = (\n                context.run.parent_task_run_id\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.UpdateStateDetails","title":"UpdateStateDetails","text":"

    Bases: BaseUniversalTransform

    Update a state's references to a corresponding flow- or task- run.

    Source code in prefect/server/orchestration/global_policy.py
    class UpdateStateDetails(BaseUniversalTransform):\n    \"\"\"\n    Update a state's references to a corresponding flow- or task- run.\n    \"\"\"\n\n    async def before_transition(\n        self,\n        context: OrchestrationContext,\n    ) -> None:\n        if self.nullified_transition():\n            return\n\n        if isinstance(context, FlowOrchestrationContext):\n            flow_run = await context.flow_run()\n            context.proposed_state.state_details.flow_run_id = flow_run.id\n\n        elif isinstance(context, TaskOrchestrationContext):\n            task_run = await context.task_run()\n            context.proposed_state.state_details.flow_run_id = task_run.flow_run_id\n            context.proposed_state.state_details.task_run_id = task_run.id\n
    "},{"location":"api-ref/server/orchestration/policies/","title":"server.orchestration.policies","text":""},{"location":"api-ref/server/orchestration/policies/#prefect.server.orchestration.policies","title":"prefect.server.orchestration.policies","text":"

    Policies are collections of orchestration rules and transforms.

    Prefect implements (most) orchestration with logic that governs a Prefect flow or task changing state. Policies organize of orchestration logic both to provide an ordering mechanism as well as provide observability into the orchestration process.

    While Prefect's orchestration rules can gracefully run independently of one another, ordering can still have an impact on the observed behavior of the system. For example, it makes no sense to secure a concurrency slot for a run if a cached state exists. Furthermore, policies, provide a mechanism to configure and observe exactly what logic will fire against a transition.

    "},{"location":"api-ref/server/orchestration/policies/#prefect.server.orchestration.policies.BaseOrchestrationPolicy","title":"BaseOrchestrationPolicy","text":"

    Bases: ABC

    An abstract base class used to organize orchestration rules in priority order.

    Different collections of orchestration rules might be used to govern various kinds of transitions. For example, flow-run states and task-run states might require different orchestration logic.

    Source code in prefect/server/orchestration/policies.py
    class BaseOrchestrationPolicy(ABC):\n    \"\"\"\n    An abstract base class used to organize orchestration rules in priority order.\n\n    Different collections of orchestration rules might be used to govern various kinds\n    of transitions. For example, flow-run states and task-run states might require\n    different orchestration logic.\n    \"\"\"\n\n    @staticmethod\n    @abstractmethod\n    def priority():\n        \"\"\"\n        A list of orchestration rules in priority order.\n        \"\"\"\n\n        return []\n\n    @classmethod\n    def compile_transition_rules(cls, from_state=None, to_state=None):\n        \"\"\"\n        Returns rules in policy that are valid for the specified state transition.\n        \"\"\"\n\n        transition_rules = []\n        for rule in cls.priority():\n            if from_state in rule.FROM_STATES and to_state in rule.TO_STATES:\n                transition_rules.append(rule)\n        return transition_rules\n
    "},{"location":"api-ref/server/orchestration/policies/#prefect.server.orchestration.policies.BaseOrchestrationPolicy.priority","title":"priority abstractmethod staticmethod","text":"

    A list of orchestration rules in priority order.

    Source code in prefect/server/orchestration/policies.py
    @staticmethod\n@abstractmethod\ndef priority():\n    \"\"\"\n    A list of orchestration rules in priority order.\n    \"\"\"\n\n    return []\n
    "},{"location":"api-ref/server/orchestration/policies/#prefect.server.orchestration.policies.BaseOrchestrationPolicy.compile_transition_rules","title":"compile_transition_rules classmethod","text":"

    Returns rules in policy that are valid for the specified state transition.

    Source code in prefect/server/orchestration/policies.py
    @classmethod\ndef compile_transition_rules(cls, from_state=None, to_state=None):\n    \"\"\"\n    Returns rules in policy that are valid for the specified state transition.\n    \"\"\"\n\n    transition_rules = []\n    for rule in cls.priority():\n        if from_state in rule.FROM_STATES and to_state in rule.TO_STATES:\n            transition_rules.append(rule)\n    return transition_rules\n
    "},{"location":"api-ref/server/orchestration/rules/","title":"server.orchestration.rules","text":""},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules","title":"prefect.server.orchestration.rules","text":"

    Prefect's flow and task-run orchestration machinery.

    This module contains all the core concepts necessary to implement Prefect's state orchestration engine. These states correspond to intuitive descriptions of all the points that a Prefect flow or task can observe executing user code and intervene, if necessary. A detailed description of states can be found in our concept documentation.

    Prefect's orchestration engine operates under the assumption that no governed user code will execute without first requesting Prefect REST API validate a change in state and record metadata about the run. With all attempts to run user code being checked against a Prefect instance, the Prefect REST API database becomes the unambiguous source of truth for managing the execution of complex interacting workflows. Orchestration rules can be implemented as discrete units of logic that operate against each state transition and can be fully observable, extensible, and customizable -- all without needing to store or parse a single line of user code.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext","title":"OrchestrationContext","text":"

    Bases: PrefectBaseModel

    A container for a state transition, governed by orchestration rules.

    Note

    An OrchestrationContext should not be instantiated directly, instead use the flow- or task- specific subclasses, FlowOrchestrationContext and TaskOrchestrationContext.

    When a flow- or task- run attempts to change state, Prefect REST API has an opportunity to decide whether this transition can proceed. All the relevant information associated with the state transition is stored in an OrchestrationContext, which is subsequently governed by nested orchestration rules implemented using the BaseOrchestrationRule ABC.

    OrchestrationContext introduces the concept of a state being None in the context of an intended state transition. An initial state can be None if a run is is attempting to set a state for the first time. The proposed state might be None if a rule governing the transition determines that no state change should occur at all and nothing is written to the database.

    Attributes:

    Name Type Description session Optional[Union[Session, AsyncSession]]

    a SQLAlchemy database session

    initial_state Optional[State]

    the initial state of a run

    proposed_state Optional[State]

    the proposed state a run is transitioning into

    validated_state Optional[State]

    a proposed state that has committed to the database

    rule_signature List[str]

    a record of rules that have fired on entry into a managed context, currently only used for debugging purposes

    finalization_signature List[str]

    a record of rules that have fired on exit from a managed context, currently only used for debugging purposes

    response_status SetStateStatus

    a SetStateStatus object used to build the API response

    response_details StateResponseDetails

    a StateResponseDetails object use to build the API response

    Parameters:

    Name Type Description Default session

    a SQLAlchemy database session

    required initial_state

    the initial state of a run

    required proposed_state

    the proposed state a run is transitioning into

    required Source code in prefect/server/orchestration/rules.py
    class OrchestrationContext(PrefectBaseModel):\n    \"\"\"\n    A container for a state transition, governed by orchestration rules.\n\n    Note:\n        An `OrchestrationContext` should not be instantiated directly, instead\n        use the flow- or task- specific subclasses, `FlowOrchestrationContext` and\n        `TaskOrchestrationContext`.\n\n    When a flow- or task- run attempts to change state, Prefect REST API has an opportunity\n    to decide whether this transition can proceed. All the relevant information\n    associated with the state transition is stored in an `OrchestrationContext`,\n    which is subsequently governed by nested orchestration rules implemented using\n    the `BaseOrchestrationRule` ABC.\n\n    `OrchestrationContext` introduces the concept of a state being `None` in the\n    context of an intended state transition. An initial state can be `None` if a run\n    is is attempting to set a state for the first time. The proposed state might be\n    `None` if a rule governing the transition determines that no state change\n    should occur at all and nothing is written to the database.\n\n    Attributes:\n        session: a SQLAlchemy database session\n        initial_state: the initial state of a run\n        proposed_state: the proposed state a run is transitioning into\n        validated_state: a proposed state that has committed to the database\n        rule_signature: a record of rules that have fired on entry into a\n            managed context, currently only used for debugging purposes\n        finalization_signature: a record of rules that have fired on exit from a\n            managed context, currently only used for debugging purposes\n        response_status: a SetStateStatus object used to build the API response\n        response_details:a StateResponseDetails object use to build the API response\n\n    Args:\n        session: a SQLAlchemy database session\n        initial_state: the initial state of a run\n        proposed_state: the proposed state a run is transitioning into\n    \"\"\"\n\n    class Config:\n        arbitrary_types_allowed = True\n\n    session: Optional[Union[sa.orm.Session, AsyncSession]] = ...\n    initial_state: Optional[states.State] = ...\n    proposed_state: Optional[states.State] = ...\n    validated_state: Optional[states.State]\n    rule_signature: List[str] = Field(default_factory=list)\n    finalization_signature: List[str] = Field(default_factory=list)\n    response_status: SetStateStatus = Field(default=SetStateStatus.ACCEPT)\n    response_details: StateResponseDetails = Field(default_factory=StateAcceptDetails)\n    orchestration_error: Optional[Exception] = Field(default=None)\n    parameters: Dict[Any, Any] = Field(default_factory=dict)\n\n    @property\n    def initial_state_type(self) -> Optional[states.StateType]:\n        \"\"\"The state type of `self.initial_state` if it exists.\"\"\"\n\n        return self.initial_state.type if self.initial_state else None\n\n    @property\n    def proposed_state_type(self) -> Optional[states.StateType]:\n        \"\"\"The state type of `self.proposed_state` if it exists.\"\"\"\n\n        return self.proposed_state.type if self.proposed_state else None\n\n    @property\n    def validated_state_type(self) -> Optional[states.StateType]:\n        \"\"\"The state type of `self.validated_state` if it exists.\"\"\"\n        return self.validated_state.type if self.validated_state else None\n\n    def safe_copy(self):\n        \"\"\"\n        Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n        Orchestration rules govern state transitions using information stored in\n        an `OrchestrationContext`. However, mutating objects stored on the context\n        directly can have unintended side-effects. To guard against this,\n        `self.safe_copy` can be used to pass information to orchestration rules\n        without risking mutation.\n\n        Returns:\n            A mutation-safe copy of the `OrchestrationContext`\n        \"\"\"\n\n        safe_copy = self.copy()\n\n        safe_copy.initial_state = (\n            self.initial_state.copy() if self.initial_state else None\n        )\n        safe_copy.proposed_state = (\n            self.proposed_state.copy() if self.proposed_state else None\n        )\n        safe_copy.validated_state = (\n            self.validated_state.copy() if self.validated_state else None\n        )\n        safe_copy.parameters = self.parameters.copy()\n        return safe_copy\n\n    def entry_context(self):\n        \"\"\"\n        A convenience method that generates input parameters for orchestration rules.\n\n        An `OrchestrationContext` defines a state transition that is managed by\n        orchestration rules which can fire hooks before a transition has been committed\n        to the database. These hooks have a consistent interface which can be generated\n        with this method.\n        \"\"\"\n\n        safe_context = self.safe_copy()\n        return safe_context.initial_state, safe_context.proposed_state, safe_context\n\n    def exit_context(self):\n        \"\"\"\n        A convenience method that generates input parameters for orchestration rules.\n\n        An `OrchestrationContext` defines a state transition that is managed by\n        orchestration rules which can fire hooks after a transition has been committed\n        to the database. These hooks have a consistent interface which can be generated\n        with this method.\n        \"\"\"\n\n        safe_context = self.safe_copy()\n        return safe_context.initial_state, safe_context.validated_state, safe_context\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.initial_state_type","title":"initial_state_type: Optional[states.StateType] property","text":"

    The state type of self.initial_state if it exists.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.proposed_state_type","title":"proposed_state_type: Optional[states.StateType] property","text":"

    The state type of self.proposed_state if it exists.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.validated_state_type","title":"validated_state_type: Optional[states.StateType] property","text":"

    The state type of self.validated_state if it exists.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.safe_copy","title":"safe_copy","text":"

    Creates a mostly-mutation-safe copy for use in orchestration rules.

    Orchestration rules govern state transitions using information stored in an OrchestrationContext. However, mutating objects stored on the context directly can have unintended side-effects. To guard against this, self.safe_copy can be used to pass information to orchestration rules without risking mutation.

    Returns:

    Type Description

    A mutation-safe copy of the OrchestrationContext

    Source code in prefect/server/orchestration/rules.py
    def safe_copy(self):\n    \"\"\"\n    Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n    Orchestration rules govern state transitions using information stored in\n    an `OrchestrationContext`. However, mutating objects stored on the context\n    directly can have unintended side-effects. To guard against this,\n    `self.safe_copy` can be used to pass information to orchestration rules\n    without risking mutation.\n\n    Returns:\n        A mutation-safe copy of the `OrchestrationContext`\n    \"\"\"\n\n    safe_copy = self.copy()\n\n    safe_copy.initial_state = (\n        self.initial_state.copy() if self.initial_state else None\n    )\n    safe_copy.proposed_state = (\n        self.proposed_state.copy() if self.proposed_state else None\n    )\n    safe_copy.validated_state = (\n        self.validated_state.copy() if self.validated_state else None\n    )\n    safe_copy.parameters = self.parameters.copy()\n    return safe_copy\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.entry_context","title":"entry_context","text":"

    A convenience method that generates input parameters for orchestration rules.

    An OrchestrationContext defines a state transition that is managed by orchestration rules which can fire hooks before a transition has been committed to the database. These hooks have a consistent interface which can be generated with this method.

    Source code in prefect/server/orchestration/rules.py
    def entry_context(self):\n    \"\"\"\n    A convenience method that generates input parameters for orchestration rules.\n\n    An `OrchestrationContext` defines a state transition that is managed by\n    orchestration rules which can fire hooks before a transition has been committed\n    to the database. These hooks have a consistent interface which can be generated\n    with this method.\n    \"\"\"\n\n    safe_context = self.safe_copy()\n    return safe_context.initial_state, safe_context.proposed_state, safe_context\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.exit_context","title":"exit_context","text":"

    A convenience method that generates input parameters for orchestration rules.

    An OrchestrationContext defines a state transition that is managed by orchestration rules which can fire hooks after a transition has been committed to the database. These hooks have a consistent interface which can be generated with this method.

    Source code in prefect/server/orchestration/rules.py
    def exit_context(self):\n    \"\"\"\n    A convenience method that generates input parameters for orchestration rules.\n\n    An `OrchestrationContext` defines a state transition that is managed by\n    orchestration rules which can fire hooks after a transition has been committed\n    to the database. These hooks have a consistent interface which can be generated\n    with this method.\n    \"\"\"\n\n    safe_context = self.safe_copy()\n    return safe_context.initial_state, safe_context.validated_state, safe_context\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.FlowOrchestrationContext","title":"FlowOrchestrationContext","text":"

    Bases: OrchestrationContext

    A container for a flow run state transition, governed by orchestration rules.

    When a flow- run attempts to change state, Prefect REST API has an opportunity to decide whether this transition can proceed. All the relevant information associated with the state transition is stored in an OrchestrationContext, which is subsequently governed by nested orchestration rules implemented using the BaseOrchestrationRule ABC.

    FlowOrchestrationContext introduces the concept of a state being None in the context of an intended state transition. An initial state can be None if a run is is attempting to set a state for the first time. The proposed state might be None if a rule governing the transition determines that no state change should occur at all and nothing is written to the database.

    Attributes:

    Name Type Description session

    a SQLAlchemy database session

    run Any

    the flow run attempting to change state

    initial_state Any

    the initial state of the run

    proposed_state Any

    the proposed state the run is transitioning into

    validated_state Any

    a proposed state that has committed to the database

    rule_signature Any

    a record of rules that have fired on entry into a managed context, currently only used for debugging purposes

    finalization_signature Any

    a record of rules that have fired on exit from a managed context, currently only used for debugging purposes

    response_status Any

    a SetStateStatus object used to build the API response

    response_details Any

    a StateResponseDetails object use to build the API response

    Parameters:

    Name Type Description Default session

    a SQLAlchemy database session

    required run

    the flow run attempting to change state

    required initial_state

    the initial state of a run

    required proposed_state

    the proposed state a run is transitioning into

    required Source code in prefect/server/orchestration/rules.py
    class FlowOrchestrationContext(OrchestrationContext):\n    \"\"\"\n    A container for a flow run state transition, governed by orchestration rules.\n\n    When a flow- run attempts to change state, Prefect REST API has an opportunity\n    to decide whether this transition can proceed. All the relevant information\n    associated with the state transition is stored in an `OrchestrationContext`,\n    which is subsequently governed by nested orchestration rules implemented using\n    the `BaseOrchestrationRule` ABC.\n\n    `FlowOrchestrationContext` introduces the concept of a state being `None` in the\n    context of an intended state transition. An initial state can be `None` if a run\n    is is attempting to set a state for the first time. The proposed state might be\n    `None` if a rule governing the transition determines that no state change\n    should occur at all and nothing is written to the database.\n\n    Attributes:\n        session: a SQLAlchemy database session\n        run: the flow run attempting to change state\n        initial_state: the initial state of the run\n        proposed_state: the proposed state the run is transitioning into\n        validated_state: a proposed state that has committed to the database\n        rule_signature: a record of rules that have fired on entry into a\n            managed context, currently only used for debugging purposes\n        finalization_signature: a record of rules that have fired on exit from a\n            managed context, currently only used for debugging purposes\n        response_status: a SetStateStatus object used to build the API response\n        response_details:a StateResponseDetails object use to build the API response\n\n    Args:\n        session: a SQLAlchemy database session\n        run: the flow run attempting to change state\n        initial_state: the initial state of a run\n        proposed_state: the proposed state a run is transitioning into\n    \"\"\"\n\n    # run: db.FlowRun = ...\n    run: Any = ...\n\n    @inject_db\n    async def validate_proposed_state(\n        self,\n        db: PrefectDBInterface,\n    ):\n        \"\"\"\n        Validates a proposed state by committing it to the database.\n\n        After the `FlowOrchestrationContext` is governed by orchestration rules, the\n        proposed state can be validated: the proposed state is added to the current\n        SQLAlchemy session and is flushed. `self.validated_state` set to the flushed\n        state. The state on the run is set to the validated state as well.\n\n        If the proposed state is `None` when this method is called, no state will be\n        written and `self.validated_state` will be set to the run's current state.\n\n        Returns:\n            None\n        \"\"\"\n        # (circular import)\n        from prefect.server.api.server import is_client_retryable_exception\n\n        try:\n            await self._validate_proposed_state()\n            return\n        except Exception as exc:\n            logger.exception(\"Encountered error during state validation\")\n            self.proposed_state = None\n\n            if is_client_retryable_exception(exc):\n                # Do not capture retryable database exceptions, this exception will be\n                # raised as a 503 in the API layer\n                raise\n\n            reason = f\"Error validating state: {exc!r}\"\n            self.response_status = SetStateStatus.ABORT\n            self.response_details = StateAbortDetails(reason=reason)\n\n    @inject_db\n    async def _validate_proposed_state(\n        self,\n        db: PrefectDBInterface,\n    ):\n        if self.proposed_state is None:\n            validated_orm_state = self.run.state\n            # We cannot access `self.run.state.data` directly for unknown reasons\n            state_data = (\n                (\n                    await artifacts.read_artifact(\n                        self.session, self.run.state.result_artifact_id\n                    )\n                ).data\n                if self.run.state.result_artifact_id\n                else None\n            )\n        else:\n            state_payload = self.proposed_state.dict(shallow=True)\n            state_data = state_payload.pop(\"data\", None)\n\n            if state_data is not None:\n                state_result_artifact = core.Artifact.from_result(state_data)\n                state_result_artifact.flow_run_id = self.run.id\n                await artifacts.create_artifact(self.session, state_result_artifact)\n                state_payload[\"result_artifact_id\"] = state_result_artifact.id\n\n            validated_orm_state = db.FlowRunState(\n                flow_run_id=self.run.id,\n                **state_payload,\n            )\n\n        self.session.add(validated_orm_state)\n        self.run.set_state(validated_orm_state)\n\n        await self.session.flush()\n        if validated_orm_state:\n            self.validated_state = states.State.from_orm_without_result(\n                validated_orm_state, with_data=state_data\n            )\n        else:\n            self.validated_state = None\n\n    def safe_copy(self):\n        \"\"\"\n        Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n        Orchestration rules govern state transitions using information stored in\n        an `OrchestrationContext`. However, mutating objects stored on the context\n        directly can have unintended side-effects. To guard against this,\n        `self.safe_copy` can be used to pass information to orchestration rules\n        without risking mutation.\n\n        Note:\n            `self.run` is an ORM model, and even when copied is unsafe to mutate\n\n        Returns:\n            A mutation-safe copy of `FlowOrchestrationContext`\n        \"\"\"\n\n        return super().safe_copy()\n\n    @property\n    def run_settings(self) -> Dict:\n        \"\"\"Run-level settings used to orchestrate the state transition.\"\"\"\n\n        return self.run.empirical_policy\n\n    async def task_run(self):\n        return None\n\n    async def flow_run(self):\n        return self.run\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.FlowOrchestrationContext.run_settings","title":"run_settings: Dict property","text":"

    Run-level settings used to orchestrate the state transition.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.FlowOrchestrationContext.validate_proposed_state","title":"validate_proposed_state async","text":"

    Validates a proposed state by committing it to the database.

    After the FlowOrchestrationContext is governed by orchestration rules, the proposed state can be validated: the proposed state is added to the current SQLAlchemy session and is flushed. self.validated_state set to the flushed state. The state on the run is set to the validated state as well.

    If the proposed state is None when this method is called, no state will be written and self.validated_state will be set to the run's current state.

    Returns:

    Type Description

    None

    Source code in prefect/server/orchestration/rules.py
    @inject_db\nasync def validate_proposed_state(\n    self,\n    db: PrefectDBInterface,\n):\n    \"\"\"\n    Validates a proposed state by committing it to the database.\n\n    After the `FlowOrchestrationContext` is governed by orchestration rules, the\n    proposed state can be validated: the proposed state is added to the current\n    SQLAlchemy session and is flushed. `self.validated_state` set to the flushed\n    state. The state on the run is set to the validated state as well.\n\n    If the proposed state is `None` when this method is called, no state will be\n    written and `self.validated_state` will be set to the run's current state.\n\n    Returns:\n        None\n    \"\"\"\n    # (circular import)\n    from prefect.server.api.server import is_client_retryable_exception\n\n    try:\n        await self._validate_proposed_state()\n        return\n    except Exception as exc:\n        logger.exception(\"Encountered error during state validation\")\n        self.proposed_state = None\n\n        if is_client_retryable_exception(exc):\n            # Do not capture retryable database exceptions, this exception will be\n            # raised as a 503 in the API layer\n            raise\n\n        reason = f\"Error validating state: {exc!r}\"\n        self.response_status = SetStateStatus.ABORT\n        self.response_details = StateAbortDetails(reason=reason)\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.FlowOrchestrationContext.safe_copy","title":"safe_copy","text":"

    Creates a mostly-mutation-safe copy for use in orchestration rules.

    Orchestration rules govern state transitions using information stored in an OrchestrationContext. However, mutating objects stored on the context directly can have unintended side-effects. To guard against this, self.safe_copy can be used to pass information to orchestration rules without risking mutation.

    Note

    self.run is an ORM model, and even when copied is unsafe to mutate

    Returns:

    Type Description

    A mutation-safe copy of FlowOrchestrationContext

    Source code in prefect/server/orchestration/rules.py
    def safe_copy(self):\n    \"\"\"\n    Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n    Orchestration rules govern state transitions using information stored in\n    an `OrchestrationContext`. However, mutating objects stored on the context\n    directly can have unintended side-effects. To guard against this,\n    `self.safe_copy` can be used to pass information to orchestration rules\n    without risking mutation.\n\n    Note:\n        `self.run` is an ORM model, and even when copied is unsafe to mutate\n\n    Returns:\n        A mutation-safe copy of `FlowOrchestrationContext`\n    \"\"\"\n\n    return super().safe_copy()\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.TaskOrchestrationContext","title":"TaskOrchestrationContext","text":"

    Bases: OrchestrationContext

    A container for a task run state transition, governed by orchestration rules.

    When a task- run attempts to change state, Prefect REST API has an opportunity to decide whether this transition can proceed. All the relevant information associated with the state transition is stored in an OrchestrationContext, which is subsequently governed by nested orchestration rules implemented using the BaseOrchestrationRule ABC.

    TaskOrchestrationContext introduces the concept of a state being None in the context of an intended state transition. An initial state can be None if a run is is attempting to set a state for the first time. The proposed state might be None if a rule governing the transition determines that no state change should occur at all and nothing is written to the database.

    Attributes:

    Name Type Description session

    a SQLAlchemy database session

    run Any

    the task run attempting to change state

    initial_state Any

    the initial state of the run

    proposed_state Any

    the proposed state the run is transitioning into

    validated_state Any

    a proposed state that has committed to the database

    rule_signature Any

    a record of rules that have fired on entry into a managed context, currently only used for debugging purposes

    finalization_signature Any

    a record of rules that have fired on exit from a managed context, currently only used for debugging purposes

    response_status Any

    a SetStateStatus object used to build the API response

    response_details Any

    a StateResponseDetails object use to build the API response

    Parameters:

    Name Type Description Default session

    a SQLAlchemy database session

    required run

    the task run attempting to change state

    required initial_state

    the initial state of a run

    required proposed_state

    the proposed state a run is transitioning into

    required Source code in prefect/server/orchestration/rules.py
    class TaskOrchestrationContext(OrchestrationContext):\n    \"\"\"\n    A container for a task run state transition, governed by orchestration rules.\n\n    When a task- run attempts to change state, Prefect REST API has an opportunity\n    to decide whether this transition can proceed. All the relevant information\n    associated with the state transition is stored in an `OrchestrationContext`,\n    which is subsequently governed by nested orchestration rules implemented using\n    the `BaseOrchestrationRule` ABC.\n\n    `TaskOrchestrationContext` introduces the concept of a state being `None` in the\n    context of an intended state transition. An initial state can be `None` if a run\n    is is attempting to set a state for the first time. The proposed state might be\n    `None` if a rule governing the transition determines that no state change\n    should occur at all and nothing is written to the database.\n\n    Attributes:\n        session: a SQLAlchemy database session\n        run: the task run attempting to change state\n        initial_state: the initial state of the run\n        proposed_state: the proposed state the run is transitioning into\n        validated_state: a proposed state that has committed to the database\n        rule_signature: a record of rules that have fired on entry into a\n            managed context, currently only used for debugging purposes\n        finalization_signature: a record of rules that have fired on exit from a\n            managed context, currently only used for debugging purposes\n        response_status: a SetStateStatus object used to build the API response\n        response_details:a StateResponseDetails object use to build the API response\n\n    Args:\n        session: a SQLAlchemy database session\n        run: the task run attempting to change state\n        initial_state: the initial state of a run\n        proposed_state: the proposed state a run is transitioning into\n    \"\"\"\n\n    # run: db.TaskRun = ...\n    run: Any = ...\n\n    @inject_db\n    async def validate_proposed_state(\n        self,\n        db: PrefectDBInterface,\n    ):\n        \"\"\"\n        Validates a proposed state by committing it to the database.\n\n        After the `TaskOrchestrationContext` is governed by orchestration rules, the\n        proposed state can be validated: the proposed state is added to the current\n        SQLAlchemy session and is flushed. `self.validated_state` set to the flushed\n        state. The state on the run is set to the validated state as well.\n\n        If the proposed state is `None` when this method is called, no state will be\n        written and `self.validated_state` will be set to the run's current state.\n\n        Returns:\n            None\n        \"\"\"\n        # (circular import)\n        from prefect.server.api.server import is_client_retryable_exception\n\n        try:\n            await self._validate_proposed_state()\n            return\n        except Exception as exc:\n            logger.exception(\"Encountered error during state validation\")\n            self.proposed_state = None\n\n            if is_client_retryable_exception(exc):\n                # Do not capture retryable database exceptions, this exception will be\n                # raised as a 503 in the API layer\n                raise\n\n            reason = f\"Error validating state: {exc!r}\"\n            self.response_status = SetStateStatus.ABORT\n            self.response_details = StateAbortDetails(reason=reason)\n\n    @inject_db\n    async def _validate_proposed_state(\n        self,\n        db: PrefectDBInterface,\n    ):\n        if self.proposed_state is None:\n            validated_orm_state = self.run.state\n            # We cannot access `self.run.state.data` directly for unknown reasons\n            state_data = (\n                (\n                    await artifacts.read_artifact(\n                        self.session, self.run.state.result_artifact_id\n                    )\n                ).data\n                if self.run.state.result_artifact_id\n                else None\n            )\n        else:\n            state_payload = self.proposed_state.dict(shallow=True)\n            state_data = state_payload.pop(\"data\", None)\n\n            if state_data is not None:\n                state_result_artifact = core.Artifact.from_result(state_data)\n                state_result_artifact.task_run_id = self.run.id\n\n                if self.run.flow_run_id is not None:\n                    flow_run = await self.flow_run()\n                    state_result_artifact.flow_run_id = flow_run.id\n\n                await artifacts.create_artifact(self.session, state_result_artifact)\n                state_payload[\"result_artifact_id\"] = state_result_artifact.id\n\n            validated_orm_state = db.TaskRunState(\n                task_run_id=self.run.id,\n                **state_payload,\n            )\n\n        self.session.add(validated_orm_state)\n        self.run.set_state(validated_orm_state)\n\n        await self.session.flush()\n        if validated_orm_state:\n            self.validated_state = states.State.from_orm_without_result(\n                validated_orm_state, with_data=state_data\n            )\n        else:\n            self.validated_state = None\n\n    def safe_copy(self):\n        \"\"\"\n        Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n        Orchestration rules govern state transitions using information stored in\n        an `OrchestrationContext`. However, mutating objects stored on the context\n        directly can have unintended side-effects. To guard against this,\n        `self.safe_copy` can be used to pass information to orchestration rules\n        without risking mutation.\n\n        Note:\n            `self.run` is an ORM model, and even when copied is unsafe to mutate\n\n        Returns:\n            A mutation-safe copy of `TaskOrchestrationContext`\n        \"\"\"\n\n        return super().safe_copy()\n\n    @property\n    def run_settings(self) -> Dict:\n        \"\"\"Run-level settings used to orchestrate the state transition.\"\"\"\n\n        return self.run.empirical_policy\n\n    async def task_run(self):\n        return self.run\n\n    async def flow_run(self):\n        return await flow_runs.read_flow_run(\n            session=self.session,\n            flow_run_id=self.run.flow_run_id,\n        )\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.TaskOrchestrationContext.run_settings","title":"run_settings: Dict property","text":"

    Run-level settings used to orchestrate the state transition.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.TaskOrchestrationContext.validate_proposed_state","title":"validate_proposed_state async","text":"

    Validates a proposed state by committing it to the database.

    After the TaskOrchestrationContext is governed by orchestration rules, the proposed state can be validated: the proposed state is added to the current SQLAlchemy session and is flushed. self.validated_state set to the flushed state. The state on the run is set to the validated state as well.

    If the proposed state is None when this method is called, no state will be written and self.validated_state will be set to the run's current state.

    Returns:

    Type Description

    None

    Source code in prefect/server/orchestration/rules.py
    @inject_db\nasync def validate_proposed_state(\n    self,\n    db: PrefectDBInterface,\n):\n    \"\"\"\n    Validates a proposed state by committing it to the database.\n\n    After the `TaskOrchestrationContext` is governed by orchestration rules, the\n    proposed state can be validated: the proposed state is added to the current\n    SQLAlchemy session and is flushed. `self.validated_state` set to the flushed\n    state. The state on the run is set to the validated state as well.\n\n    If the proposed state is `None` when this method is called, no state will be\n    written and `self.validated_state` will be set to the run's current state.\n\n    Returns:\n        None\n    \"\"\"\n    # (circular import)\n    from prefect.server.api.server import is_client_retryable_exception\n\n    try:\n        await self._validate_proposed_state()\n        return\n    except Exception as exc:\n        logger.exception(\"Encountered error during state validation\")\n        self.proposed_state = None\n\n        if is_client_retryable_exception(exc):\n            # Do not capture retryable database exceptions, this exception will be\n            # raised as a 503 in the API layer\n            raise\n\n        reason = f\"Error validating state: {exc!r}\"\n        self.response_status = SetStateStatus.ABORT\n        self.response_details = StateAbortDetails(reason=reason)\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.TaskOrchestrationContext.safe_copy","title":"safe_copy","text":"

    Creates a mostly-mutation-safe copy for use in orchestration rules.

    Orchestration rules govern state transitions using information stored in an OrchestrationContext. However, mutating objects stored on the context directly can have unintended side-effects. To guard against this, self.safe_copy can be used to pass information to orchestration rules without risking mutation.

    Note

    self.run is an ORM model, and even when copied is unsafe to mutate

    Returns:

    Type Description

    A mutation-safe copy of TaskOrchestrationContext

    Source code in prefect/server/orchestration/rules.py
    def safe_copy(self):\n    \"\"\"\n    Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n    Orchestration rules govern state transitions using information stored in\n    an `OrchestrationContext`. However, mutating objects stored on the context\n    directly can have unintended side-effects. To guard against this,\n    `self.safe_copy` can be used to pass information to orchestration rules\n    without risking mutation.\n\n    Note:\n        `self.run` is an ORM model, and even when copied is unsafe to mutate\n\n    Returns:\n        A mutation-safe copy of `TaskOrchestrationContext`\n    \"\"\"\n\n    return super().safe_copy()\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule","title":"BaseOrchestrationRule","text":"

    Bases: AbstractAsyncContextManager

    An abstract base class used to implement a discrete piece of orchestration logic.

    An OrchestrationRule is a stateful context manager that directly governs a state transition. Complex orchestration is achieved by nesting multiple rules. Each rule runs against an OrchestrationContext that contains the transition details; this context is then passed to subsequent rules. The context can be modified by hooks that fire before and after a new state is validated and committed to the database. These hooks will fire as long as the state transition is considered \"valid\" and govern a transition by either modifying the proposed state before it is validated or by producing a side-effect.

    A state transition occurs whenever a flow- or task- run changes state, prompting Prefect REST API to decide whether or not this transition can proceed. The current state of the run is referred to as the \"initial state\", and the state a run is attempting to transition into is the \"proposed state\". Together, the initial state transitioning into the proposed state is the intended transition that is governed by these orchestration rules. After using rules to enter a runtime context, the OrchestrationContext will contain a proposed state that has been governed by each rule, and at that point can validate the proposed state and commit it to the database. The validated state will be set on the context as context.validated_state, and rules will call the self.after_transition hook upon exiting the managed context.

    Examples:

    Create a rule:\n\n>>> class BasicRule(BaseOrchestrationRule):\n>>>     # allowed initial state types\n>>>     FROM_STATES = [StateType.RUNNING]\n>>>     # allowed proposed state types\n>>>     TO_STATES = [StateType.COMPLETED, StateType.FAILED]\n>>>\n>>>     async def before_transition(initial_state, proposed_state, ctx):\n>>>         # side effects and proposed state mutation can happen here\n>>>         ...\n>>>\n>>>     async def after_transition(initial_state, validated_state, ctx):\n>>>         # operations on states that have been validated can happen here\n>>>         ...\n>>>\n>>>     async def cleanup(intitial_state, validated_state, ctx):\n>>>         # reverts side effects generated by `before_transition` if necessary\n>>>         ...\n\nUse a rule:\n\n>>> intended_transition = (StateType.RUNNING, StateType.COMPLETED)\n>>> async with BasicRule(context, *intended_transition):\n>>>     # context.proposed_state has been governed by BasicRule\n>>>     ...\n\nUse multiple rules:\n\n>>> rules = [BasicRule, BasicRule]\n>>> intended_transition = (StateType.RUNNING, StateType.COMPLETED)\n>>> async with contextlib.AsyncExitStack() as stack:\n>>>     for rule in rules:\n>>>         stack.enter_async_context(rule(context, *intended_transition))\n>>>\n>>>     # context.proposed_state has been governed by all rules\n>>>     ...\n

    Attributes:

    Name Type Description FROM_STATES Iterable

    list of valid initial state types this rule governs

    TO_STATES Iterable

    list of valid proposed state types this rule governs

    context

    the orchestration context

    from_state_type

    the state type a run is currently in

    to_state_type

    the intended proposed state type prior to any orchestration

    Parameters:

    Name Type Description Default context OrchestrationContext

    A FlowOrchestrationContext or TaskOrchestrationContext that is passed between rules

    required from_state_type Optional[StateType]

    The state type of the initial state of a run, if this state type is not contained in FROM_STATES, no hooks will fire

    required to_state_type Optional[StateType]

    The state type of the proposed state before orchestration, if this state type is not contained in TO_STATES, no hooks will fire

    required Source code in prefect/server/orchestration/rules.py
    class BaseOrchestrationRule(contextlib.AbstractAsyncContextManager):\n    \"\"\"\n    An abstract base class used to implement a discrete piece of orchestration logic.\n\n    An `OrchestrationRule` is a stateful context manager that directly governs a state\n    transition. Complex orchestration is achieved by nesting multiple rules.\n    Each rule runs against an `OrchestrationContext` that contains the transition\n    details; this context is then passed to subsequent rules. The context can be\n    modified by hooks that fire before and after a new state is validated and committed\n    to the database. These hooks will fire as long as the state transition is\n    considered \"valid\" and govern a transition by either modifying the proposed state\n    before it is validated or by producing a side-effect.\n\n    A state transition occurs whenever a flow- or task- run changes state, prompting\n    Prefect REST API to decide whether or not this transition can proceed. The current state of\n    the run is referred to as the \"initial state\", and the state a run is\n    attempting to transition into is the \"proposed state\". Together, the initial state\n    transitioning into the proposed state is the intended transition that is governed\n    by these orchestration rules. After using rules to enter a runtime context, the\n    `OrchestrationContext` will contain a proposed state that has been governed by\n    each rule, and at that point can validate the proposed state and commit it to\n    the database. The validated state will be set on the context as\n    `context.validated_state`, and rules will call the `self.after_transition` hook\n    upon exiting the managed context.\n\n    Examples:\n\n        Create a rule:\n\n        >>> class BasicRule(BaseOrchestrationRule):\n        >>>     # allowed initial state types\n        >>>     FROM_STATES = [StateType.RUNNING]\n        >>>     # allowed proposed state types\n        >>>     TO_STATES = [StateType.COMPLETED, StateType.FAILED]\n        >>>\n        >>>     async def before_transition(initial_state, proposed_state, ctx):\n        >>>         # side effects and proposed state mutation can happen here\n        >>>         ...\n        >>>\n        >>>     async def after_transition(initial_state, validated_state, ctx):\n        >>>         # operations on states that have been validated can happen here\n        >>>         ...\n        >>>\n        >>>     async def cleanup(intitial_state, validated_state, ctx):\n        >>>         # reverts side effects generated by `before_transition` if necessary\n        >>>         ...\n\n        Use a rule:\n\n        >>> intended_transition = (StateType.RUNNING, StateType.COMPLETED)\n        >>> async with BasicRule(context, *intended_transition):\n        >>>     # context.proposed_state has been governed by BasicRule\n        >>>     ...\n\n        Use multiple rules:\n\n        >>> rules = [BasicRule, BasicRule]\n        >>> intended_transition = (StateType.RUNNING, StateType.COMPLETED)\n        >>> async with contextlib.AsyncExitStack() as stack:\n        >>>     for rule in rules:\n        >>>         stack.enter_async_context(rule(context, *intended_transition))\n        >>>\n        >>>     # context.proposed_state has been governed by all rules\n        >>>     ...\n\n    Attributes:\n        FROM_STATES: list of valid initial state types this rule governs\n        TO_STATES: list of valid proposed state types this rule governs\n        context: the orchestration context\n        from_state_type: the state type a run is currently in\n        to_state_type: the intended proposed state type prior to any orchestration\n\n    Args:\n        context: A `FlowOrchestrationContext` or `TaskOrchestrationContext` that is\n            passed between rules\n        from_state_type: The state type of the initial state of a run, if this\n            state type is not contained in `FROM_STATES`, no hooks will fire\n        to_state_type: The state type of the proposed state before orchestration, if\n            this state type is not contained in `TO_STATES`, no hooks will fire\n    \"\"\"\n\n    FROM_STATES: Iterable = []\n    TO_STATES: Iterable = []\n\n    def __init__(\n        self,\n        context: OrchestrationContext,\n        from_state_type: Optional[states.StateType],\n        to_state_type: Optional[states.StateType],\n    ):\n        self.context = context\n        self.from_state_type = from_state_type\n        self.to_state_type = to_state_type\n        self._invalid_on_entry = None\n\n    async def __aenter__(self) -> OrchestrationContext:\n        \"\"\"\n        Enter an async runtime context governed by this rule.\n\n        The `with` statement will bind a governed `OrchestrationContext` to the target\n        specified by the `as` clause. If the transition proposed by the\n        `OrchestrationContext` is considered invalid on entry, entering this context\n        will do nothing. Otherwise, `self.before_transition` will fire.\n        \"\"\"\n\n        if await self.invalid():\n            pass\n        else:\n            try:\n                entry_context = self.context.entry_context()\n                await self.before_transition(*entry_context)\n                self.context.rule_signature.append(str(self.__class__))\n            except Exception as before_transition_error:\n                reason = (\n                    f\"Aborting orchestration due to error in {self.__class__!r}:\"\n                    f\" !{before_transition_error!r}\"\n                )\n                logger.exception(\n                    f\"Error running before-transition hook in rule {self.__class__!r}:\"\n                    f\" !{before_transition_error!r}\"\n                )\n\n                self.context.proposed_state = None\n                self.context.response_status = SetStateStatus.ABORT\n                self.context.response_details = StateAbortDetails(reason=reason)\n                self.context.orchestration_error = before_transition_error\n\n        return self.context\n\n    async def __aexit__(\n        self,\n        exc_type: Optional[Type[BaseException]],\n        exc_val: Optional[BaseException],\n        exc_tb: Optional[TracebackType],\n    ) -> None:\n        \"\"\"\n        Exit the async runtime context governed by this rule.\n\n        One of three outcomes can happen upon exiting this rule's context depending on\n        the state of the rule. If the rule was found to be invalid on entry, nothing\n        happens. If the rule was valid on entry and continues to be valid on exit,\n        `self.after_transition` will fire. If the rule was valid on entry but invalid\n        on exit, the rule will \"fizzle\" and `self.cleanup` will fire in order to revert\n        any side-effects produced by `self.before_transition`.\n        \"\"\"\n\n        exit_context = self.context.exit_context()\n        if await self.invalid():\n            pass\n        elif await self.fizzled():\n            await self.cleanup(*exit_context)\n        else:\n            await self.after_transition(*exit_context)\n            self.context.finalization_signature.append(str(self.__class__))\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        \"\"\"\n        Implements a hook that can fire before a state is committed to the database.\n\n        This hook may produce side-effects or mutate the proposed state of a\n        transition using one of four methods: `self.reject_transition`,\n        `self.delay_transition`, `self.abort_transition`, and `self.rename_state`.\n\n        Note:\n            As currently implemented, the `before_transition` hook is not\n            perfectly isolated from mutating the transition. It is a standard instance\n            method that has access to `self`, and therefore `self.context`. This should\n            never be modified directly. Furthermore, `context.run` is an ORM model, and\n            mutating the run can also cause unintended writes to the database.\n\n        Args:\n            initial_state: The initial state of a transition\n            proposed_state: The proposed state of a transition\n            context: A safe copy of the `OrchestrationContext`, with the exception of\n                `context.run`, mutating this context will have no effect on the broader\n                orchestration environment.\n\n        Returns:\n            None\n        \"\"\"\n\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        \"\"\"\n        Implements a hook that can fire after a state is committed to the database.\n\n        Args:\n            initial_state: The initial state of a transition\n            validated_state: The governed state that has been committed to the database\n            context: A safe copy of the `OrchestrationContext`, with the exception of\n                `context.run`, mutating this context will have no effect on the broader\n                orchestration environment.\n\n        Returns:\n            None\n        \"\"\"\n\n    async def cleanup(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        \"\"\"\n        Implements a hook that can fire after a state is committed to the database.\n\n        The intended use of this method is to revert side-effects produced by\n        `self.before_transition` when the transition is found to be invalid on exit.\n        This allows multiple rules to be gracefully run in sequence, without logic that\n        keeps track of all other rules that might govern a transition.\n\n        Args:\n            initial_state: The initial state of a transition\n            validated_state: The governed state that has been committed to the database\n            context: A safe copy of the `OrchestrationContext`, with the exception of\n                `context.run`, mutating this context will have no effect on the broader\n                orchestration environment.\n\n        Returns:\n            None\n        \"\"\"\n\n    async def invalid(self) -> bool:\n        \"\"\"\n        Determines if a rule is invalid.\n\n        Invalid rules do nothing and no hooks fire upon entering or exiting a governed\n        context. Rules are invalid if the transition states types are not contained in\n        `self.FROM_STATES` and `self.TO_STATES`, or if the context is proposing\n        a transition that differs from the transition the rule was instantiated with.\n\n        Returns:\n            True if the rules in invalid, False otherwise.\n        \"\"\"\n        # invalid and fizzled states are mutually exclusive,\n        # `_invalid_on_entry` holds this statefulness\n        if self.from_state_type not in self.FROM_STATES:\n            self._invalid_on_entry = True\n        if self.to_state_type not in self.TO_STATES:\n            self._invalid_on_entry = True\n\n        if self._invalid_on_entry is None:\n            self._invalid_on_entry = await self.invalid_transition()\n        return self._invalid_on_entry\n\n    async def fizzled(self) -> bool:\n        \"\"\"\n        Determines if a rule is fizzled and side-effects need to be reverted.\n\n        Rules are fizzled if the transitions were valid on entry (thus firing\n        `self.before_transition`) but are invalid upon exiting the governed context,\n        most likely caused by another rule mutating the transition.\n\n        Returns:\n            True if the rule is fizzled, False otherwise.\n        \"\"\"\n\n        if self._invalid_on_entry:\n            return False\n        return await self.invalid_transition()\n\n    async def invalid_transition(self) -> bool:\n        \"\"\"\n        Determines if the transition proposed by the `OrchestrationContext` is invalid.\n\n        If the `OrchestrationContext` is attempting to manage a transition with this\n        rule that differs from the transition the rule was instantiated with, the\n        transition is considered to be invalid. Depending on the context, a rule with an\n        invalid transition is either \"invalid\" or \"fizzled\".\n\n        Returns:\n            True if the transition is invalid, False otherwise.\n        \"\"\"\n\n        initial_state_type = self.context.initial_state_type\n        proposed_state_type = self.context.proposed_state_type\n        return (self.from_state_type != initial_state_type) or (\n            self.to_state_type != proposed_state_type\n        )\n\n    async def reject_transition(self, state: Optional[states.State], reason: str):\n        \"\"\"\n        Rejects a proposed transition before the transition is validated.\n\n        This method will reject a proposed transition, mutating the proposed state to\n        the provided `state`. A reason for rejecting the transition is also passed on\n        to the `OrchestrationContext`. Rules that reject the transition will not fizzle,\n        despite the proposed state type changing.\n\n        Args:\n            state: The new proposed state. If `None`, the current run state will be\n                returned in the result instead.\n            reason: The reason for rejecting the transition\n        \"\"\"\n\n        # don't run if the transition is already validated\n        if self.context.validated_state:\n            raise RuntimeError(\"The transition is already validated\")\n\n        # the current state will be used if a new one is not provided\n        if state is None:\n            if self.from_state_type is None:\n                raise OrchestrationError(\n                    \"The current run has no state; this transition cannot be \"\n                    \"rejected without providing a new state.\"\n                )\n            self.to_state_type = None\n            self.context.proposed_state = None\n        else:\n            # a rule that mutates state should not fizzle itself\n            self.to_state_type = state.type\n            self.context.proposed_state = state\n\n        self.context.response_status = SetStateStatus.REJECT\n        self.context.response_details = StateRejectDetails(reason=reason)\n\n    async def delay_transition(\n        self,\n        delay_seconds: int,\n        reason: str,\n    ):\n        \"\"\"\n        Delays a proposed transition before the transition is validated.\n\n        This method will delay a proposed transition, setting the proposed state to\n        `None`, signaling to the `OrchestrationContext` that no state should be\n        written to the database. The number of seconds a transition should be delayed is\n        passed to the `OrchestrationContext`. A reason for delaying the transition is\n        also provided. Rules that delay the transition will not fizzle, despite the\n        proposed state type changing.\n\n        Args:\n            delay_seconds: The number of seconds the transition should be delayed\n            reason: The reason for delaying the transition\n        \"\"\"\n\n        # don't run if the transition is already validated\n        if self.context.validated_state:\n            raise RuntimeError(\"The transition is already validated\")\n\n        # a rule that mutates state should not fizzle itself\n        self.to_state_type = None\n        self.context.proposed_state = None\n        self.context.response_status = SetStateStatus.WAIT\n        self.context.response_details = StateWaitDetails(\n            delay_seconds=delay_seconds, reason=reason\n        )\n\n    async def abort_transition(self, reason: str):\n        \"\"\"\n        Aborts a proposed transition before the transition is validated.\n\n        This method will abort a proposed transition, expecting no further action to\n        occur for this run. The proposed state is set to `None`, signaling to the\n        `OrchestrationContext` that no state should be written to the database. A\n        reason for aborting the transition is also provided. Rules that abort the\n        transition will not fizzle, despite the proposed state type changing.\n\n        Args:\n            reason: The reason for aborting the transition\n        \"\"\"\n\n        # don't run if the transition is already validated\n        if self.context.validated_state:\n            raise RuntimeError(\"The transition is already validated\")\n\n        # a rule that mutates state should not fizzle itself\n        self.to_state_type = None\n        self.context.proposed_state = None\n        self.context.response_status = SetStateStatus.ABORT\n        self.context.response_details = StateAbortDetails(reason=reason)\n\n    async def rename_state(self, state_name):\n        \"\"\"\n        Sets the \"name\" attribute on a proposed state.\n\n        The name of a state is an annotation intended to provide rich, human-readable\n        context for how a run is progressing. This method only updates the name and not\n        the canonical state TYPE, and will not fizzle or invalidate any other rules\n        that might govern this state transition.\n        \"\"\"\n        if self.context.proposed_state is not None:\n            self.context.proposed_state.name = state_name\n\n    async def update_context_parameters(self, key, value):\n        \"\"\"\n        Updates the \"parameters\" dictionary attribute with the specified key-value pair.\n\n        This mechanism streamlines the process of passing messages and information\n        between orchestration rules if necessary and is simpler and more ephemeral than\n        message-passing via the database or some other side-effect. This mechanism can\n        be used to break up large rules for ease of testing or comprehension, but note\n        that any rules coupled this way (or any other way) are no longer independent and\n        the order in which they appear in the orchestration policy priority will matter.\n        \"\"\"\n\n        self.context.parameters.update({key: value})\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.before_transition","title":"before_transition async","text":"

    Implements a hook that can fire before a state is committed to the database.

    This hook may produce side-effects or mutate the proposed state of a transition using one of four methods: self.reject_transition, self.delay_transition, self.abort_transition, and self.rename_state.

    Note

    As currently implemented, the before_transition hook is not perfectly isolated from mutating the transition. It is a standard instance method that has access to self, and therefore self.context. This should never be modified directly. Furthermore, context.run is an ORM model, and mutating the run can also cause unintended writes to the database.

    Parameters:

    Name Type Description Default initial_state Optional[State]

    The initial state of a transition

    required proposed_state Optional[State]

    The proposed state of a transition

    required context OrchestrationContext

    A safe copy of the OrchestrationContext, with the exception of context.run, mutating this context will have no effect on the broader orchestration environment.

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def before_transition(\n    self,\n    initial_state: Optional[states.State],\n    proposed_state: Optional[states.State],\n    context: OrchestrationContext,\n) -> None:\n    \"\"\"\n    Implements a hook that can fire before a state is committed to the database.\n\n    This hook may produce side-effects or mutate the proposed state of a\n    transition using one of four methods: `self.reject_transition`,\n    `self.delay_transition`, `self.abort_transition`, and `self.rename_state`.\n\n    Note:\n        As currently implemented, the `before_transition` hook is not\n        perfectly isolated from mutating the transition. It is a standard instance\n        method that has access to `self`, and therefore `self.context`. This should\n        never be modified directly. Furthermore, `context.run` is an ORM model, and\n        mutating the run can also cause unintended writes to the database.\n\n    Args:\n        initial_state: The initial state of a transition\n        proposed_state: The proposed state of a transition\n        context: A safe copy of the `OrchestrationContext`, with the exception of\n            `context.run`, mutating this context will have no effect on the broader\n            orchestration environment.\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.after_transition","title":"after_transition async","text":"

    Implements a hook that can fire after a state is committed to the database.

    Parameters:

    Name Type Description Default initial_state Optional[State]

    The initial state of a transition

    required validated_state Optional[State]

    The governed state that has been committed to the database

    required context OrchestrationContext

    A safe copy of the OrchestrationContext, with the exception of context.run, mutating this context will have no effect on the broader orchestration environment.

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def after_transition(\n    self,\n    initial_state: Optional[states.State],\n    validated_state: Optional[states.State],\n    context: OrchestrationContext,\n) -> None:\n    \"\"\"\n    Implements a hook that can fire after a state is committed to the database.\n\n    Args:\n        initial_state: The initial state of a transition\n        validated_state: The governed state that has been committed to the database\n        context: A safe copy of the `OrchestrationContext`, with the exception of\n            `context.run`, mutating this context will have no effect on the broader\n            orchestration environment.\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.cleanup","title":"cleanup async","text":"

    Implements a hook that can fire after a state is committed to the database.

    The intended use of this method is to revert side-effects produced by self.before_transition when the transition is found to be invalid on exit. This allows multiple rules to be gracefully run in sequence, without logic that keeps track of all other rules that might govern a transition.

    Parameters:

    Name Type Description Default initial_state Optional[State]

    The initial state of a transition

    required validated_state Optional[State]

    The governed state that has been committed to the database

    required context OrchestrationContext

    A safe copy of the OrchestrationContext, with the exception of context.run, mutating this context will have no effect on the broader orchestration environment.

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def cleanup(\n    self,\n    initial_state: Optional[states.State],\n    validated_state: Optional[states.State],\n    context: OrchestrationContext,\n) -> None:\n    \"\"\"\n    Implements a hook that can fire after a state is committed to the database.\n\n    The intended use of this method is to revert side-effects produced by\n    `self.before_transition` when the transition is found to be invalid on exit.\n    This allows multiple rules to be gracefully run in sequence, without logic that\n    keeps track of all other rules that might govern a transition.\n\n    Args:\n        initial_state: The initial state of a transition\n        validated_state: The governed state that has been committed to the database\n        context: A safe copy of the `OrchestrationContext`, with the exception of\n            `context.run`, mutating this context will have no effect on the broader\n            orchestration environment.\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.invalid","title":"invalid async","text":"

    Determines if a rule is invalid.

    Invalid rules do nothing and no hooks fire upon entering or exiting a governed context. Rules are invalid if the transition states types are not contained in self.FROM_STATES and self.TO_STATES, or if the context is proposing a transition that differs from the transition the rule was instantiated with.

    Returns:

    Type Description bool

    True if the rules in invalid, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    async def invalid(self) -> bool:\n    \"\"\"\n    Determines if a rule is invalid.\n\n    Invalid rules do nothing and no hooks fire upon entering or exiting a governed\n    context. Rules are invalid if the transition states types are not contained in\n    `self.FROM_STATES` and `self.TO_STATES`, or if the context is proposing\n    a transition that differs from the transition the rule was instantiated with.\n\n    Returns:\n        True if the rules in invalid, False otherwise.\n    \"\"\"\n    # invalid and fizzled states are mutually exclusive,\n    # `_invalid_on_entry` holds this statefulness\n    if self.from_state_type not in self.FROM_STATES:\n        self._invalid_on_entry = True\n    if self.to_state_type not in self.TO_STATES:\n        self._invalid_on_entry = True\n\n    if self._invalid_on_entry is None:\n        self._invalid_on_entry = await self.invalid_transition()\n    return self._invalid_on_entry\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.fizzled","title":"fizzled async","text":"

    Determines if a rule is fizzled and side-effects need to be reverted.

    Rules are fizzled if the transitions were valid on entry (thus firing self.before_transition) but are invalid upon exiting the governed context, most likely caused by another rule mutating the transition.

    Returns:

    Type Description bool

    True if the rule is fizzled, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    async def fizzled(self) -> bool:\n    \"\"\"\n    Determines if a rule is fizzled and side-effects need to be reverted.\n\n    Rules are fizzled if the transitions were valid on entry (thus firing\n    `self.before_transition`) but are invalid upon exiting the governed context,\n    most likely caused by another rule mutating the transition.\n\n    Returns:\n        True if the rule is fizzled, False otherwise.\n    \"\"\"\n\n    if self._invalid_on_entry:\n        return False\n    return await self.invalid_transition()\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.invalid_transition","title":"invalid_transition async","text":"

    Determines if the transition proposed by the OrchestrationContext is invalid.

    If the OrchestrationContext is attempting to manage a transition with this rule that differs from the transition the rule was instantiated with, the transition is considered to be invalid. Depending on the context, a rule with an invalid transition is either \"invalid\" or \"fizzled\".

    Returns:

    Type Description bool

    True if the transition is invalid, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    async def invalid_transition(self) -> bool:\n    \"\"\"\n    Determines if the transition proposed by the `OrchestrationContext` is invalid.\n\n    If the `OrchestrationContext` is attempting to manage a transition with this\n    rule that differs from the transition the rule was instantiated with, the\n    transition is considered to be invalid. Depending on the context, a rule with an\n    invalid transition is either \"invalid\" or \"fizzled\".\n\n    Returns:\n        True if the transition is invalid, False otherwise.\n    \"\"\"\n\n    initial_state_type = self.context.initial_state_type\n    proposed_state_type = self.context.proposed_state_type\n    return (self.from_state_type != initial_state_type) or (\n        self.to_state_type != proposed_state_type\n    )\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.reject_transition","title":"reject_transition async","text":"

    Rejects a proposed transition before the transition is validated.

    This method will reject a proposed transition, mutating the proposed state to the provided state. A reason for rejecting the transition is also passed on to the OrchestrationContext. Rules that reject the transition will not fizzle, despite the proposed state type changing.

    Parameters:

    Name Type Description Default state Optional[State]

    The new proposed state. If None, the current run state will be returned in the result instead.

    required reason str

    The reason for rejecting the transition

    required Source code in prefect/server/orchestration/rules.py
    async def reject_transition(self, state: Optional[states.State], reason: str):\n    \"\"\"\n    Rejects a proposed transition before the transition is validated.\n\n    This method will reject a proposed transition, mutating the proposed state to\n    the provided `state`. A reason for rejecting the transition is also passed on\n    to the `OrchestrationContext`. Rules that reject the transition will not fizzle,\n    despite the proposed state type changing.\n\n    Args:\n        state: The new proposed state. If `None`, the current run state will be\n            returned in the result instead.\n        reason: The reason for rejecting the transition\n    \"\"\"\n\n    # don't run if the transition is already validated\n    if self.context.validated_state:\n        raise RuntimeError(\"The transition is already validated\")\n\n    # the current state will be used if a new one is not provided\n    if state is None:\n        if self.from_state_type is None:\n            raise OrchestrationError(\n                \"The current run has no state; this transition cannot be \"\n                \"rejected without providing a new state.\"\n            )\n        self.to_state_type = None\n        self.context.proposed_state = None\n    else:\n        # a rule that mutates state should not fizzle itself\n        self.to_state_type = state.type\n        self.context.proposed_state = state\n\n    self.context.response_status = SetStateStatus.REJECT\n    self.context.response_details = StateRejectDetails(reason=reason)\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.delay_transition","title":"delay_transition async","text":"

    Delays a proposed transition before the transition is validated.

    This method will delay a proposed transition, setting the proposed state to None, signaling to the OrchestrationContext that no state should be written to the database. The number of seconds a transition should be delayed is passed to the OrchestrationContext. A reason for delaying the transition is also provided. Rules that delay the transition will not fizzle, despite the proposed state type changing.

    Parameters:

    Name Type Description Default delay_seconds int

    The number of seconds the transition should be delayed

    required reason str

    The reason for delaying the transition

    required Source code in prefect/server/orchestration/rules.py
    async def delay_transition(\n    self,\n    delay_seconds: int,\n    reason: str,\n):\n    \"\"\"\n    Delays a proposed transition before the transition is validated.\n\n    This method will delay a proposed transition, setting the proposed state to\n    `None`, signaling to the `OrchestrationContext` that no state should be\n    written to the database. The number of seconds a transition should be delayed is\n    passed to the `OrchestrationContext`. A reason for delaying the transition is\n    also provided. Rules that delay the transition will not fizzle, despite the\n    proposed state type changing.\n\n    Args:\n        delay_seconds: The number of seconds the transition should be delayed\n        reason: The reason for delaying the transition\n    \"\"\"\n\n    # don't run if the transition is already validated\n    if self.context.validated_state:\n        raise RuntimeError(\"The transition is already validated\")\n\n    # a rule that mutates state should not fizzle itself\n    self.to_state_type = None\n    self.context.proposed_state = None\n    self.context.response_status = SetStateStatus.WAIT\n    self.context.response_details = StateWaitDetails(\n        delay_seconds=delay_seconds, reason=reason\n    )\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.abort_transition","title":"abort_transition async","text":"

    Aborts a proposed transition before the transition is validated.

    This method will abort a proposed transition, expecting no further action to occur for this run. The proposed state is set to None, signaling to the OrchestrationContext that no state should be written to the database. A reason for aborting the transition is also provided. Rules that abort the transition will not fizzle, despite the proposed state type changing.

    Parameters:

    Name Type Description Default reason str

    The reason for aborting the transition

    required Source code in prefect/server/orchestration/rules.py
    async def abort_transition(self, reason: str):\n    \"\"\"\n    Aborts a proposed transition before the transition is validated.\n\n    This method will abort a proposed transition, expecting no further action to\n    occur for this run. The proposed state is set to `None`, signaling to the\n    `OrchestrationContext` that no state should be written to the database. A\n    reason for aborting the transition is also provided. Rules that abort the\n    transition will not fizzle, despite the proposed state type changing.\n\n    Args:\n        reason: The reason for aborting the transition\n    \"\"\"\n\n    # don't run if the transition is already validated\n    if self.context.validated_state:\n        raise RuntimeError(\"The transition is already validated\")\n\n    # a rule that mutates state should not fizzle itself\n    self.to_state_type = None\n    self.context.proposed_state = None\n    self.context.response_status = SetStateStatus.ABORT\n    self.context.response_details = StateAbortDetails(reason=reason)\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.rename_state","title":"rename_state async","text":"

    Sets the \"name\" attribute on a proposed state.

    The name of a state is an annotation intended to provide rich, human-readable context for how a run is progressing. This method only updates the name and not the canonical state TYPE, and will not fizzle or invalidate any other rules that might govern this state transition.

    Source code in prefect/server/orchestration/rules.py
    async def rename_state(self, state_name):\n    \"\"\"\n    Sets the \"name\" attribute on a proposed state.\n\n    The name of a state is an annotation intended to provide rich, human-readable\n    context for how a run is progressing. This method only updates the name and not\n    the canonical state TYPE, and will not fizzle or invalidate any other rules\n    that might govern this state transition.\n    \"\"\"\n    if self.context.proposed_state is not None:\n        self.context.proposed_state.name = state_name\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.update_context_parameters","title":"update_context_parameters async","text":"

    Updates the \"parameters\" dictionary attribute with the specified key-value pair.

    This mechanism streamlines the process of passing messages and information between orchestration rules if necessary and is simpler and more ephemeral than message-passing via the database or some other side-effect. This mechanism can be used to break up large rules for ease of testing or comprehension, but note that any rules coupled this way (or any other way) are no longer independent and the order in which they appear in the orchestration policy priority will matter.

    Source code in prefect/server/orchestration/rules.py
    async def update_context_parameters(self, key, value):\n    \"\"\"\n    Updates the \"parameters\" dictionary attribute with the specified key-value pair.\n\n    This mechanism streamlines the process of passing messages and information\n    between orchestration rules if necessary and is simpler and more ephemeral than\n    message-passing via the database or some other side-effect. This mechanism can\n    be used to break up large rules for ease of testing or comprehension, but note\n    that any rules coupled this way (or any other way) are no longer independent and\n    the order in which they appear in the orchestration policy priority will matter.\n    \"\"\"\n\n    self.context.parameters.update({key: value})\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform","title":"BaseUniversalTransform","text":"

    Bases: AbstractAsyncContextManager

    An abstract base class used to implement privileged bookkeeping logic.

    Warning

    In almost all cases, use the BaseOrchestrationRule base class instead.

    Beyond the orchestration rules implemented with the BaseOrchestrationRule ABC, Universal transforms are not stateful, and fire their before- and after- transition hooks on every state transition unless the proposed state is None, indicating that no state should be written to the database. Because there are no guardrails in place to prevent directly mutating state or other parts of the orchestration context, universal transforms should only be used with care.

    Attributes:

    Name Type Description FROM_STATES Iterable

    for compatibility with BaseOrchestrationPolicy

    TO_STATES Iterable

    for compatibility with BaseOrchestrationPolicy

    context

    the orchestration context

    from_state_type

    the state type a run is currently in

    to_state_type

    the intended proposed state type prior to any orchestration

    Parameters:

    Name Type Description Default context OrchestrationContext

    A FlowOrchestrationContext or TaskOrchestrationContext that is passed between transforms

    required Source code in prefect/server/orchestration/rules.py
    class BaseUniversalTransform(contextlib.AbstractAsyncContextManager):\n    \"\"\"\n    An abstract base class used to implement privileged bookkeeping logic.\n\n    Warning:\n        In almost all cases, use the `BaseOrchestrationRule` base class instead.\n\n    Beyond the orchestration rules implemented with the `BaseOrchestrationRule` ABC,\n    Universal transforms are not stateful, and fire their before- and after- transition\n    hooks on every state transition unless the proposed state is `None`, indicating that\n    no state should be written to the database. Because there are no guardrails in place\n    to prevent directly mutating state or other parts of the orchestration context,\n    universal transforms should only be used with care.\n\n    Attributes:\n        FROM_STATES: for compatibility with `BaseOrchestrationPolicy`\n        TO_STATES: for compatibility with `BaseOrchestrationPolicy`\n        context: the orchestration context\n        from_state_type: the state type a run is currently in\n        to_state_type: the intended proposed state type prior to any orchestration\n\n    Args:\n        context: A `FlowOrchestrationContext` or `TaskOrchestrationContext` that is\n            passed between transforms\n    \"\"\"\n\n    # `BaseUniversalTransform` will always fire on non-null transitions\n    FROM_STATES: Iterable = ALL_ORCHESTRATION_STATES\n    TO_STATES: Iterable = ALL_ORCHESTRATION_STATES\n\n    def __init__(\n        self,\n        context: OrchestrationContext,\n        from_state_type: Optional[states.StateType],\n        to_state_type: Optional[states.StateType],\n    ):\n        self.context = context\n        self.from_state_type = from_state_type\n        self.to_state_type = to_state_type\n\n    async def __aenter__(self):\n        \"\"\"\n        Enter an async runtime context governed by this transform.\n\n        The `with` statement will bind a governed `OrchestrationContext` to the target\n        specified by the `as` clause. If the transition proposed by the\n        `OrchestrationContext` has been nullified on entry and `context.proposed_state`\n        is `None`, entering this context will do nothing. Otherwise\n        `self.before_transition` will fire.\n        \"\"\"\n\n        await self.before_transition(self.context)\n        self.context.rule_signature.append(str(self.__class__))\n        return self.context\n\n    async def __aexit__(\n        self,\n        exc_type: Optional[Type[BaseException]],\n        exc_val: Optional[BaseException],\n        exc_tb: Optional[TracebackType],\n    ) -> None:\n        \"\"\"\n        Exit the async runtime context governed by this transform.\n\n        If the transition has been nullified or errorred upon exiting this transforms's context,\n        nothing happens. Otherwise, `self.after_transition` will fire on every non-null\n        proposed state.\n        \"\"\"\n\n        if not self.exception_in_transition():\n            await self.after_transition(self.context)\n            self.context.finalization_signature.append(str(self.__class__))\n\n    async def before_transition(self, context) -> None:\n        \"\"\"\n        Implements a hook that fires before a state is committed to the database.\n\n        Args:\n            context: the `OrchestrationContext` that contains transition details\n\n        Returns:\n            None\n        \"\"\"\n\n    async def after_transition(self, context) -> None:\n        \"\"\"\n        Implements a hook that can fire after a state is committed to the database.\n\n        Args:\n            context: the `OrchestrationContext` that contains transition details\n\n        Returns:\n            None\n        \"\"\"\n\n    def nullified_transition(self) -> bool:\n        \"\"\"\n        Determines if the transition has been nullified.\n\n        Transitions are nullified if the proposed state is `None`, indicating that\n        nothing should be written to the database.\n\n        Returns:\n            True if the transition is nullified, False otherwise.\n        \"\"\"\n\n        return self.context.proposed_state is None\n\n    def exception_in_transition(self) -> bool:\n        \"\"\"\n        Determines if the transition has encountered an exception.\n\n        Returns:\n            True if the transition is encountered an exception, False otherwise.\n        \"\"\"\n\n        return self.context.orchestration_error is not None\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform.before_transition","title":"before_transition async","text":"

    Implements a hook that fires before a state is committed to the database.

    Parameters:

    Name Type Description Default context

    the OrchestrationContext that contains transition details

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def before_transition(self, context) -> None:\n    \"\"\"\n    Implements a hook that fires before a state is committed to the database.\n\n    Args:\n        context: the `OrchestrationContext` that contains transition details\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform.after_transition","title":"after_transition async","text":"

    Implements a hook that can fire after a state is committed to the database.

    Parameters:

    Name Type Description Default context

    the OrchestrationContext that contains transition details

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def after_transition(self, context) -> None:\n    \"\"\"\n    Implements a hook that can fire after a state is committed to the database.\n\n    Args:\n        context: the `OrchestrationContext` that contains transition details\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform.nullified_transition","title":"nullified_transition","text":"

    Determines if the transition has been nullified.

    Transitions are nullified if the proposed state is None, indicating that nothing should be written to the database.

    Returns:

    Type Description bool

    True if the transition is nullified, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    def nullified_transition(self) -> bool:\n    \"\"\"\n    Determines if the transition has been nullified.\n\n    Transitions are nullified if the proposed state is `None`, indicating that\n    nothing should be written to the database.\n\n    Returns:\n        True if the transition is nullified, False otherwise.\n    \"\"\"\n\n    return self.context.proposed_state is None\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform.exception_in_transition","title":"exception_in_transition","text":"

    Determines if the transition has encountered an exception.

    Returns:

    Type Description bool

    True if the transition is encountered an exception, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    def exception_in_transition(self) -> bool:\n    \"\"\"\n    Determines if the transition has encountered an exception.\n\n    Returns:\n        True if the transition is encountered an exception, False otherwise.\n    \"\"\"\n\n    return self.context.orchestration_error is not None\n
    "},{"location":"api-ref/server/schemas/actions/","title":"server.schemas.actions","text":""},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions","title":"prefect.server.schemas.actions","text":"

    Reduced schemas for accepting API actions.

    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ArtifactCreate","title":"ArtifactCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create an artifact.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ArtifactCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create an artifact.\"\"\"\n\n    key: Optional[str] = FieldFrom(schemas.core.Artifact)\n    type: Optional[str] = FieldFrom(schemas.core.Artifact)\n    description: Optional[str] = FieldFrom(schemas.core.Artifact)\n    data: Optional[Union[Dict[str, Any], Any]] = FieldFrom(schemas.core.Artifact)\n    metadata_: Optional[Dict[str, str]] = FieldFrom(schemas.core.Artifact)\n    flow_run_id: Optional[UUID] = FieldFrom(schemas.core.Artifact)\n    task_run_id: Optional[UUID] = FieldFrom(schemas.core.Artifact)\n\n    _validate_artifact_format = validator(\"key\", allow_reuse=True)(\n        validate_artifact_key\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ArtifactCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ArtifactUpdate","title":"ArtifactUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update an artifact.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ArtifactUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update an artifact.\"\"\"\n\n    data: Optional[Union[Dict[str, Any], Any]] = FieldFrom(schemas.core.Artifact)\n    description: Optional[str] = FieldFrom(schemas.core.Artifact)\n    metadata_: Optional[Dict[str, str]] = FieldFrom(schemas.core.Artifact)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ArtifactUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentCreate","title":"BlockDocumentCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block document.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block document.\"\"\"\n\n    name: Optional[str] = FieldFrom(schemas.core.BlockDocument)\n    data: dict = FieldFrom(schemas.core.BlockDocument)\n    block_schema_id: UUID = FieldFrom(schemas.core.BlockDocument)\n    block_type_id: UUID = FieldFrom(schemas.core.BlockDocument)\n    is_anonymous: bool = FieldFrom(schemas.core.BlockDocument)\n\n    _validate_name_format = validator(\"name\", allow_reuse=True)(\n        validate_block_document_name\n    )\n\n    @root_validator\n    def validate_name_is_present_if_not_anonymous(cls, values):\n        # TODO: We should find an elegant way to reuse this logic from the origin model\n        if not values.get(\"is_anonymous\") and not values.get(\"name\"):\n            raise ValueError(\"Names must be provided for block documents.\")\n        return values\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentReferenceCreate","title":"BlockDocumentReferenceCreate","text":"

    Bases: ActionBaseModel

    Data used to create block document reference.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentReferenceCreate(ActionBaseModel):\n    \"\"\"Data used to create block document reference.\"\"\"\n\n    id: UUID = FieldFrom(schemas.core.BlockDocumentReference)\n    parent_block_document_id: UUID = FieldFrom(schemas.core.BlockDocumentReference)\n    reference_block_document_id: UUID = FieldFrom(schemas.core.BlockDocumentReference)\n    name: str = FieldFrom(schemas.core.BlockDocumentReference)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentReferenceCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentUpdate","title":"BlockDocumentUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a block document.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a block document.\"\"\"\n\n    block_schema_id: Optional[UUID] = Field(\n        default=None, description=\"A block schema ID\"\n    )\n    data: dict = FieldFrom(schemas.core.BlockDocument)\n    merge_existing_data: bool = True\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockSchemaCreate","title":"BlockSchemaCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block schema.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockSchemaCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block schema.\"\"\"\n\n    fields: dict = FieldFrom(schemas.core.BlockSchema)\n    block_type_id: Optional[UUID] = FieldFrom(schemas.core.BlockSchema)\n    capabilities: List[str] = FieldFrom(schemas.core.BlockSchema)\n    version: str = FieldFrom(schemas.core.BlockSchema)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockSchemaCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockTypeCreate","title":"BlockTypeCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block type.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockTypeCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block type.\"\"\"\n\n    name: str = FieldFrom(schemas.core.BlockType)\n    slug: str = FieldFrom(schemas.core.BlockType)\n    logo_url: Optional[schemas.core.HttpUrl] = FieldFrom(schemas.core.BlockType)\n    documentation_url: Optional[schemas.core.HttpUrl] = FieldFrom(\n        schemas.core.BlockType\n    )\n    description: Optional[str] = FieldFrom(schemas.core.BlockType)\n    code_example: Optional[str] = FieldFrom(schemas.core.BlockType)\n\n    # validators\n    _validate_slug_format = validator(\"slug\", allow_reuse=True)(\n        validate_block_type_slug\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockTypeCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockTypeUpdate","title":"BlockTypeUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a block type.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockTypeUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a block type.\"\"\"\n\n    logo_url: Optional[schemas.core.HttpUrl] = FieldFrom(schemas.core.BlockType)\n    documentation_url: Optional[schemas.core.HttpUrl] = FieldFrom(\n        schemas.core.BlockType\n    )\n    description: Optional[str] = FieldFrom(schemas.core.BlockType)\n    code_example: Optional[str] = FieldFrom(schemas.core.BlockType)\n\n    @classmethod\n    def updatable_fields(cls) -> set:\n        return get_class_fields_only(cls)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockTypeUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitCreate","title":"ConcurrencyLimitCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a concurrency limit.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ConcurrencyLimitCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a concurrency limit.\"\"\"\n\n    tag: str = FieldFrom(schemas.core.ConcurrencyLimit)\n    concurrency_limit: int = FieldFrom(schemas.core.ConcurrencyLimit)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitV2Create","title":"ConcurrencyLimitV2Create","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a v2 concurrency limit.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ConcurrencyLimitV2Create(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a v2 concurrency limit.\"\"\"\n\n    active: bool = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    name: str = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    limit: int = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    active_slots: int = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    denied_slots: int = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    slot_decay_per_second: float = FieldFrom(schemas.core.ConcurrencyLimitV2)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitV2Create.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitV2Update","title":"ConcurrencyLimitV2Update","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a v2 concurrency limit.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ConcurrencyLimitV2Update(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a v2 concurrency limit.\"\"\"\n\n    active: Optional[bool] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    name: Optional[str] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    limit: Optional[int] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    active_slots: Optional[int] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    denied_slots: Optional[int] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    slot_decay_per_second: Optional[float] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitV2Update.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentCreate","title":"DeploymentCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a deployment.

    Source code in prefect/server/schemas/actions.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None,\n)\n@copy_model_fields\nclass DeploymentCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a deployment.\"\"\"\n\n    @root_validator(pre=True)\n    def remove_old_fields(cls, values):\n        # 2.7.7 removed worker_pool_queue_id in lieu of worker_pool_name and\n        # worker_pool_queue_name. Those fields were later renamed to work_pool_name\n        # and work_queue_name. This validator removes old fields provided\n        # by older clients to avoid 422 errors.\n        values_copy = copy(values)\n        worker_pool_queue_id = values_copy.pop(\"worker_pool_queue_id\", None)\n        worker_pool_name = values_copy.pop(\"worker_pool_name\", None)\n        worker_pool_queue_name = values_copy.pop(\"worker_pool_queue_name\", None)\n        work_pool_queue_name = values_copy.pop(\"work_pool_queue_name\", None)\n        if worker_pool_queue_id:\n            warnings.warn(\n                (\n                    \"`worker_pool_queue_id` is no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        if worker_pool_name or worker_pool_queue_name or work_pool_queue_name:\n            warnings.warn(\n                (\n                    \"`worker_pool_name`, `worker_pool_queue_name`, and \"\n                    \"`work_pool_name` are\"\n                    \"no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        return values_copy\n\n    name: str = FieldFrom(schemas.core.Deployment)\n    flow_id: UUID = FieldFrom(schemas.core.Deployment)\n    is_schedule_active: Optional[bool] = FieldFrom(schemas.core.Deployment)\n    enforce_parameter_schema: bool = FieldFrom(schemas.core.Deployment)\n    parameter_openapi_schema: Optional[Dict[str, Any]] = FieldFrom(\n        schemas.core.Deployment\n    )\n    parameters: Dict[str, Any] = FieldFrom(schemas.core.Deployment)\n    tags: List[str] = FieldFrom(schemas.core.Deployment)\n    pull_steps: Optional[List[dict]] = FieldFrom(schemas.core.Deployment)\n\n    manifest_path: Optional[str] = FieldFrom(schemas.core.Deployment)\n    work_queue_name: Optional[str] = FieldFrom(schemas.core.Deployment)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the deployment's work pool.\",\n        example=\"my-work-pool\",\n    )\n    storage_document_id: Optional[UUID] = FieldFrom(schemas.core.Deployment)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.Deployment)\n    schedule: Optional[schemas.schedules.SCHEDULE_TYPES] = FieldFrom(\n        schemas.core.Deployment\n    )\n    description: Optional[str] = FieldFrom(schemas.core.Deployment)\n    path: Optional[str] = FieldFrom(schemas.core.Deployment)\n    version: Optional[str] = FieldFrom(schemas.core.Deployment)\n    entrypoint: Optional[str] = FieldFrom(schemas.core.Deployment)\n    infra_overrides: Optional[Dict[str, Any]] = FieldFrom(schemas.core.Deployment)\n\n    def check_valid_configuration(self, base_job_template: dict):\n        \"\"\"Check that the combination of base_job_template defaults\n        and infra_overrides conforms to the specified schema.\n        \"\"\"\n        variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n        if variables_schema is not None:\n            # jsonschema considers required fields, even if that field has a default,\n            # to still be required. To get around this we remove the fields from\n            # required if there is a default present.\n            required = variables_schema.get(\"required\")\n            properties = variables_schema.get(\"properties\")\n            if required is not None and properties is not None:\n                for k, v in properties.items():\n                    if \"default\" in v and k in required:\n                        required.remove(k)\n\n            jsonschema.validate(self.infra_overrides, variables_schema)\n\n    @validator(\"parameters\")\n    def _validate_parameters_conform_to_schema(cls, value, values):\n        \"\"\"Validate that the parameters conform to the parameter schema.\"\"\"\n        if values.get(\"enforce_parameter_schema\"):\n            validate_values_conform_to_schema(\n                value, values.get(\"parameter_openapi_schema\"), ignore_required=True\n            )\n        return value\n\n    @validator(\"parameter_openapi_schema\")\n    def _validate_parameter_openapi_schema(cls, value, values):\n        \"\"\"Validate that the parameter_openapi_schema is a valid json schema.\"\"\"\n        if values.get(\"enforce_parameter_schema\"):\n            validate_schema(value)\n        return value\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentCreate.check_valid_configuration","title":"check_valid_configuration","text":"

    Check that the combination of base_job_template defaults and infra_overrides conforms to the specified schema.

    Source code in prefect/server/schemas/actions.py
    def check_valid_configuration(self, base_job_template: dict):\n    \"\"\"Check that the combination of base_job_template defaults\n    and infra_overrides conforms to the specified schema.\n    \"\"\"\n    variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n    if variables_schema is not None:\n        # jsonschema considers required fields, even if that field has a default,\n        # to still be required. To get around this we remove the fields from\n        # required if there is a default present.\n        required = variables_schema.get(\"required\")\n        properties = variables_schema.get(\"properties\")\n        if required is not None and properties is not None:\n            for k, v in properties.items():\n                if \"default\" in v and k in required:\n                    required.remove(k)\n\n        jsonschema.validate(self.infra_overrides, variables_schema)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentFlowRunCreate","title":"DeploymentFlowRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run from a deployment.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass DeploymentFlowRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run from a deployment.\"\"\"\n\n    # FlowRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the flow run to create\"\n    )\n\n    name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parameters: dict = FieldFrom(schemas.core.FlowRun)\n    context: dict = FieldFrom(schemas.core.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    empirical_policy: schemas.core.FlowRunPolicy = FieldFrom(schemas.core.FlowRun)\n    tags: List[str] = FieldFrom(schemas.core.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    work_queue_name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentFlowRunCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentUpdate","title":"DeploymentUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a deployment.

    Source code in prefect/server/schemas/actions.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None,\n)\n@copy_model_fields\nclass DeploymentUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a deployment.\"\"\"\n\n    @root_validator(pre=True)\n    def remove_old_fields(cls, values):\n        # 2.7.7 removed worker_pool_queue_id in lieu of worker_pool_name and\n        # worker_pool_queue_name. Those fields were later renamed to work_pool_name\n        # and work_queue_name. This validator removes old fields provided\n        # by older clients to avoid 422 errors.\n        values_copy = copy(values)\n        worker_pool_queue_id = values_copy.pop(\"worker_pool_queue_id\", None)\n        worker_pool_name = values_copy.pop(\"worker_pool_name\", None)\n        worker_pool_queue_name = values_copy.pop(\"worker_pool_queue_name\", None)\n        work_pool_queue_name = values_copy.pop(\"work_pool_queue_name\", None)\n        if worker_pool_queue_id:\n            warnings.warn(\n                (\n                    \"`worker_pool_queue_id` is no longer supported for updating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        if worker_pool_name or worker_pool_queue_name or work_pool_queue_name:\n            warnings.warn(\n                (\n                    \"`worker_pool_name`, `worker_pool_queue_name`, and \"\n                    \"`work_pool_name` are\"\n                    \"no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        return values_copy\n\n    version: Optional[str] = FieldFrom(schemas.core.Deployment)\n    schedule: Optional[schemas.schedules.SCHEDULE_TYPES] = FieldFrom(\n        schemas.core.Deployment\n    )\n    description: Optional[str] = FieldFrom(schemas.core.Deployment)\n    is_schedule_active: bool = FieldFrom(schemas.core.Deployment)\n    parameters: Optional[Dict[str, Any]] = Field(\n        default=None,\n        description=\"Parameters for flow runs scheduled by the deployment.\",\n    )\n    tags: List[str] = FieldFrom(schemas.core.Deployment)\n    work_queue_name: Optional[str] = FieldFrom(schemas.core.Deployment)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the deployment's work pool.\",\n        example=\"my-work-pool\",\n    )\n    path: Optional[str] = FieldFrom(schemas.core.Deployment)\n    infra_overrides: Optional[Dict[str, Any]] = FieldFrom(schemas.core.Deployment)\n    entrypoint: Optional[str] = FieldFrom(schemas.core.Deployment)\n    manifest_path: Optional[str] = FieldFrom(schemas.core.Deployment)\n    storage_document_id: Optional[UUID] = FieldFrom(schemas.core.Deployment)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.Deployment)\n    enforce_parameter_schema: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n\n    def check_valid_configuration(self, base_job_template: dict):\n        \"\"\"Check that the combination of base_job_template defaults\n        and infra_overrides conforms to the specified schema.\n        \"\"\"\n        variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n        if variables_schema is not None:\n            # jsonschema considers required fields, even if that field has a default,\n            # to still be required. To get around this we remove the fields from\n            # required if there is a default present.\n            required = variables_schema.get(\"required\")\n            properties = variables_schema.get(\"properties\")\n            if required is not None and properties is not None:\n                for k, v in properties.items():\n                    if \"default\" in v and k in required:\n                        required.remove(k)\n\n        if variables_schema is not None:\n            jsonschema.validate(self.infra_overrides, variables_schema)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentUpdate.check_valid_configuration","title":"check_valid_configuration","text":"

    Check that the combination of base_job_template defaults and infra_overrides conforms to the specified schema.

    Source code in prefect/server/schemas/actions.py
    def check_valid_configuration(self, base_job_template: dict):\n    \"\"\"Check that the combination of base_job_template defaults\n    and infra_overrides conforms to the specified schema.\n    \"\"\"\n    variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n    if variables_schema is not None:\n        # jsonschema considers required fields, even if that field has a default,\n        # to still be required. To get around this we remove the fields from\n        # required if there is a default present.\n        required = variables_schema.get(\"required\")\n        properties = variables_schema.get(\"properties\")\n        if required is not None and properties is not None:\n            for k, v in properties.items():\n                if \"default\" in v and k in required:\n                    required.remove(k)\n\n    if variables_schema is not None:\n        jsonschema.validate(self.infra_overrides, variables_schema)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowCreate","title":"FlowCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow.\"\"\"\n\n    name: str = FieldFrom(schemas.core.Flow)\n    tags: List[str] = FieldFrom(schemas.core.Flow)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunCreate","title":"FlowRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run.\"\"\"\n\n    # FlowRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the flow run to create\"\n    )\n\n    name: str = FieldFrom(schemas.core.FlowRun)\n    flow_id: UUID = FieldFrom(schemas.core.FlowRun)\n    flow_version: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parameters: dict = FieldFrom(schemas.core.FlowRun)\n    context: dict = FieldFrom(schemas.core.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    empirical_policy: schemas.core.FlowRunPolicy = FieldFrom(schemas.core.FlowRun)\n    tags: List[str] = FieldFrom(schemas.core.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(schemas.core.FlowRun)\n\n    # DEPRECATED\n\n    deployment_id: Optional[UUID] = Field(\n        None,\n        description=(\n            \"DEPRECATED: The id of the deployment associated with this flow run, if\"\n            \" available.\"\n        ),\n        deprecated=True,\n    )\n\n    class Config(ActionBaseModel.Config):\n        json_dumps = orjson_dumps_extra_compatible\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunNotificationPolicyCreate","title":"FlowRunNotificationPolicyCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run notification policy.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowRunNotificationPolicyCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run notification policy.\"\"\"\n\n    is_active: bool = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    state_names: List[str] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    tags: List[str] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    block_document_id: UUID = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    message_template: Optional[str] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunNotificationPolicyCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunNotificationPolicyUpdate","title":"FlowRunNotificationPolicyUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow run notification policy.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowRunNotificationPolicyUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow run notification policy.\"\"\"\n\n    is_active: Optional[bool] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    state_names: Optional[List[str]] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    tags: Optional[List[str]] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    block_document_id: Optional[UUID] = FieldFrom(\n        schemas.core.FlowRunNotificationPolicy\n    )\n    message_template: Optional[str] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunNotificationPolicyUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunUpdate","title":"FlowRunUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow run.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowRunUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow run.\"\"\"\n\n    name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    flow_version: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parameters: dict = FieldFrom(schemas.core.FlowRun)\n    empirical_policy: schemas.core.FlowRunPolicy = FieldFrom(schemas.core.FlowRun)\n    tags: List[str] = FieldFrom(schemas.core.FlowRun)\n    infrastructure_pid: Optional[str] = FieldFrom(schemas.core.FlowRun)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowUpdate","title":"FlowUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow.\"\"\"\n\n    tags: List[str] = FieldFrom(schemas.core.Flow)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.LogCreate","title":"LogCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a log.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass LogCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a log.\"\"\"\n\n    name: str = FieldFrom(schemas.core.Log)\n    level: int = FieldFrom(schemas.core.Log)\n    message: str = FieldFrom(schemas.core.Log)\n    timestamp: DateTimeTZ = FieldFrom(schemas.core.Log)\n    flow_run_id: Optional[UUID] = FieldFrom(schemas.core.Log)\n    task_run_id: Optional[UUID] = FieldFrom(schemas.core.Log)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.LogCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.SavedSearchCreate","title":"SavedSearchCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a saved search.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass SavedSearchCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a saved search.\"\"\"\n\n    name: str = FieldFrom(schemas.core.SavedSearch)\n    filters: List[schemas.core.SavedSearchFilter] = FieldFrom(schemas.core.SavedSearch)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.SavedSearchCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.StateCreate","title":"StateCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a new state.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass StateCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a new state.\"\"\"\n\n    type: schemas.states.StateType = FieldFrom(schemas.states.State)\n    name: Optional[str] = FieldFrom(schemas.states.State)\n    message: Optional[str] = FieldFrom(schemas.states.State)\n    data: Optional[Any] = FieldFrom(schemas.states.State)\n    state_details: schemas.states.StateDetails = FieldFrom(schemas.states.State)\n\n    # DEPRECATED\n\n    timestamp: Optional[DateTimeTZ] = Field(\n        default=None,\n        repr=False,\n        ignored=True,\n    )\n    id: Optional[UUID] = Field(default=None, repr=False, ignored=True)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.StateCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.TaskRunCreate","title":"TaskRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a task run

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass TaskRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a task run\"\"\"\n\n    # TaskRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the task run to create\"\n    )\n\n    name: str = FieldFrom(schemas.core.TaskRun)\n    flow_run_id: Optional[UUID] = FieldFrom(schemas.core.TaskRun)\n    task_key: str = FieldFrom(schemas.core.TaskRun)\n    dynamic_key: str = FieldFrom(schemas.core.TaskRun)\n    cache_key: Optional[str] = FieldFrom(schemas.core.TaskRun)\n    cache_expiration: Optional[DateTimeTZ] = FieldFrom(schemas.core.TaskRun)\n    task_version: Optional[str] = FieldFrom(schemas.core.TaskRun)\n    empirical_policy: schemas.core.TaskRunPolicy = FieldFrom(schemas.core.TaskRun)\n    tags: List[str] = FieldFrom(schemas.core.TaskRun)\n    task_inputs: Dict[\n        str,\n        List[\n            Union[\n                schemas.core.TaskRunResult,\n                schemas.core.Parameter,\n                schemas.core.Constant,\n            ]\n        ],\n    ] = FieldFrom(schemas.core.TaskRun)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.TaskRunCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.TaskRunUpdate","title":"TaskRunUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a task run

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass TaskRunUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a task run\"\"\"\n\n    name: str = FieldFrom(schemas.core.TaskRun)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.TaskRunUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.VariableCreate","title":"VariableCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a Variable.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass VariableCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a Variable.\"\"\"\n\n    name: str = FieldFrom(schemas.core.Variable)\n    value: str = FieldFrom(schemas.core.Variable)\n    tags: Optional[List[str]] = FieldFrom(schemas.core.Variable)\n\n    # validators\n    _validate_name_format = validator(\"name\", allow_reuse=True)(validate_variable_name)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.VariableCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.VariableUpdate","title":"VariableUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a Variable.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass VariableUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a Variable.\"\"\"\n\n    name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the variable\",\n        example=\"my_variable\",\n        max_length=schemas.core.MAX_VARIABLE_NAME_LENGTH,\n    )\n    value: Optional[str] = Field(\n        default=None,\n        description=\"The value of the variable\",\n        example=\"my-value\",\n        max_length=schemas.core.MAX_VARIABLE_VALUE_LENGTH,\n    )\n    tags: Optional[List[str]] = FieldFrom(schemas.core.Variable)\n\n    # validators\n    _validate_name_format = validator(\"name\", allow_reuse=True)(validate_variable_name)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.VariableUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkPoolCreate","title":"WorkPoolCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a work pool.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass WorkPoolCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a work pool.\"\"\"\n\n    name: str = FieldFrom(schemas.core.WorkPool)\n    description: Optional[str] = FieldFrom(schemas.core.WorkPool)\n    type: str = Field(description=\"The work pool type.\", default=\"prefect-agent\")\n    base_job_template: Dict[str, Any] = FieldFrom(schemas.core.WorkPool)\n    is_paused: bool = FieldFrom(schemas.core.WorkPool)\n    concurrency_limit: Optional[int] = FieldFrom(schemas.core.WorkPool)\n\n    _validate_base_job_template = validator(\"base_job_template\", allow_reuse=True)(\n        validate_base_job_template\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkPoolCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkPoolUpdate","title":"WorkPoolUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a work pool.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass WorkPoolUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a work pool.\"\"\"\n\n    description: Optional[str] = FieldFrom(schemas.core.WorkPool)\n    is_paused: Optional[bool] = FieldFrom(schemas.core.WorkPool)\n    base_job_template: Optional[Dict[str, Any]] = FieldFrom(schemas.core.WorkPool)\n    concurrency_limit: Optional[int] = FieldFrom(schemas.core.WorkPool)\n\n    _validate_base_job_template = validator(\"base_job_template\", allow_reuse=True)(\n        validate_base_job_template\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkPoolUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkQueueCreate","title":"WorkQueueCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a work queue.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass WorkQueueCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a work queue.\"\"\"\n\n    name: str = FieldFrom(schemas.core.WorkQueue)\n    description: Optional[str] = FieldFrom(schemas.core.WorkQueue)\n    is_paused: bool = FieldFrom(schemas.core.WorkQueue)\n    concurrency_limit: Optional[int] = FieldFrom(schemas.core.WorkQueue)\n    priority: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The queue's priority. Lower values are higher priority (1 is the highest).\"\n        ),\n    )\n\n    # DEPRECATED\n\n    filter: Optional[schemas.core.QueueFilter] = Field(\n        None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkQueueCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkQueueUpdate","title":"WorkQueueUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a work queue.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass WorkQueueUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a work queue.\"\"\"\n\n    name: str = FieldFrom(schemas.core.WorkQueue)\n    description: Optional[str] = FieldFrom(schemas.core.WorkQueue)\n    is_paused: bool = FieldFrom(schemas.core.WorkQueue)\n    concurrency_limit: Optional[int] = FieldFrom(schemas.core.WorkQueue)\n    priority: Optional[int] = FieldFrom(schemas.core.WorkQueue)\n    last_polled: Optional[DateTimeTZ] = FieldFrom(schemas.core.WorkQueue)\n\n    # DEPRECATED\n\n    filter: Optional[schemas.core.QueueFilter] = Field(\n        None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkQueueUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/core/","title":"server.schemas.core","text":""},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core","title":"prefect.server.schemas.core","text":"

    Full schemas of Prefect REST API objects.

    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Agent","title":"Agent","text":"

    Bases: ORMBaseModel

    An ORM representation of an agent

    Source code in prefect/server/schemas/core.py
    class Agent(ORMBaseModel):\n    \"\"\"An ORM representation of an agent\"\"\"\n\n    name: str = Field(\n        default_factory=lambda: generate_slug(2),\n        description=(\n            \"The name of the agent. If a name is not provided, it will be\"\n            \" auto-generated.\"\n        ),\n    )\n    work_queue_id: UUID = Field(\n        default=..., description=\"The work queue with which the agent is associated.\"\n    )\n    last_activity_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The last time this agent polled for work.\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockDocument","title":"BlockDocument","text":"

    Bases: ORMBaseModel

    An ORM representation of a block document.

    Source code in prefect/server/schemas/core.py
    class BlockDocument(ORMBaseModel):\n    \"\"\"An ORM representation of a block document.\"\"\"\n\n    name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The block document's name. Not required for anonymous block documents.\"\n        ),\n    )\n    data: dict = Field(default_factory=dict, description=\"The block document's data\")\n    block_schema_id: UUID = Field(default=..., description=\"A block schema ID\")\n    block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The associated block schema\"\n    )\n    block_type_id: UUID = Field(default=..., description=\"A block type ID\")\n    block_type_name: Optional[str] = Field(\n        default=None, description=\"The associated block type's name\"\n    )\n    block_type: Optional[BlockType] = Field(\n        default=None, description=\"The associated block type\"\n    )\n    block_document_references: Dict[str, Dict[str, Any]] = Field(\n        default_factory=dict, description=\"Record of the block document's references\"\n    )\n    is_anonymous: bool = Field(\n        default=False,\n        description=(\n            \"Whether the block is anonymous (anonymous blocks are usually created by\"\n            \" Prefect automatically)\"\n        ),\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        # the BlockDocumentCreate subclass allows name=None\n        # and will inherit this validator\n        if v is not None:\n            raise_on_name_with_banned_characters(v)\n        return v\n\n    @root_validator\n    def validate_name_is_present_if_not_anonymous(cls, values):\n        # anonymous blocks may have no name prior to actually being\n        # stored in the database\n        if not values.get(\"is_anonymous\") and not values.get(\"name\"):\n            raise ValueError(\"Names must be provided for block documents.\")\n        return values\n\n    @classmethod\n    async def from_orm_model(\n        cls,\n        session,\n        orm_block_document: \"prefect.server.database.orm_models.ORMBlockDocument\",\n        include_secrets: bool = False,\n    ):\n        data = await orm_block_document.decrypt_data(session=session)\n        # if secrets are not included, obfuscate them based on the schema's\n        # `secret_fields`. Note this walks any nested blocks as well. If the\n        # nested blocks were recovered from named blocks, they will already\n        # be obfuscated, but if nested fields were hardcoded into the parent\n        # blocks data, this is the only opportunity to obfuscate them.\n        if not include_secrets:\n            flat_data = dict_to_flatdict(data)\n            # iterate over the (possibly nested) secret fields\n            # and obfuscate their data\n            for secret_field in orm_block_document.block_schema.fields.get(\n                \"secret_fields\", []\n            ):\n                secret_key = tuple(secret_field.split(\".\"))\n                if flat_data.get(secret_key) is not None:\n                    flat_data[secret_key] = obfuscate_string(flat_data[secret_key])\n                # If a wildcard (*) is in the current secret key path, we take the portion\n                # of the path before the wildcard and compare it to the same level of each\n                # key. A match means that the field is nested under the secret key and should\n                # be obfuscated.\n                elif \"*\" in secret_key:\n                    wildcard_index = secret_key.index(\"*\")\n                    for data_key in flat_data.keys():\n                        if secret_key[0:wildcard_index] == data_key[0:wildcard_index]:\n                            flat_data[data_key] = obfuscate(flat_data[data_key])\n            data = flatdict_to_dict(flat_data)\n\n        return cls(\n            id=orm_block_document.id,\n            created=orm_block_document.created,\n            updated=orm_block_document.updated,\n            name=orm_block_document.name,\n            data=data,\n            block_schema_id=orm_block_document.block_schema_id,\n            block_schema=orm_block_document.block_schema,\n            block_type_id=orm_block_document.block_type_id,\n            block_type_name=orm_block_document.block_type_name,\n            block_type=orm_block_document.block_type,\n            is_anonymous=orm_block_document.is_anonymous,\n        )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockDocumentReference","title":"BlockDocumentReference","text":"

    Bases: ORMBaseModel

    An ORM representation of a block document reference.

    Source code in prefect/server/schemas/core.py
    class BlockDocumentReference(ORMBaseModel):\n    \"\"\"An ORM representation of a block document reference.\"\"\"\n\n    parent_block_document_id: UUID = Field(\n        default=..., description=\"ID of block document the reference is nested within\"\n    )\n    parent_block_document: Optional[BlockDocument] = Field(\n        default=None, description=\"The block document the reference is nested within\"\n    )\n    reference_block_document_id: UUID = Field(\n        default=..., description=\"ID of the nested block document\"\n    )\n    reference_block_document: Optional[BlockDocument] = Field(\n        default=None, description=\"The nested block document\"\n    )\n    name: str = Field(\n        default=..., description=\"The name that the reference is nested under\"\n    )\n\n    @root_validator\n    def validate_parent_and_ref_are_different(cls, values):\n        parent_id = values.get(\"parent_block_document_id\")\n        ref_id = values.get(\"reference_block_document_id\")\n        if parent_id and ref_id and parent_id == ref_id:\n            raise ValueError(\n                \"`parent_block_document_id` and `reference_block_document_id` cannot be\"\n                \" the same\"\n            )\n        return values\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockSchema","title":"BlockSchema","text":"

    Bases: ORMBaseModel

    An ORM representation of a block schema.

    Source code in prefect/server/schemas/core.py
    class BlockSchema(ORMBaseModel):\n    \"\"\"An ORM representation of a block schema.\"\"\"\n\n    checksum: str = Field(default=..., description=\"The block schema's unique checksum\")\n    fields: dict = Field(\n        default_factory=dict, description=\"The block schema's field schema\"\n    )\n    block_type_id: Optional[UUID] = Field(default=..., description=\"A block type ID\")\n    block_type: Optional[BlockType] = Field(\n        default=None, description=\"The associated block type\"\n    )\n    capabilities: List[str] = Field(\n        default_factory=list,\n        description=\"A list of Block capabilities\",\n    )\n    version: str = Field(\n        default=DEFAULT_BLOCK_SCHEMA_VERSION,\n        description=\"Human readable identifier for the block schema\",\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockSchemaReference","title":"BlockSchemaReference","text":"

    Bases: ORMBaseModel

    An ORM representation of a block schema reference.

    Source code in prefect/server/schemas/core.py
    class BlockSchemaReference(ORMBaseModel):\n    \"\"\"An ORM representation of a block schema reference.\"\"\"\n\n    parent_block_schema_id: UUID = Field(\n        default=..., description=\"ID of block schema the reference is nested within\"\n    )\n    parent_block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The block schema the reference is nested within\"\n    )\n    reference_block_schema_id: UUID = Field(\n        default=..., description=\"ID of the nested block schema\"\n    )\n    reference_block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The nested block schema\"\n    )\n    name: str = Field(\n        default=..., description=\"The name that the reference is nested under\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockType","title":"BlockType","text":"

    Bases: ORMBaseModel

    An ORM representation of a block type

    Source code in prefect/server/schemas/core.py
    class BlockType(ORMBaseModel):\n    \"\"\"An ORM representation of a block type\"\"\"\n\n    name: str = Field(default=..., description=\"A block type's name\")\n    slug: str = Field(default=..., description=\"A block type's slug\")\n    logo_url: Optional[HttpUrl] = Field(\n        default=None, description=\"Web URL for the block type's logo\"\n    )\n    documentation_url: Optional[HttpUrl] = Field(\n        default=None, description=\"Web URL for the block type's documentation\"\n    )\n    description: Optional[str] = Field(\n        default=None,\n        description=\"A short blurb about the corresponding block's intended use\",\n    )\n    code_example: Optional[str] = Field(\n        default=None,\n        description=\"A code snippet demonstrating use of the corresponding block\",\n    )\n    is_protected: bool = Field(\n        default=False, description=\"Protected block types cannot be modified via API.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.ConcurrencyLimit","title":"ConcurrencyLimit","text":"

    Bases: ORMBaseModel

    An ORM representation of a concurrency limit.

    Source code in prefect/server/schemas/core.py
    class ConcurrencyLimit(ORMBaseModel):\n    \"\"\"An ORM representation of a concurrency limit.\"\"\"\n\n    tag: str = Field(\n        default=..., description=\"A tag the concurrency limit is applied to.\"\n    )\n    concurrency_limit: int = Field(default=..., description=\"The concurrency limit.\")\n    active_slots: List[UUID] = Field(\n        default_factory=list,\n        description=\"A list of active run ids using a concurrency slot\",\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.ConcurrencyLimitV2","title":"ConcurrencyLimitV2","text":"

    Bases: ORMBaseModel

    An ORM representation of a v2 concurrency limit.

    Source code in prefect/server/schemas/core.py
    class ConcurrencyLimitV2(ORMBaseModel):\n    \"\"\"An ORM representation of a v2 concurrency limit.\"\"\"\n\n    active: bool = Field(\n        default=True, description=\"Whether the concurrency limit is active.\"\n    )\n    name: str = Field(default=..., description=\"The name of the concurrency limit.\")\n    limit: int = Field(default=..., description=\"The concurrency limit.\")\n    active_slots: int = Field(default=0, description=\"The number of active slots.\")\n    denied_slots: int = Field(default=0, description=\"The number of denied slots.\")\n    slot_decay_per_second: float = Field(\n        default=0,\n        description=\"The decay rate for active slots when used as a rate limit.\",\n    )\n    avg_slot_occupancy_seconds: float = Field(\n        default=2.0, description=\"The average amount of time a slot is occupied.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Configuration","title":"Configuration","text":"

    Bases: ORMBaseModel

    An ORM representation of account info.

    Source code in prefect/server/schemas/core.py
    class Configuration(ORMBaseModel):\n    \"\"\"An ORM representation of account info.\"\"\"\n\n    key: str = Field(default=..., description=\"Account info key\")\n    value: dict = Field(default=..., description=\"Account info\")\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Deployment","title":"Deployment","text":"

    Bases: ORMBaseModel

    An ORM representation of deployment data.

    Source code in prefect/server/schemas/core.py
    class Deployment(ORMBaseModel):\n    \"\"\"An ORM representation of deployment data.\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the deployment.\")\n    version: Optional[str] = Field(\n        default=None, description=\"An optional version for the deployment.\"\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"A description for the deployment.\"\n    )\n    flow_id: UUID = Field(\n        default=..., description=\"The flow id associated with the deployment.\"\n    )\n    schedule: Optional[schedules.SCHEDULE_TYPES] = Field(\n        default=None, description=\"A schedule for the deployment.\"\n    )\n    is_schedule_active: bool = Field(\n        default=True, description=\"Whether or not the deployment schedule is active.\"\n    )\n    infra_overrides: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Overrides to apply to the base infrastructure block at runtime.\",\n    )\n    parameters: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Parameters for flow runs scheduled by the deployment.\",\n    )\n    pull_steps: Optional[List[dict]] = Field(\n        default=None,\n        description=\"Pull steps for cloning and running this deployment.\",\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags for the deployment\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The work queue for the deployment. If no work queue is set, work will not\"\n            \" be scheduled.\"\n        ),\n    )\n    last_polled: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The last time the deployment was polled for status updates.\",\n    )\n    parameter_openapi_schema: Optional[Dict[str, Any]] = Field(\n        default=None,\n        description=\"The parameter schema of the flow, including defaults.\",\n    )\n    path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the working directory for the workflow, relative to remote\"\n            \" storage or an absolute path.\"\n        ),\n    )\n    entrypoint: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the entrypoint for the workflow, relative to the `path`.\"\n        ),\n    )\n    manifest_path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the flow's manifest file, relative to the chosen storage.\"\n        ),\n    )\n    storage_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining storage used for this flow.\",\n    )\n    infrastructure_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining infrastructure to use for flow runs.\",\n    )\n    created_by: Optional[CreatedBy] = Field(\n        default=None,\n        description=\"Optional information about the creator of this deployment.\",\n    )\n    updated_by: Optional[UpdatedBy] = Field(\n        default=None,\n        description=\"Optional information about the updater of this deployment.\",\n    )\n    work_queue_id: UUID = Field(\n        default=None,\n        description=(\n            \"The id of the work pool queue to which this deployment is assigned.\"\n        ),\n    )\n    enforce_parameter_schema: bool = Field(\n        default=False,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Flow","title":"Flow","text":"

    Bases: ORMBaseModel

    An ORM representation of flow data.

    Source code in prefect/server/schemas/core.py
    class Flow(ORMBaseModel):\n    \"\"\"An ORM representation of flow data.\"\"\"\n\n    name: str = Field(\n        default=..., description=\"The name of the flow\", example=\"my-flow\"\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of flow tags\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRun","title":"FlowRun","text":"

    Bases: ORMBaseModel

    An ORM representation of flow run data.

    Source code in prefect/server/schemas/core.py
    class FlowRun(ORMBaseModel):\n    \"\"\"An ORM representation of flow run data.\"\"\"\n\n    name: str = Field(\n        default_factory=lambda: generate_slug(2),\n        description=(\n            \"The name of the flow run. Defaults to a random slug if not specified.\"\n        ),\n        example=\"my-flow-run\",\n    )\n    flow_id: UUID = Field(default=..., description=\"The id of the flow being run.\")\n    state_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the flow run's current state.\"\n    )\n    deployment_id: Optional[UUID] = Field(\n        default=None,\n        description=(\n            \"The id of the deployment associated with this flow run, if available.\"\n        ),\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None, description=\"The work queue that handled this flow run.\"\n    )\n    flow_version: Optional[str] = Field(\n        default=None,\n        description=\"The version of the flow executed in this flow run.\",\n        example=\"1.0\",\n    )\n    parameters: dict = Field(\n        default_factory=dict, description=\"Parameters for the flow run.\"\n    )\n    idempotency_key: Optional[str] = Field(\n        default=None,\n        description=(\n            \"An optional idempotency key for the flow run. Used to ensure the same flow\"\n            \" run is not created multiple times.\"\n        ),\n    )\n    context: dict = Field(\n        default_factory=dict,\n        description=\"Additional context for the flow run.\",\n        example={\"my_var\": \"my_val\"},\n    )\n    empirical_policy: FlowRunPolicy = Field(\n        default_factory=FlowRunPolicy,\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags on the flow run\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    parent_task_run_id: Optional[UUID] = Field(\n        default=None,\n        description=(\n            \"If the flow run is a subflow, the id of the 'dummy' task in the parent\"\n            \" flow used to track subflow state.\"\n        ),\n    )\n\n    state_type: Optional[states.StateType] = Field(\n        default=None, description=\"The type of the current flow run state.\"\n    )\n    state_name: Optional[str] = Field(\n        default=None, description=\"The name of the current flow run state.\"\n    )\n    run_count: int = Field(\n        default=0, description=\"The number of times the flow run was executed.\"\n    )\n    expected_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The flow run's expected start time.\",\n    )\n    next_scheduled_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The next time the flow run is scheduled to start.\",\n    )\n    start_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual start time.\"\n    )\n    end_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual end time.\"\n    )\n    total_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=(\n            \"Total run time. If the flow run was executed multiple times, the time of\"\n            \" each run will be summed.\"\n        ),\n    )\n    estimated_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"A real-time estimate of the total run time.\",\n    )\n    estimated_start_time_delta: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"The difference between actual and expected start time.\",\n    )\n    auto_scheduled: bool = Field(\n        default=False,\n        description=\"Whether or not the flow run was automatically scheduled.\",\n    )\n    infrastructure_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining infrastructure to use this flow run.\",\n    )\n    infrastructure_pid: Optional[str] = Field(\n        default=None,\n        description=\"The id of the flow run as returned by an infrastructure block.\",\n    )\n    created_by: Optional[CreatedBy] = Field(\n        default=None,\n        description=\"Optional information about the creator of this flow run.\",\n    )\n    work_queue_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the run's work pool queue.\"\n    )\n\n    # relationships\n    # flow: Flow = None\n    # task_runs: List[\"TaskRun\"] = Field(default_factory=list)\n    state: Optional[states.State] = Field(\n        default=None, description=\"The current state of the flow run.\"\n    )\n    # parent_task_run: \"TaskRun\" = None\n\n    @validator(\"name\", pre=True)\n    def set_name(cls, name):\n        return name or generate_slug(2)\n\n    def __eq__(self, other: Any) -> bool:\n        \"\"\"\n        Check for \"equality\" to another flow run schema\n\n        Estimates times are rolling and will always change with repeated queries for\n        a flow run so we ignore them during equality checks.\n        \"\"\"\n        if isinstance(other, FlowRun):\n            exclude_fields = {\"estimated_run_time\", \"estimated_start_time_delta\"}\n            return self.dict(exclude=exclude_fields) == other.dict(\n                exclude=exclude_fields\n            )\n        return super().__eq__(other)\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRunNotificationPolicy","title":"FlowRunNotificationPolicy","text":"

    Bases: ORMBaseModel

    An ORM representation of a flow run notification.

    Source code in prefect/server/schemas/core.py
    class FlowRunNotificationPolicy(ORMBaseModel):\n    \"\"\"An ORM representation of a flow run notification.\"\"\"\n\n    is_active: bool = Field(\n        default=True, description=\"Whether the policy is currently active\"\n    )\n    state_names: List[str] = Field(\n        default=..., description=\"The flow run states that trigger notifications\"\n    )\n    tags: List[str] = Field(\n        default=...,\n        description=\"The flow run tags that trigger notifications (set [] to disable)\",\n    )\n    block_document_id: UUID = Field(\n        default=..., description=\"The block document ID used for sending notifications\"\n    )\n    message_template: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A templatable notification message. Use {braces} to add variables.\"\n            \" Valid variables include:\"\n            f\" {listrepr(sorted(FLOW_RUN_NOTIFICATION_TEMPLATE_KWARGS), sep=', ')}\"\n        ),\n        example=(\n            \"Flow run {flow_run_name} with id {flow_run_id} entered state\"\n            \" {flow_run_state_name}.\"\n        ),\n    )\n\n    @validator(\"message_template\")\n    def validate_message_template_variables(cls, v):\n        if v is not None:\n            try:\n                v.format(**{k: \"test\" for k in FLOW_RUN_NOTIFICATION_TEMPLATE_KWARGS})\n            except KeyError as exc:\n                raise ValueError(f\"Invalid template variable provided: '{exc.args[0]}'\")\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRunPolicy","title":"FlowRunPolicy","text":"

    Bases: PrefectBaseModel

    Defines of how a flow run should retry.

    Source code in prefect/server/schemas/core.py
    class FlowRunPolicy(PrefectBaseModel):\n    \"\"\"Defines of how a flow run should retry.\"\"\"\n\n    # TODO: Determine how to separate between infrastructure and within-process level\n    #       retries\n    max_retries: int = Field(\n        default=0,\n        description=(\n            \"The maximum number of retries. Field is not used. Please use `retries`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retry_delay_seconds: float = Field(\n        default=0,\n        description=(\n            \"The delay between retries. Field is not used. Please use `retry_delay`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retries: Optional[int] = Field(default=None, description=\"The number of retries.\")\n    retry_delay: Optional[int] = Field(\n        default=None, description=\"The delay time between retries, in seconds.\"\n    )\n    pause_keys: Optional[set] = Field(\n        default_factory=set, description=\"Tracks pauses this run has observed.\"\n    )\n    resuming: Optional[bool] = Field(\n        default=False, description=\"Indicates if this run is resuming from a pause.\"\n    )\n\n    @root_validator\n    def populate_deprecated_fields(cls, values):\n        \"\"\"\n        If deprecated fields are provided, populate the corresponding new fields\n        to preserve orchestration behavior.\n        \"\"\"\n        if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n            values[\"retries\"] = values[\"max_retries\"]\n        if (\n            not values.get(\"retry_delay\", None)\n            and values.get(\"retry_delay_seconds\", 0) != 0\n        ):\n            values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n        return values\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRunPolicy.populate_deprecated_fields","title":"populate_deprecated_fields","text":"

    If deprecated fields are provided, populate the corresponding new fields to preserve orchestration behavior.

    Source code in prefect/server/schemas/core.py
    @root_validator\ndef populate_deprecated_fields(cls, values):\n    \"\"\"\n    If deprecated fields are provided, populate the corresponding new fields\n    to preserve orchestration behavior.\n    \"\"\"\n    if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n        values[\"retries\"] = values[\"max_retries\"]\n    if (\n        not values.get(\"retry_delay\", None)\n        and values.get(\"retry_delay_seconds\", 0) != 0\n    ):\n        values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n    return values\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRunnerSettings","title":"FlowRunnerSettings","text":"

    Bases: PrefectBaseModel

    An API schema for passing details about the flow runner.

    This schema is agnostic to the types and configuration provided by clients

    Source code in prefect/server/schemas/core.py
    class FlowRunnerSettings(PrefectBaseModel):\n    \"\"\"\n    An API schema for passing details about the flow runner.\n\n    This schema is agnostic to the types and configuration provided by clients\n    \"\"\"\n\n    type: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The type of the flow runner which can be used by the client for\"\n            \" dispatching.\"\n        ),\n    )\n    config: Optional[dict] = Field(\n        default=None, description=\"The configuration for the given flow runner type.\"\n    )\n\n    # The following is required for composite compatibility in the ORM\n\n    def __init__(self, type: str = None, config: dict = None, **kwargs) -> None:\n        # Pydantic does not support positional arguments so they must be converted to\n        # keyword arguments\n        super().__init__(type=type, config=config, **kwargs)\n\n    def __composite_values__(self):\n        return self.type, self.config\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Log","title":"Log","text":"

    Bases: ORMBaseModel

    An ORM representation of log data.

    Source code in prefect/server/schemas/core.py
    class Log(ORMBaseModel):\n    \"\"\"An ORM representation of log data.\"\"\"\n\n    name: str = Field(default=..., description=\"The logger name.\")\n    level: int = Field(default=..., description=\"The log level.\")\n    message: str = Field(default=..., description=\"The log message.\")\n    timestamp: DateTimeTZ = Field(default=..., description=\"The log timestamp.\")\n    flow_run_id: Optional[UUID] = Field(\n        default=None, description=\"The flow run ID associated with the log.\"\n    )\n    task_run_id: Optional[UUID] = Field(\n        default=None, description=\"The task run ID associated with the log.\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.QueueFilter","title":"QueueFilter","text":"

    Bases: PrefectBaseModel

    Filter criteria definition for a work queue.

    Source code in prefect/server/schemas/core.py
    class QueueFilter(PrefectBaseModel):\n    \"\"\"Filter criteria definition for a work queue.\"\"\"\n\n    tags: Optional[List[str]] = Field(\n        default=None,\n        description=\"Only include flow runs with these tags in the work queue.\",\n    )\n    deployment_ids: Optional[List[UUID]] = Field(\n        default=None,\n        description=\"Only include flow runs from these deployments in the work queue.\",\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.SavedSearch","title":"SavedSearch","text":"

    Bases: ORMBaseModel

    An ORM representation of saved search data. Represents a set of filter criteria.

    Source code in prefect/server/schemas/core.py
    class SavedSearch(ORMBaseModel):\n    \"\"\"An ORM representation of saved search data. Represents a set of filter criteria.\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the saved search.\")\n    filters: List[SavedSearchFilter] = Field(\n        default_factory=list, description=\"The filter set for the saved search.\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.SavedSearchFilter","title":"SavedSearchFilter","text":"

    Bases: PrefectBaseModel

    A filter for a saved search model. Intended for use by the Prefect UI.

    Source code in prefect/server/schemas/core.py
    class SavedSearchFilter(PrefectBaseModel):\n    \"\"\"A filter for a saved search model. Intended for use by the Prefect UI.\"\"\"\n\n    object: str = Field(default=..., description=\"The object over which to filter.\")\n    property: str = Field(\n        default=..., description=\"The property of the object on which to filter.\"\n    )\n    type: str = Field(default=..., description=\"The type of the property.\")\n    operation: str = Field(\n        default=...,\n        description=\"The operator to apply to the object. For example, `equals`.\",\n    )\n    value: Any = Field(\n        default=..., description=\"A JSON-compatible value for the filter.\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.TaskRun","title":"TaskRun","text":"

    Bases: ORMBaseModel

    An ORM representation of task run data.

    Source code in prefect/server/schemas/core.py
    class TaskRun(ORMBaseModel):\n    \"\"\"An ORM representation of task run data.\"\"\"\n\n    name: str = Field(default_factory=lambda: generate_slug(2), example=\"my-task-run\")\n    flow_run_id: Optional[UUID] = Field(\n        default=None, description=\"The flow run id of the task run.\"\n    )\n    task_key: str = Field(\n        default=..., description=\"A unique identifier for the task being run.\"\n    )\n    dynamic_key: str = Field(\n        default=...,\n        description=(\n            \"A dynamic key used to differentiate between multiple runs of the same task\"\n            \" within the same flow run.\"\n        ),\n    )\n    cache_key: Optional[str] = Field(\n        default=None,\n        description=(\n            \"An optional cache key. If a COMPLETED state associated with this cache key\"\n            \" is found, the cached COMPLETED state will be used instead of executing\"\n            \" the task run.\"\n        ),\n    )\n    cache_expiration: Optional[DateTimeTZ] = Field(\n        default=None, description=\"Specifies when the cached state should expire.\"\n    )\n    task_version: Optional[str] = Field(\n        default=None, description=\"The version of the task being run.\"\n    )\n    empirical_policy: TaskRunPolicy = Field(\n        default_factory=TaskRunPolicy,\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags for the task run.\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    state_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the current task run state.\"\n    )\n    task_inputs: Dict[str, List[Union[TaskRunResult, Parameter, Constant]]] = Field(\n        default_factory=dict,\n        description=(\n            \"Tracks the source of inputs to a task run. Used for internal bookkeeping.\"\n        ),\n    )\n    state_type: Optional[states.StateType] = Field(\n        default=None, description=\"The type of the current task run state.\"\n    )\n    state_name: Optional[str] = Field(\n        default=None, description=\"The name of the current task run state.\"\n    )\n    run_count: int = Field(\n        default=0, description=\"The number of times the task run has been executed.\"\n    )\n    flow_run_run_count: int = Field(\n        default=0,\n        description=(\n            \"If the parent flow has retried, this indicates the flow retry this run is\"\n            \" associated with.\"\n        ),\n    )\n    expected_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The task run's expected start time.\",\n    )\n\n    # the next scheduled start time will be populated\n    # whenever the run is in a scheduled state\n    next_scheduled_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The next time the task run is scheduled to start.\",\n    )\n    start_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual start time.\"\n    )\n    end_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual end time.\"\n    )\n    total_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=(\n            \"Total run time. If the task run was executed multiple times, the time of\"\n            \" each run will be summed.\"\n        ),\n    )\n    estimated_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"A real-time estimate of total run time.\",\n    )\n    estimated_start_time_delta: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"The difference between actual and expected start time.\",\n    )\n\n    # relationships\n    # flow_run: FlowRun = None\n    # subflow_runs: List[FlowRun] = Field(default_factory=list)\n    state: Optional[states.State] = Field(\n        default=None, description=\"The current task run state.\"\n    )\n\n    @validator(\"name\", pre=True)\n    def set_name(cls, name):\n        return name or generate_slug(2)\n\n    @validator(\"cache_key\")\n    def validate_cache_key_length(cls, cache_key):\n        if cache_key and len(cache_key) > PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH.value():\n            raise ValueError(\n                \"Cache key exceeded maximum allowed length of\"\n                f\" {PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH.value()} characters.\"\n            )\n        return cache_key\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkPool","title":"WorkPool","text":"

    Bases: ORMBaseModel

    An ORM representation of a work pool

    Source code in prefect/server/schemas/core.py
    class WorkPool(ORMBaseModel):\n    \"\"\"An ORM representation of a work pool\"\"\"\n\n    name: str = Field(\n        description=\"The name of the work pool.\",\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"A description of the work pool.\"\n    )\n    type: str = Field(description=\"The work pool type.\")\n    base_job_template: Dict[str, Any] = Field(\n        default_factory=dict, description=\"The work pool's base job template.\"\n    )\n    is_paused: bool = Field(\n        default=False,\n        description=\"Pausing the work pool stops the delivery of all work.\",\n    )\n    concurrency_limit: Optional[conint(ge=0)] = Field(\n        default=None, description=\"A concurrency limit for the work pool.\"\n    )\n    # this required field has a default of None so that the custom validator\n    # below will be called and produce a more helpful error message\n    default_queue_id: UUID = Field(\n        None, description=\"The id of the pool's default queue.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n\n    @validator(\"default_queue_id\", always=True)\n    def helpful_error_for_missing_default_queue_id(cls, v):\n        \"\"\"\n        Default queue ID is required because all pools must have a default queue\n        ID, but it represents a circular foreign key relationship to a\n        WorkQueue (which can't be created until the work pool exists).\n        Therefore, while this field can *technically* be null, it shouldn't be.\n        This should only be an issue when creating new pools, as reading\n        existing ones will always have this field populated. This custom error\n        message will help users understand that they should use the\n        `actions.WorkPoolCreate` model in that case.\n        \"\"\"\n        if v is None:\n            raise ValueError(\n                \"`default_queue_id` is a required field. If you are \"\n                \"creating a new WorkPool and don't have a queue \"\n                \"ID yet, use the `actions.WorkPoolCreate` model instead.\"\n            )\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkPool.helpful_error_for_missing_default_queue_id","title":"helpful_error_for_missing_default_queue_id","text":"

    Default queue ID is required because all pools must have a default queue ID, but it represents a circular foreign key relationship to a WorkQueue (which can't be created until the work pool exists). Therefore, while this field can technically be null, it shouldn't be. This should only be an issue when creating new pools, as reading existing ones will always have this field populated. This custom error message will help users understand that they should use the actions.WorkPoolCreate model in that case.

    Source code in prefect/server/schemas/core.py
    @validator(\"default_queue_id\", always=True)\ndef helpful_error_for_missing_default_queue_id(cls, v):\n    \"\"\"\n    Default queue ID is required because all pools must have a default queue\n    ID, but it represents a circular foreign key relationship to a\n    WorkQueue (which can't be created until the work pool exists).\n    Therefore, while this field can *technically* be null, it shouldn't be.\n    This should only be an issue when creating new pools, as reading\n    existing ones will always have this field populated. This custom error\n    message will help users understand that they should use the\n    `actions.WorkPoolCreate` model in that case.\n    \"\"\"\n    if v is None:\n        raise ValueError(\n            \"`default_queue_id` is a required field. If you are \"\n            \"creating a new WorkPool and don't have a queue \"\n            \"ID yet, use the `actions.WorkPoolCreate` model instead.\"\n        )\n    return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkQueue","title":"WorkQueue","text":"

    Bases: ORMBaseModel

    An ORM representation of a work queue

    Source code in prefect/server/schemas/core.py
    class WorkQueue(ORMBaseModel):\n    \"\"\"An ORM representation of a work queue\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the work queue.\")\n    description: Optional[str] = Field(\n        default=\"\", description=\"An optional description for the work queue.\"\n    )\n    is_paused: bool = Field(\n        default=False, description=\"Whether or not the work queue is paused.\"\n    )\n    concurrency_limit: Optional[conint(ge=0)] = Field(\n        default=None, description=\"An optional concurrency limit for the work queue.\"\n    )\n    priority: conint(ge=1) = Field(\n        default=1,\n        description=(\n            \"The queue's priority. Lower values are higher priority (1 is the highest).\"\n        ),\n    )\n    # Will be required after a future migration\n    work_pool_id: Optional[UUID] = Field(\n        default=None, description=\"The work pool with which the queue is associated.\"\n    )\n    filter: Optional[QueueFilter] = Field(\n        default=None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n    last_polled: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The last time an agent polled this queue for work.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkQueueHealthPolicy","title":"WorkQueueHealthPolicy","text":"

    Bases: PrefectBaseModel

    Source code in prefect/server/schemas/core.py
    class WorkQueueHealthPolicy(PrefectBaseModel):\n    maximum_late_runs: Optional[int] = Field(\n        default=0,\n        description=(\n            \"The maximum number of late runs in the work queue before it is deemed\"\n            \" unhealthy. Defaults to `0`.\"\n        ),\n    )\n    maximum_seconds_since_last_polled: Optional[int] = Field(\n        default=60,\n        description=(\n            \"The maximum number of time in seconds elapsed since work queue has been\"\n            \" polled before it is deemed unhealthy. Defaults to `60`.\"\n        ),\n    )\n\n    def evaluate_health_status(\n        self, late_runs_count: int, last_polled: Optional[DateTimeTZ] = None\n    ) -> bool:\n        \"\"\"\n        Given empirical information about the state of the work queue, evaluate its health status.\n\n        Args:\n            late_runs: the count of late runs for the work queue.\n            last_polled: the last time the work queue was polled, if available.\n\n        Returns:\n            bool: whether or not the work queue is healthy.\n        \"\"\"\n        healthy = True\n        if (\n            self.maximum_late_runs is not None\n            and late_runs_count > self.maximum_late_runs\n        ):\n            healthy = False\n\n        if self.maximum_seconds_since_last_polled is not None:\n            if (\n                last_polled is None\n                or pendulum.now(\"UTC\").diff(last_polled).in_seconds()\n                > self.maximum_seconds_since_last_polled\n            ):\n                healthy = False\n\n        return healthy\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkQueueHealthPolicy.evaluate_health_status","title":"evaluate_health_status","text":"

    Given empirical information about the state of the work queue, evaluate its health status.

    Parameters:

    Name Type Description Default late_runs

    the count of late runs for the work queue.

    required last_polled Optional[DateTimeTZ]

    the last time the work queue was polled, if available.

    None

    Returns:

    Name Type Description bool bool

    whether or not the work queue is healthy.

    Source code in prefect/server/schemas/core.py
    def evaluate_health_status(\n    self, late_runs_count: int, last_polled: Optional[DateTimeTZ] = None\n) -> bool:\n    \"\"\"\n    Given empirical information about the state of the work queue, evaluate its health status.\n\n    Args:\n        late_runs: the count of late runs for the work queue.\n        last_polled: the last time the work queue was polled, if available.\n\n    Returns:\n        bool: whether or not the work queue is healthy.\n    \"\"\"\n    healthy = True\n    if (\n        self.maximum_late_runs is not None\n        and late_runs_count > self.maximum_late_runs\n    ):\n        healthy = False\n\n    if self.maximum_seconds_since_last_polled is not None:\n        if (\n            last_polled is None\n            or pendulum.now(\"UTC\").diff(last_polled).in_seconds()\n            > self.maximum_seconds_since_last_polled\n        ):\n            healthy = False\n\n    return healthy\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Worker","title":"Worker","text":"

    Bases: ORMBaseModel

    An ORM representation of a worker

    Source code in prefect/server/schemas/core.py
    class Worker(ORMBaseModel):\n    \"\"\"An ORM representation of a worker\"\"\"\n\n    name: str = Field(description=\"The name of the worker.\")\n    work_pool_id: UUID = Field(\n        description=\"The work pool with which the queue is associated.\"\n    )\n    last_heartbeat_time: datetime.datetime = Field(\n        None, description=\"The last time the worker process sent a heartbeat.\"\n    )\n    heartbeat_interval_seconds: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The number of seconds to expect between heartbeats sent by the worker.\"\n        ),\n    )\n
    "},{"location":"api-ref/server/schemas/filters/","title":"server.schemas.filters","text":""},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters","title":"prefect.server.schemas.filters","text":"

    Schemas that define Prefect REST API filtering operations.

    Each filter schema includes logic for transforming itself into a SQL where clause.

    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilter","title":"ArtifactCollectionFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter artifact collections. Only artifact collections matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter artifact collections. Only artifact collections matching all criteria will be returned\"\"\"\n\n    latest_id: Optional[ArtifactCollectionFilterLatestId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.id`\"\n    )\n    key: Optional[ArtifactCollectionFilterKey] = Field(\n        default=None, description=\"Filter criteria for `Artifact.key`\"\n    )\n    flow_run_id: Optional[ArtifactCollectionFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.flow_run_id`\"\n    )\n    task_run_id: Optional[ArtifactCollectionFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.task_run_id`\"\n    )\n    type: Optional[ArtifactCollectionFilterType] = Field(\n        default=None, description=\"Filter criteria for `Artifact.type`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.latest_id is not None:\n            filters.append(self.latest_id.as_sql_filter(db))\n        if self.key is not None:\n            filters.append(self.key.as_sql_filter(db))\n        if self.flow_run_id is not None:\n            filters.append(self.flow_run_id.as_sql_filter(db))\n        if self.task_run_id is not None:\n            filters.append(self.task_run_id.as_sql_filter(db))\n        if self.type is not None:\n            filters.append(self.type.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterFlowRunId","title":"ArtifactCollectionFilterFlowRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.flow_run_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterFlowRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.flow_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterFlowRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterKey","title":"ArtifactCollectionFilterKey","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.key.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterKey(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.key`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact keys to include\"\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match artifact keys against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-artifact-%\",\n    )\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If `true`, only include artifacts with a non-null key. If `false`, \"\n            \"only include artifacts with a null key. Should return all rows in \"\n            \"the ArtifactCollection table if specified.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.key.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.ArtifactCollection.key.ilike(f\"%{self.like_}%\"))\n        if self.exists_ is not None:\n            filters.append(\n                db.ArtifactCollection.key.isnot(None)\n                if self.exists_\n                else db.ArtifactCollection.key.is_(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterKey.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterLatestId","title":"ArtifactCollectionFilterLatestId","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.latest_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterLatestId(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.latest_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of artifact ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.latest_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterLatestId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterTaskRunId","title":"ArtifactCollectionFilterTaskRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.task_run_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterTaskRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.task_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterTaskRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterType","title":"ArtifactCollectionFilterType","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.type.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterType(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to exclude\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.type.in_(self.any_))\n        if self.not_any_ is not None:\n            filters.append(db.ArtifactCollection.type.notin_(self.not_any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilter","title":"ArtifactFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter artifacts. Only artifacts matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter artifacts. Only artifacts matching all criteria will be returned\"\"\"\n\n    id: Optional[ArtifactFilterId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.id`\"\n    )\n    key: Optional[ArtifactFilterKey] = Field(\n        default=None, description=\"Filter criteria for `Artifact.key`\"\n    )\n    flow_run_id: Optional[ArtifactFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.flow_run_id`\"\n    )\n    task_run_id: Optional[ArtifactFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.task_run_id`\"\n    )\n    type: Optional[ArtifactFilterType] = Field(\n        default=None, description=\"Filter criteria for `Artifact.type`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.key is not None:\n            filters.append(self.key.as_sql_filter(db))\n        if self.flow_run_id is not None:\n            filters.append(self.flow_run_id.as_sql_filter(db))\n        if self.task_run_id is not None:\n            filters.append(self.task_run_id.as_sql_filter(db))\n        if self.type is not None:\n            filters.append(self.type.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterFlowRunId","title":"ArtifactFilterFlowRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.flow_run_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterFlowRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.flow_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterFlowRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterId","title":"ArtifactFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of artifact ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterKey","title":"ArtifactFilterKey","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.key.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterKey(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.key`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact keys to include\"\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match artifact keys against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-artifact-%\",\n    )\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If `true`, only include artifacts with a non-null key. If `false`, \"\n            \"only include artifacts with a null key.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.key.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Artifact.key.ilike(f\"%{self.like_}%\"))\n        if self.exists_ is not None:\n            filters.append(\n                db.Artifact.key.isnot(None)\n                if self.exists_\n                else db.Artifact.key.is_(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterKey.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterTaskRunId","title":"ArtifactFilterTaskRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.task_run_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterTaskRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.task_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterTaskRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterType","title":"ArtifactFilterType","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.type.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterType(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to exclude\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.type.in_(self.any_))\n        if self.not_any_ is not None:\n            filters.append(db.Artifact.type.notin_(self.not_any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilter","title":"BlockDocumentFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter BlockDocuments. Only BlockDocuments matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter BlockDocuments. Only BlockDocuments matching all criteria will be returned\"\"\"\n\n    id: Optional[BlockDocumentFilterId] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.id`\"\n    )\n    is_anonymous: Optional[BlockDocumentFilterIsAnonymous] = Field(\n        # default is to exclude anonymous blocks\n        BlockDocumentFilterIsAnonymous(eq_=False),\n        description=(\n            \"Filter criteria for `BlockDocument.is_anonymous`. \"\n            \"Defaults to excluding anonymous blocks.\"\n        ),\n    )\n    block_type_id: Optional[BlockDocumentFilterBlockTypeId] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.block_type_id`\"\n    )\n    name: Optional[BlockDocumentFilterName] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.name`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.is_anonymous is not None:\n            filters.append(self.is_anonymous.as_sql_filter(db))\n        if self.block_type_id is not None:\n            filters.append(self.block_type_id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterBlockTypeId","title":"BlockDocumentFilterBlockTypeId","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockDocument.block_type_id.

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilterBlockTypeId(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockDocument.block_type_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block type ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockDocument.block_type_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterBlockTypeId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterId","title":"BlockDocumentFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockDocument.id.

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockDocument.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockDocument.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterIsAnonymous","title":"BlockDocumentFilterIsAnonymous","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockDocument.is_anonymous.

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilterIsAnonymous(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockDocument.is_anonymous`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Filter block documents for only those that are or are not anonymous.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.eq_ is not None:\n            filters.append(db.BlockDocument.is_anonymous.is_(self.eq_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterIsAnonymous.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterName","title":"BlockDocumentFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockDocument.name.

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockDocument.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of block names to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match block names against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-block%\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockDocument.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.BlockDocument.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilter","title":"BlockSchemaFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter BlockSchemas

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter BlockSchemas\"\"\"\n\n    block_type_id: Optional[BlockSchemaFilterBlockTypeId] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.block_type_id`\"\n    )\n    block_capabilities: Optional[BlockSchemaFilterCapabilities] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.capabilities`\"\n    )\n    id: Optional[BlockSchemaFilterId] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.id`\"\n    )\n    version: Optional[BlockSchemaFilterVersion] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.version`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.block_type_id is not None:\n            filters.append(self.block_type_id.as_sql_filter(db))\n        if self.block_capabilities is not None:\n            filters.append(self.block_capabilities.as_sql_filter(db))\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.version is not None:\n            filters.append(self.version.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterBlockTypeId","title":"BlockSchemaFilterBlockTypeId","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockSchema.block_type_id.

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilterBlockTypeId(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockSchema.block_type_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block type ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockSchema.block_type_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterBlockTypeId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterCapabilities","title":"BlockSchemaFilterCapabilities","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockSchema.capabilities

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilterCapabilities(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockSchema.capabilities`\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"write-storage\", \"read-storage\"],\n        description=(\n            \"A list of block capabilities. Block entities will be returned only if an\"\n            \" associated block schema has a superset of the defined capabilities.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.BlockSchema.capabilities, self.all_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterCapabilities.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterId","title":"BlockSchemaFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockSchema.id

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by BlockSchema.id\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockSchema.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterVersion","title":"BlockSchemaFilterVersion","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockSchema.capabilities

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilterVersion(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockSchema.capabilities`\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"2.0.0\", \"2.1.0\"],\n        description=\"A list of block schema versions.\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        pass\n\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockSchema.version.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterVersion.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilter","title":"BlockTypeFilter","text":"

    Bases: PrefectFilterBaseModel

    Filter BlockTypes

    Source code in prefect/server/schemas/filters.py
    class BlockTypeFilter(PrefectFilterBaseModel):\n    \"\"\"Filter BlockTypes\"\"\"\n\n    name: Optional[BlockTypeFilterName] = Field(\n        default=None, description=\"Filter criteria for `BlockType.name`\"\n    )\n\n    slug: Optional[BlockTypeFilterSlug] = Field(\n        default=None, description=\"Filter criteria for `BlockType.slug`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.slug is not None:\n            filters.append(self.slug.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilterName","title":"BlockTypeFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockType.name

    Source code in prefect/server/schemas/filters.py
    class BlockTypeFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockType.name`\"\"\"\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.like_ is not None:\n            filters.append(db.BlockType.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilterSlug","title":"BlockTypeFilterSlug","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockType.slug

    Source code in prefect/server/schemas/filters.py
    class BlockTypeFilterSlug(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockType.slug`\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of slugs to match\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockType.slug.in_(self.any_))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilterSlug.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilter","title":"DeploymentFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter for deployments. Only deployments matching all criteria will be returned.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter for deployments. Only deployments matching all criteria will be returned.\"\"\"\n\n    id: Optional[DeploymentFilterId] = Field(\n        default=None, description=\"Filter criteria for `Deployment.id`\"\n    )\n    name: Optional[DeploymentFilterName] = Field(\n        default=None, description=\"Filter criteria for `Deployment.name`\"\n    )\n    is_schedule_active: Optional[DeploymentFilterIsScheduleActive] = Field(\n        default=None, description=\"Filter criteria for `Deployment.is_schedule_active`\"\n    )\n    tags: Optional[DeploymentFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Deployment.tags`\"\n    )\n    work_queue_name: Optional[DeploymentFilterWorkQueueName] = Field(\n        default=None, description=\"Filter criteria for `Deployment.work_queue_name`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.is_schedule_active is not None:\n            filters.append(self.is_schedule_active.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n        if self.work_queue_name is not None:\n            filters.append(self.work_queue_name.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterId","title":"DeploymentFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Deployment.id.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Deployment.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of deployment ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Deployment.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterIsScheduleActive","title":"DeploymentFilterIsScheduleActive","text":"

    Bases: PrefectFilterBaseModel

    Filter by Deployment.is_schedule_active.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterIsScheduleActive(PrefectFilterBaseModel):\n    \"\"\"Filter by `Deployment.is_schedule_active`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=\"Only returns where deployment schedule is/is not active\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.eq_ is not None:\n            filters.append(db.Deployment.is_schedule_active.is_(self.eq_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterIsScheduleActive.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterName","title":"DeploymentFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Deployment.name.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Deployment.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of deployment names to include\",\n        example=[\"my-deployment-1\", \"my-deployment-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Deployment.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Deployment.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterTags","title":"DeploymentFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by Deployment.tags.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `Deployment.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Deployments will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include deployments without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.Deployment.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.Deployment.tags == [] if self.is_null_ else db.Deployment.tags != []\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterWorkQueueName","title":"DeploymentFilterWorkQueueName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Deployment.work_queue_name.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterWorkQueueName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Deployment.work_queue_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"work_queue_1\", \"work_queue_2\"],\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Deployment.work_queue_name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterWorkQueueName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FilterSet","title":"FilterSet","text":"

    Bases: PrefectBaseModel

    A collection of filters for common objects

    Source code in prefect/server/schemas/filters.py
    class FilterSet(PrefectBaseModel):\n    \"\"\"A collection of filters for common objects\"\"\"\n\n    flows: FlowFilter = Field(\n        default_factory=FlowFilter, description=\"Filters that apply to flows\"\n    )\n    flow_runs: FlowRunFilter = Field(\n        default_factory=FlowRunFilter, description=\"Filters that apply to flow runs\"\n    )\n    task_runs: TaskRunFilter = Field(\n        default_factory=TaskRunFilter, description=\"Filters that apply to task runs\"\n    )\n    deployments: DeploymentFilter = Field(\n        default_factory=DeploymentFilter,\n        description=\"Filters that apply to deployments\",\n    )\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FilterSet.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilter","title":"FlowFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter for flows. Only flows matching all criteria will be returned.

    Source code in prefect/server/schemas/filters.py
    class FlowFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter for flows. Only flows matching all criteria will be returned.\"\"\"\n\n    id: Optional[FlowFilterId] = Field(\n        default=None, description=\"Filter criteria for `Flow.id`\"\n    )\n    deployment: Optional[FlowFilterDeployment] = Field(\n        default=None, description=\"Filter criteria for Flow deployments\"\n    )\n    name: Optional[FlowFilterName] = Field(\n        default=None, description=\"Filter criteria for `Flow.name`\"\n    )\n    tags: Optional[FlowFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Flow.tags`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.deployment is not None:\n            filters.append(self.deployment.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterDeployment","title":"FlowFilterDeployment","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by flows by deployment

    Source code in prefect/server/schemas/filters.py
    class FlowFilterDeployment(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by flows by deployment\"\"\"\n\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flows without deployments\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.is_null_ is not None:\n            deployments_subquery = (\n                sa.select(db.Deployment.flow_id).distinct().subquery()\n            )\n\n            if self.is_null_:\n                filters.append(\n                    db.Flow.id.not_in(sa.select(deployments_subquery.c.flow_id))\n                )\n            else:\n                filters.append(\n                    db.Flow.id.in_(sa.select(deployments_subquery.c.flow_id))\n                )\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterDeployment.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterId","title":"FlowFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Flow.id.

    Source code in prefect/server/schemas/filters.py
    class FlowFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Flow.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Flow.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterName","title":"FlowFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Flow.name.

    Source code in prefect/server/schemas/filters.py
    class FlowFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Flow.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of flow names to include\",\n        example=[\"my-flow-1\", \"my-flow-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Flow.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Flow.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterTags","title":"FlowFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by Flow.tags.

    Source code in prefect/server/schemas/filters.py
    class FlowFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `Flow.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Flows will be returned only if their tags are a superset\"\n            \" of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include flows without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.Flow.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(db.Flow.tags == [] if self.is_null_ else db.Flow.tags != [])\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilter","title":"FlowRunFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter flow runs. Only flow runs matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter flow runs. Only flow runs matching all criteria will be returned\"\"\"\n\n    id: Optional[FlowRunFilterId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.id`\"\n    )\n    name: Optional[FlowRunFilterName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.name`\"\n    )\n    tags: Optional[FlowRunFilterTags] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.tags`\"\n    )\n    deployment_id: Optional[FlowRunFilterDeploymentId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.deployment_id`\"\n    )\n    work_queue_name: Optional[FlowRunFilterWorkQueueName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.work_queue_name\"\n    )\n    state: Optional[FlowRunFilterState] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.state`\"\n    )\n    flow_version: Optional[FlowRunFilterFlowVersion] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.flow_version`\"\n    )\n    start_time: Optional[FlowRunFilterStartTime] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.start_time`\"\n    )\n    expected_start_time: Optional[FlowRunFilterExpectedStartTime] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.expected_start_time`\"\n    )\n    next_scheduled_start_time: Optional[FlowRunFilterNextScheduledStartTime] = Field(\n        default=None,\n        description=\"Filter criteria for `FlowRun.next_scheduled_start_time`\",\n    )\n    parent_flow_run_id: Optional[FlowRunFilterParentFlowRunId] = Field(\n        default=None, description=\"Filter criteria for subflows of the given flow runs\"\n    )\n    parent_task_run_id: Optional[FlowRunFilterParentTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.parent_task_run_id`\"\n    )\n    idempotency_key: Optional[FlowRunFilterIdempotencyKey] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.idempotency_key`\"\n    )\n\n    def only_filters_on_id(self):\n        return (\n            self.id is not None\n            and (self.id.any_ and not self.id.not_any_)\n            and self.name is None\n            and self.tags is None\n            and self.deployment_id is None\n            and self.work_queue_name is None\n            and self.state is None\n            and self.flow_version is None\n            and self.start_time is None\n            and self.expected_start_time is None\n            and self.next_scheduled_start_time is None\n            and self.parent_flow_run_id is None\n            and self.parent_task_run_id is None\n            and self.idempotency_key is None\n        )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n        if self.deployment_id is not None:\n            filters.append(self.deployment_id.as_sql_filter(db))\n        if self.work_queue_name is not None:\n            filters.append(self.work_queue_name.as_sql_filter(db))\n        if self.flow_version is not None:\n            filters.append(self.flow_version.as_sql_filter(db))\n        if self.state is not None:\n            filters.append(self.state.as_sql_filter(db))\n        if self.start_time is not None:\n            filters.append(self.start_time.as_sql_filter(db))\n        if self.expected_start_time is not None:\n            filters.append(self.expected_start_time.as_sql_filter(db))\n        if self.next_scheduled_start_time is not None:\n            filters.append(self.next_scheduled_start_time.as_sql_filter(db))\n        if self.parent_flow_run_id is not None:\n            filters.append(self.parent_flow_run_id.as_sql_filter(db))\n        if self.parent_task_run_id is not None:\n            filters.append(self.parent_task_run_id.as_sql_filter(db))\n        if self.idempotency_key is not None:\n            filters.append(self.idempotency_key.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterDeploymentId","title":"FlowRunFilterDeploymentId","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.deployment_id.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterDeploymentId(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.deployment_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run deployment ids to include\"\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without deployment ids\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.deployment_id.in_(self.any_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.deployment_id.is_(None)\n                if self.is_null_\n                else db.FlowRun.deployment_id.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterDeploymentId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterExpectedStartTime","title":"FlowRunFilterExpectedStartTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.expected_start_time.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterExpectedStartTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.expected_start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs scheduled to start at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs scheduled to start at or after this time\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.FlowRun.expected_start_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.FlowRun.expected_start_time >= self.after_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterExpectedStartTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterFlowVersion","title":"FlowRunFilterFlowVersion","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.flow_version.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterFlowVersion(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.flow_version`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run flow_versions to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.flow_version.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterFlowVersion.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterId","title":"FlowRunFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.id.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run ids to include\"\n    )\n    not_any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run ids to exclude\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.id.in_(self.any_))\n        if self.not_any_ is not None:\n            filters.append(db.FlowRun.id.not_in(self.not_any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterIdempotencyKey","title":"FlowRunFilterIdempotencyKey","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.idempotency_key.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterIdempotencyKey(PrefectFilterBaseModel):\n    \"\"\"Filter by FlowRun.idempotency_key.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run idempotency keys to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run idempotency keys to exclude\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.idempotency_key.in_(self.any_))\n        if self.not_any_ is not None:\n            filters.append(db.FlowRun.idempotency_key.not_in(self.not_any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterIdempotencyKey.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterName","title":"FlowRunFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.name.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of flow run names to include\",\n        example=[\"my-flow-run-1\", \"my-flow-run-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.FlowRun.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterNextScheduledStartTime","title":"FlowRunFilterNextScheduledStartTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.next_scheduled_start_time.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterNextScheduledStartTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.next_scheduled_start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include flow runs with a next_scheduled_start_time or before this\"\n            \" time\"\n        ),\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include flow runs with a next_scheduled_start_time at or after this\"\n            \" time\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.FlowRun.next_scheduled_start_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.FlowRun.next_scheduled_start_time >= self.after_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterNextScheduledStartTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterParentFlowRunId","title":"FlowRunFilterParentFlowRunId","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter for subflows of a given flow run

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterParentFlowRunId(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter for subflows of a given flow run\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of parent flow run ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(\n                db.FlowRun.id.in_(\n                    sa.select(db.FlowRun.id)\n                    .join(\n                        db.TaskRun,\n                        sa.and_(\n                            db.TaskRun.id == db.FlowRun.parent_task_run_id,\n                        ),\n                    )\n                    .where(db.TaskRun.flow_run_id.in_(self.any_))\n                )\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterParentFlowRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterParentTaskRunId","title":"FlowRunFilterParentTaskRunId","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.parent_task_run_id.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterParentTaskRunId(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.parent_task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run parent_task_run_ids to include\"\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without parent_task_run_id\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.parent_task_run_id.in_(self.any_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.parent_task_run_id.is_(None)\n                if self.is_null_\n                else db.FlowRun.parent_task_run_id.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterParentTaskRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStartTime","title":"FlowRunFilterStartTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.start_time.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterStartTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs starting at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs starting at or after this time\",\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only return flow runs without a start time\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.FlowRun.start_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.FlowRun.start_time >= self.after_)\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.start_time.is_(None)\n                if self.is_null_\n                else db.FlowRun.start_time.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStartTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterState","title":"FlowRunFilterState","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.state_type and FlowRun.state_name.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterState(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.state_type` and `FlowRun.state_name`.\"\"\"\n\n    type: Optional[FlowRunFilterStateType] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.state_type`\"\n    )\n    name: Optional[FlowRunFilterStateName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.state_name`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.type is not None:\n            filters.extend(self.type._get_filter_list(db))\n        if self.name is not None:\n            filters.extend(self.name._get_filter_list(db))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterState.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStateName","title":"FlowRunFilterStateName","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.state_name.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterStateName(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.state_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run state names to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.state_name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStateName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStateType","title":"FlowRunFilterStateType","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.state_type.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterStateType(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.state_type`.\"\"\"\n\n    any_: Optional[List[schemas.states.StateType]] = Field(\n        default=None, description=\"A list of flow run state types to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.state_type.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStateType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterTags","title":"FlowRunFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.tags.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Flow runs will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include flow runs without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.FlowRun.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.tags == [] if self.is_null_ else db.FlowRun.tags != []\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterWorkQueueName","title":"FlowRunFilterWorkQueueName","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.work_queue_name.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterWorkQueueName(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.work_queue_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"work_queue_1\", \"work_queue_2\"],\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without work queue names\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.work_queue_name.in_(self.any_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.work_queue_name.is_(None)\n                if self.is_null_\n                else db.FlowRun.work_queue_name.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterWorkQueueName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunNotificationPolicyFilter","title":"FlowRunNotificationPolicyFilter","text":"

    Bases: PrefectFilterBaseModel

    Filter FlowRunNotificationPolicies.

    Source code in prefect/server/schemas/filters.py
    class FlowRunNotificationPolicyFilter(PrefectFilterBaseModel):\n    \"\"\"Filter FlowRunNotificationPolicies.\"\"\"\n\n    is_active: Optional[FlowRunNotificationPolicyFilterIsActive] = Field(\n        default=FlowRunNotificationPolicyFilterIsActive(eq_=False),\n        description=\"Filter criteria for `FlowRunNotificationPolicy.is_active`. \",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.is_active is not None:\n            filters.append(self.is_active.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunNotificationPolicyFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunNotificationPolicyFilterIsActive","title":"FlowRunNotificationPolicyFilterIsActive","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRunNotificationPolicy.is_active.

    Source code in prefect/server/schemas/filters.py
    class FlowRunNotificationPolicyFilterIsActive(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRunNotificationPolicy.is_active`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Filter notification policies for only those that are or are not active.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.eq_ is not None:\n            filters.append(db.FlowRunNotificationPolicy.is_active.is_(self.eq_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunNotificationPolicyFilterIsActive.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilter","title":"LogFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter logs. Only logs matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class LogFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter logs. Only logs matching all criteria will be returned\"\"\"\n\n    level: Optional[LogFilterLevel] = Field(\n        default=None, description=\"Filter criteria for `Log.level`\"\n    )\n    timestamp: Optional[LogFilterTimestamp] = Field(\n        default=None, description=\"Filter criteria for `Log.timestamp`\"\n    )\n    flow_run_id: Optional[LogFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Log.flow_run_id`\"\n    )\n    task_run_id: Optional[LogFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Log.task_run_id`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.level is not None:\n            filters.append(self.level.as_sql_filter(db))\n        if self.timestamp is not None:\n            filters.append(self.timestamp.as_sql_filter(db))\n        if self.flow_run_id is not None:\n            filters.append(self.flow_run_id.as_sql_filter(db))\n        if self.task_run_id is not None:\n            filters.append(self.task_run_id.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterFlowRunId","title":"LogFilterFlowRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.flow_run_id.

    Source code in prefect/server/schemas/filters.py
    class LogFilterFlowRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Log.flow_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterFlowRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterLevel","title":"LogFilterLevel","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.level.

    Source code in prefect/server/schemas/filters.py
    class LogFilterLevel(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.level`.\"\"\"\n\n    ge_: Optional[int] = Field(\n        default=None,\n        description=\"Include logs with a level greater than or equal to this level\",\n        example=20,\n    )\n\n    le_: Optional[int] = Field(\n        default=None,\n        description=\"Include logs with a level less than or equal to this level\",\n        example=50,\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.ge_ is not None:\n            filters.append(db.Log.level >= self.ge_)\n        if self.le_ is not None:\n            filters.append(db.Log.level <= self.le_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterLevel.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterName","title":"LogFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.name.

    Source code in prefect/server/schemas/filters.py
    class LogFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of log names to include\",\n        example=[\"prefect.logger.flow_runs\", \"prefect.logger.task_runs\"],\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Log.name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterTaskRunId","title":"LogFilterTaskRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.task_run_id.

    Source code in prefect/server/schemas/filters.py
    class LogFilterTaskRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Log.task_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterTaskRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterTimestamp","title":"LogFilterTimestamp","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.timestamp.

    Source code in prefect/server/schemas/filters.py
    class LogFilterTimestamp(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.timestamp`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include logs with a timestamp at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include logs with a timestamp at or after this time\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.Log.timestamp <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.Log.timestamp >= self.after_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterTimestamp.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.Operator","title":"Operator","text":"

    Bases: AutoEnum

    Operators for combining filter criteria.

    Source code in prefect/server/schemas/filters.py
    class Operator(AutoEnum):\n    \"\"\"Operators for combining filter criteria.\"\"\"\n\n    and_ = AutoEnum.auto()\n    or_ = AutoEnum.auto()\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.Operator.auto","title":"auto staticmethod","text":"

    Exposes enum.auto() to avoid requiring a second import to use AutoEnum

    Source code in prefect/utilities/collections.py
    @staticmethod\ndef auto():\n    \"\"\"\n    Exposes `enum.auto()` to avoid requiring a second import to use `AutoEnum`\n    \"\"\"\n    return auto()\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.PrefectFilterBaseModel","title":"PrefectFilterBaseModel","text":"

    Bases: PrefectBaseModel

    Base model for Prefect filters

    Source code in prefect/server/schemas/filters.py
    class PrefectFilterBaseModel(PrefectBaseModel):\n    \"\"\"Base model for Prefect filters\"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    def as_sql_filter(self, db: \"PrefectDBInterface\") -> \"BooleanClauseList\":\n        \"\"\"Generate SQL filter from provided filter parameters. If no filters parameters are available, return a TRUE filter.\"\"\"\n        filters = self._get_filter_list(db)\n        if not filters:\n            return True\n        return sa.and_(*filters)\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        \"\"\"Return a list of boolean filter statements based on filter parameters\"\"\"\n        raise NotImplementedError(\"_get_filter_list must be implemented\")\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.PrefectFilterBaseModel.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.PrefectOperatorFilterBaseModel","title":"PrefectOperatorFilterBaseModel","text":"

    Bases: PrefectFilterBaseModel

    Base model for Prefect filters that combines criteria with a user-provided operator

    Source code in prefect/server/schemas/filters.py
    class PrefectOperatorFilterBaseModel(PrefectFilterBaseModel):\n    \"\"\"Base model for Prefect filters that combines criteria with a user-provided operator\"\"\"\n\n    operator: Operator = Field(\n        default=Operator.and_,\n        description=\"Operator for combining filter criteria. Defaults to 'and_'.\",\n    )\n\n    def as_sql_filter(self, db: \"PrefectDBInterface\") -> \"BooleanClauseList\":\n        filters = self._get_filter_list(db)\n        if not filters:\n            return True\n        return sa.and_(*filters) if self.operator == Operator.and_ else sa.or_(*filters)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.PrefectOperatorFilterBaseModel.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilter","title":"TaskRunFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter task runs. Only task runs matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter task runs. Only task runs matching all criteria will be returned\"\"\"\n\n    id: Optional[TaskRunFilterId] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.id`\"\n    )\n    name: Optional[TaskRunFilterName] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.name`\"\n    )\n    tags: Optional[TaskRunFilterTags] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.tags`\"\n    )\n    state: Optional[TaskRunFilterState] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.state`\"\n    )\n    start_time: Optional[TaskRunFilterStartTime] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.start_time`\"\n    )\n    subflow_runs: Optional[TaskRunFilterSubFlowRuns] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.subflow_run`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n        if self.state is not None:\n            filters.append(self.state.as_sql_filter(db))\n        if self.start_time is not None:\n            filters.append(self.start_time.as_sql_filter(db))\n        if self.subflow_runs is not None:\n            filters.append(self.subflow_runs.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterId","title":"TaskRunFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.id.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.TaskRun.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterName","title":"TaskRunFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.name.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of task run names to include\",\n        example=[\"my-task-run-1\", \"my-task-run-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.TaskRun.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.TaskRun.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStartTime","title":"TaskRunFilterStartTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.start_time.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterStartTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include task runs starting at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include task runs starting at or after this time\",\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only return task runs without a start time\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.TaskRun.start_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.TaskRun.start_time >= self.after_)\n        if self.is_null_ is not None:\n            filters.append(\n                db.TaskRun.start_time.is_(None)\n                if self.is_null_\n                else db.TaskRun.start_time.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStartTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterState","title":"TaskRunFilterState","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by TaskRun.type and TaskRun.name.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterState(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `TaskRun.type` and `TaskRun.name`.\"\"\"\n\n    type: Optional[TaskRunFilterStateType]\n    name: Optional[TaskRunFilterStateName]\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.type is not None:\n            filters.extend(self.type._get_filter_list(db))\n        if self.name is not None:\n            filters.extend(self.name._get_filter_list(db))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterState.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStateName","title":"TaskRunFilterStateName","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.state_name.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterStateName(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.state_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of task run state names to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.TaskRun.state_name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStateName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStateType","title":"TaskRunFilterStateType","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.state_type.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterStateType(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.state_type`.\"\"\"\n\n    any_: Optional[List[schemas.states.StateType]] = Field(\n        default=None, description=\"A list of task run state types to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.TaskRun.state_type.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStateType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterSubFlowRuns","title":"TaskRunFilterSubFlowRuns","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.subflow_run.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterSubFlowRuns(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.subflow_run`.\"\"\"\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If true, only include task runs that are subflow run parents; if false,\"\n            \" exclude parent task runs\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.exists_ is True:\n            filters.append(db.TaskRun.subflow_run.has())\n        elif self.exists_ is False:\n            filters.append(sa.not_(db.TaskRun.subflow_run.has()))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterSubFlowRuns.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterTags","title":"TaskRunFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by TaskRun.tags.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `TaskRun.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Task runs will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include task runs without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.TaskRun.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.TaskRun.tags == [] if self.is_null_ else db.TaskRun.tags != []\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilter","title":"VariableFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter variables. Only variables matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class VariableFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter variables. Only variables matching all criteria will be returned\"\"\"\n\n    id: Optional[VariableFilterId] = Field(\n        default=None, description=\"Filter criteria for `Variable.id`\"\n    )\n    name: Optional[VariableFilterName] = Field(\n        default=None, description=\"Filter criteria for `Variable.name`\"\n    )\n    value: Optional[VariableFilterValue] = Field(\n        default=None, description=\"Filter criteria for `Variable.value`\"\n    )\n    tags: Optional[VariableFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Variable.tags`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.value is not None:\n            filters.append(self.value.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterId","title":"VariableFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Variable.id.

    Source code in prefect/server/schemas/filters.py
    class VariableFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Variable.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of variable ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Variable.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterName","title":"VariableFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Variable.name.

    Source code in prefect/server/schemas/filters.py
    class VariableFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Variable.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of variables names to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match variable names against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my_variable_%\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Variable.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Variable.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterTags","title":"VariableFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by Variable.tags.

    Source code in prefect/server/schemas/filters.py
    class VariableFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `Variable.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Variables will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include Variables without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.Variable.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.Variable.tags == [] if self.is_null_ else db.Variable.tags != []\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterValue","title":"VariableFilterValue","text":"

    Bases: PrefectFilterBaseModel

    Filter by Variable.value.

    Source code in prefect/server/schemas/filters.py
    class VariableFilterValue(PrefectFilterBaseModel):\n    \"\"\"Filter by `Variable.value`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of variables value to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match variable value against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-value-%\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Variable.value.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Variable.value.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterValue.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilter","title":"WorkPoolFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter work pools. Only work pools matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class WorkPoolFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter work pools. Only work pools matching all criteria will be returned\"\"\"\n\n    id: Optional[WorkPoolFilterId] = Field(\n        default=None, description=\"Filter criteria for `WorkPool.id`\"\n    )\n    name: Optional[WorkPoolFilterName] = Field(\n        default=None, description=\"Filter criteria for `WorkPool.name`\"\n    )\n    type: Optional[WorkPoolFilterType] = Field(\n        default=None, description=\"Filter criteria for `WorkPool.type`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.type is not None:\n            filters.append(self.type.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterId","title":"WorkPoolFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkPool.id.

    Source code in prefect/server/schemas/filters.py
    class WorkPoolFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkPool.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of work pool ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkPool.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterName","title":"WorkPoolFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkPool.name.

    Source code in prefect/server/schemas/filters.py
    class WorkPoolFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkPool.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of work pool names to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkPool.name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterType","title":"WorkPoolFilterType","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkPool.type.

    Source code in prefect/server/schemas/filters.py
    class WorkPoolFilterType(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkPool.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of work pool types to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkPool.type.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilter","title":"WorkQueueFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter work queues. Only work queues matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class WorkQueueFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter work queues. Only work queues matching all criteria will be\n    returned\"\"\"\n\n    id: Optional[WorkQueueFilterId] = Field(\n        default=None, description=\"Filter criteria for `WorkQueue.id`\"\n    )\n\n    name: Optional[WorkQueueFilterName] = Field(\n        default=None, description=\"Filter criteria for `WorkQueue.name`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilterId","title":"WorkQueueFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkQueue.id.

    Source code in prefect/server/schemas/filters.py
    class WorkQueueFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkQueue.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None,\n        description=\"A list of work queue ids to include\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkQueue.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilterName","title":"WorkQueueFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkQueue.name.

    Source code in prefect/server/schemas/filters.py
    class WorkQueueFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkQueue.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"wq-1\", \"wq-2\"],\n    )\n\n    startswith_: Optional[List[str]] = Field(\n        default=None,\n        description=(\n            \"A list of case-insensitive starts-with matches. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', and 'Marvin-robot', but not 'sad-marvin'.\"\n        ),\n        example=[\"marvin\", \"Marvin-robot\"],\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkQueue.name.in_(self.any_))\n        if self.startswith_ is not None:\n            filters.append(\n                sa.or_(\n                    *[db.WorkQueue.name.ilike(f\"{item}%\") for item in self.startswith_]\n                )\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilter","title":"WorkerFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by Worker.last_heartbeat_time.

    Source code in prefect/server/schemas/filters.py
    class WorkerFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `Worker.last_heartbeat_time`.\"\"\"\n\n    # worker_config_id: Optional[WorkerFilterWorkPoolId] = Field(\n    #     default=None, description=\"Filter criteria for `Worker.worker_config_id`\"\n    # )\n\n    last_heartbeat_time: Optional[WorkerFilterLastHeartbeatTime] = Field(\n        default=None,\n        description=\"Filter criteria for `Worker.last_heartbeat_time`\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.last_heartbeat_time is not None:\n            filters.append(self.last_heartbeat_time.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilterLastHeartbeatTime","title":"WorkerFilterLastHeartbeatTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by Worker.last_heartbeat_time.

    Source code in prefect/server/schemas/filters.py
    class WorkerFilterLastHeartbeatTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `Worker.last_heartbeat_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include processes whose last heartbeat was at or before this time\"\n        ),\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include processes whose last heartbeat was at or after this time\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.Worker.last_heartbeat_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.Worker.last_heartbeat_time >= self.after_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilterLastHeartbeatTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilterWorkPoolId","title":"WorkerFilterWorkPoolId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Worker.worker_config_id.

    Source code in prefect/server/schemas/filters.py
    class WorkerFilterWorkPoolId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Worker.worker_config_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of work pool ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Worker.worker_config_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilterWorkPoolId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/responses/","title":"server.schemas.responses","text":""},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses","title":"prefect.server.schemas.responses","text":"

    Schemas for special responses from the Prefect REST API.

    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.FlowRunResponse","title":"FlowRunResponse","text":"

    Bases: ORMBaseModel

    Source code in prefect/server/schemas/responses.py
    @copy_model_fields\nclass FlowRunResponse(ORMBaseModel):\n    name: str = FieldFrom(schemas.core.FlowRun)\n    flow_id: UUID = FieldFrom(schemas.core.FlowRun)\n    state_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    deployment_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    work_queue_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    work_queue_name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    flow_version: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parameters: dict = FieldFrom(schemas.core.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    context: dict = FieldFrom(schemas.core.FlowRun)\n    empirical_policy: FlowRunPolicy = FieldFrom(schemas.core.FlowRun)\n    tags: List[str] = FieldFrom(schemas.core.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    state_type: Optional[schemas.states.StateType] = FieldFrom(schemas.core.FlowRun)\n    state_name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    run_count: int = FieldFrom(schemas.core.FlowRun)\n    expected_start_time: Optional[DateTimeTZ] = FieldFrom(schemas.core.FlowRun)\n    next_scheduled_start_time: Optional[DateTimeTZ] = FieldFrom(schemas.core.FlowRun)\n    start_time: Optional[DateTimeTZ] = FieldFrom(schemas.core.FlowRun)\n    end_time: Optional[DateTimeTZ] = FieldFrom(schemas.core.FlowRun)\n    total_run_time: datetime.timedelta = FieldFrom(schemas.core.FlowRun)\n    estimated_run_time: datetime.timedelta = FieldFrom(schemas.core.FlowRun)\n    estimated_start_time_delta: datetime.timedelta = FieldFrom(schemas.core.FlowRun)\n    auto_scheduled: bool = FieldFrom(schemas.core.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    infrastructure_pid: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    created_by: Optional[CreatedBy] = FieldFrom(schemas.core.FlowRun)\n    work_pool_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The id of the flow run's work pool.\",\n    )\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the flow run's work pool.\",\n        example=\"my-work-pool\",\n    )\n    state: Optional[schemas.states.State] = FieldFrom(schemas.core.FlowRun)\n\n    @classmethod\n    def from_orm(cls, orm_flow_run: \"prefect.server.database.orm_models.ORMFlowRun\"):\n        response = super().from_orm(orm_flow_run)\n        if orm_flow_run.work_queue:\n            response.work_queue_id = orm_flow_run.work_queue.id\n            response.work_queue_name = orm_flow_run.work_queue.name\n            if orm_flow_run.work_queue.work_pool:\n                response.work_pool_id = orm_flow_run.work_queue.work_pool.id\n                response.work_pool_name = orm_flow_run.work_queue.work_pool.name\n\n        return response\n\n    def __eq__(self, other: Any) -> bool:\n        \"\"\"\n        Check for \"equality\" to another flow run schema\n\n        Estimates times are rolling and will always change with repeated queries for\n        a flow run so we ignore them during equality checks.\n        \"\"\"\n        if isinstance(other, FlowRunResponse):\n            exclude_fields = {\"estimated_run_time\", \"estimated_start_time_delta\"}\n            return self.dict(exclude=exclude_fields) == other.dict(\n                exclude=exclude_fields\n            )\n        return super().__eq__(other)\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.HistoryResponse","title":"HistoryResponse","text":"

    Bases: PrefectBaseModel

    Represents a history of aggregation states over an interval

    Source code in prefect/server/schemas/responses.py
    class HistoryResponse(PrefectBaseModel):\n    \"\"\"Represents a history of aggregation states over an interval\"\"\"\n\n    interval_start: DateTimeTZ = Field(\n        default=..., description=\"The start date of the interval.\"\n    )\n    interval_end: DateTimeTZ = Field(\n        default=..., description=\"The end date of the interval.\"\n    )\n    states: List[HistoryResponseState] = Field(\n        default=..., description=\"A list of state histories during the interval.\"\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.HistoryResponseState","title":"HistoryResponseState","text":"

    Bases: PrefectBaseModel

    Represents a single state's history over an interval.

    Source code in prefect/server/schemas/responses.py
    class HistoryResponseState(PrefectBaseModel):\n    \"\"\"Represents a single state's history over an interval.\"\"\"\n\n    state_type: schemas.states.StateType = Field(\n        default=..., description=\"The state type.\"\n    )\n    state_name: str = Field(default=..., description=\"The state name.\")\n    count_runs: int = Field(\n        default=...,\n        description=\"The number of runs in the specified state during the interval.\",\n    )\n    sum_estimated_run_time: datetime.timedelta = Field(\n        default=...,\n        description=\"The total estimated run time of all runs during the interval.\",\n    )\n    sum_estimated_lateness: datetime.timedelta = Field(\n        default=...,\n        description=(\n            \"The sum of differences between actual and expected start time during the\"\n            \" interval.\"\n        ),\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.OrchestrationResult","title":"OrchestrationResult","text":"

    Bases: PrefectBaseModel

    A container for the output of state orchestration.

    Source code in prefect/server/schemas/responses.py
    class OrchestrationResult(PrefectBaseModel):\n    \"\"\"\n    A container for the output of state orchestration.\n    \"\"\"\n\n    state: Optional[schemas.states.State]\n    status: SetStateStatus\n    details: StateResponseDetails\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.SetStateStatus","title":"SetStateStatus","text":"

    Bases: AutoEnum

    Enumerates return statuses for setting run states.

    Source code in prefect/server/schemas/responses.py
    class SetStateStatus(AutoEnum):\n    \"\"\"Enumerates return statuses for setting run states.\"\"\"\n\n    ACCEPT = AutoEnum.auto()\n    REJECT = AutoEnum.auto()\n    ABORT = AutoEnum.auto()\n    WAIT = AutoEnum.auto()\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.StateAbortDetails","title":"StateAbortDetails","text":"

    Bases: PrefectBaseModel

    Details associated with an ABORT state transition.

    Source code in prefect/server/schemas/responses.py
    class StateAbortDetails(PrefectBaseModel):\n    \"\"\"Details associated with an ABORT state transition.\"\"\"\n\n    type: Literal[\"abort_details\"] = Field(\n        default=\"abort_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition was aborted.\"\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.StateAcceptDetails","title":"StateAcceptDetails","text":"

    Bases: PrefectBaseModel

    Details associated with an ACCEPT state transition.

    Source code in prefect/server/schemas/responses.py
    class StateAcceptDetails(PrefectBaseModel):\n    \"\"\"Details associated with an ACCEPT state transition.\"\"\"\n\n    type: Literal[\"accept_details\"] = Field(\n        default=\"accept_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.StateRejectDetails","title":"StateRejectDetails","text":"

    Bases: PrefectBaseModel

    Details associated with a REJECT state transition.

    Source code in prefect/server/schemas/responses.py
    class StateRejectDetails(PrefectBaseModel):\n    \"\"\"Details associated with a REJECT state transition.\"\"\"\n\n    type: Literal[\"reject_details\"] = Field(\n        default=\"reject_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition was rejected.\"\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.StateWaitDetails","title":"StateWaitDetails","text":"

    Bases: PrefectBaseModel

    Details associated with a WAIT state transition.

    Source code in prefect/server/schemas/responses.py
    class StateWaitDetails(PrefectBaseModel):\n    \"\"\"Details associated with a WAIT state transition.\"\"\"\n\n    type: Literal[\"wait_details\"] = Field(\n        default=\"wait_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    delay_seconds: int = Field(\n        default=...,\n        description=(\n            \"The length of time in seconds the client should wait before transitioning\"\n            \" states.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition should wait.\"\n    )\n
    "},{"location":"api-ref/server/schemas/schedules/","title":"server.schemas.schedules","text":""},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules","title":"prefect.server.schemas.schedules","text":"

    Schedule schemas

    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.CronSchedule","title":"CronSchedule","text":"

    Bases: PrefectBaseModel

    Cron schedule

    NOTE: If the timezone is a DST-observing one, then the schedule will adjust itself appropriately. Cron's rules for DST are based on schedule times, not intervals. This means that an hourly cron schedule will fire on every new schedule hour, not every elapsed hour; for example, when clocks are set back this will result in a two-hour pause as the schedule will fire the first time 1am is reached and the first time 2am is reached, 120 minutes later. Longer schedules, such as one that fires at 9am every morning, will automatically adjust for DST.

    Parameters:

    Name Type Description Default cron str

    a valid cron string

    required timezone str

    a valid timezone string in IANA tzdata format (for example, America/New_York).

    required day_or bool

    Control how croniter handles day and day_of_week entries. Defaults to True, matching cron which connects those values using OR. If the switch is set to False, the values are connected using AND. This behaves like fcron and enables you to e.g. define a job that executes each 2nd friday of a month by setting the days of month and the weekday.

    required Source code in prefect/server/schemas/schedules.py
    class CronSchedule(PrefectBaseModel):\n    \"\"\"\n    Cron schedule\n\n    NOTE: If the timezone is a DST-observing one, then the schedule will adjust\n    itself appropriately. Cron's rules for DST are based on schedule times, not\n    intervals. This means that an hourly cron schedule will fire on every new\n    schedule hour, not every elapsed hour; for example, when clocks are set back\n    this will result in a two-hour pause as the schedule will fire *the first\n    time* 1am is reached and *the first time* 2am is reached, 120 minutes later.\n    Longer schedules, such as one that fires at 9am every morning, will\n    automatically adjust for DST.\n\n    Args:\n        cron (str): a valid cron string\n        timezone (str): a valid timezone string in IANA tzdata format (for example,\n            America/New_York).\n        day_or (bool, optional): Control how croniter handles `day` and `day_of_week`\n            entries. Defaults to True, matching cron which connects those values using\n            OR. If the switch is set to False, the values are connected using AND. This\n            behaves like fcron and enables you to e.g. define a job that executes each\n            2nd friday of a month by setting the days of month and the weekday.\n\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    cron: str = Field(default=..., example=\"0 0 * * *\")\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n    day_or: bool = Field(\n        default=True,\n        description=(\n            \"Control croniter behavior for handling day and day_of_week entries.\"\n        ),\n    )\n\n    @validator(\"timezone\")\n    def valid_timezone(cls, v):\n        if v and v not in pendulum.tz.timezones:\n            raise ValueError(\n                f'Invalid timezone: \"{v}\" (specify in IANA tzdata format, for example,'\n                \" America/New_York)\"\n            )\n        return v\n\n    @validator(\"cron\")\n    def valid_cron_string(cls, v):\n        # croniter allows \"random\" and \"hashed\" expressions\n        # which we do not support https://github.com/kiorky/croniter\n        if not croniter.is_valid(v):\n            raise ValueError(f'Invalid cron string: \"{v}\"')\n        elif any(c for c in v.split() if c.casefold() in [\"R\", \"H\", \"r\", \"h\"]):\n            raise ValueError(\n                f'Random and Hashed expressions are unsupported, received: \"{v}\"'\n            )\n        return v\n\n    async def get_dates(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> List[pendulum.DateTime]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to None.  If a timezone-naive datetime is\n                provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): The maximum scheduled date to return. If\n                a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: A list of dates\n        \"\"\"\n        return sorted(self._get_dates_generator(n=n, start=start, end=end))\n\n    def _get_dates_generator(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> Generator[pendulum.DateTime, None, None]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to the current date. If a timezone-naive\n                datetime is provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): No returned date will exceed this date.\n                If a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: a list of dates\n        \"\"\"\n        if start is None:\n            start = pendulum.now(\"UTC\")\n\n        start, end = _prepare_scheduling_start_and_end(start, end, self.timezone)\n\n        if n is None:\n            # if an end was supplied, we do our best to supply all matching dates (up to\n            # MAX_ITERATIONS)\n            if end is not None:\n                n = MAX_ITERATIONS\n            else:\n                n = 1\n\n        elif self.timezone:\n            start = start.in_tz(self.timezone)\n\n        # subtract one second from the start date, so that croniter returns it\n        # as an event (if it meets the cron criteria)\n        start = start.subtract(seconds=1)\n\n        # Respect microseconds by rounding up\n        if start.microsecond > 0:\n            start += datetime.timedelta(seconds=1)\n\n        # croniter's DST logic interferes with all other datetime libraries except pytz\n        start_localized = pytz.timezone(start.tz.name).localize(\n            datetime.datetime(\n                year=start.year,\n                month=start.month,\n                day=start.day,\n                hour=start.hour,\n                minute=start.minute,\n                second=start.second,\n                microsecond=start.microsecond,\n            )\n        )\n        start_naive_tz = start.naive()\n\n        cron = croniter(self.cron, start_naive_tz, day_or=self.day_or)  # type: ignore\n        dates = set()\n        counter = 0\n\n        while True:\n            # croniter does not handle DST properly when the start time is\n            # in and around when the actual shift occurs. To work around this,\n            # we use the naive start time to get the next cron date delta, then\n            # add that time to the original scheduling anchor.\n            next_time = cron.get_next(datetime.datetime)\n            delta = next_time - start_naive_tz\n            next_date = pendulum.instance(start_localized + delta)\n\n            # if the end date was exceeded, exit\n            if end and next_date > end:\n                break\n            # ensure no duplicates; weird things can happen with DST\n            if next_date not in dates:\n                dates.add(next_date)\n                yield next_date\n\n            # if enough dates have been collected or enough attempts were made, exit\n            if len(dates) >= n or counter > MAX_ITERATIONS:\n                break\n\n            counter += 1\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.CronSchedule.get_dates","title":"get_dates async","text":"

    Retrieves dates from the schedule. Up to 1,000 candidate dates are checked following the start date.

    Parameters:

    Name Type Description Default n int

    The number of dates to generate

    None start datetime

    The first returned date will be on or after this date. Defaults to None. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None end datetime

    The maximum scheduled date to return. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None

    Returns:

    Type Description List[DateTime]

    List[pendulum.DateTime]: A list of dates

    Source code in prefect/server/schemas/schedules.py
    async def get_dates(\n    self,\n    n: int = None,\n    start: datetime.datetime = None,\n    end: datetime.datetime = None,\n) -> List[pendulum.DateTime]:\n    \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n    following the start date.\n\n    Args:\n        n (int): The number of dates to generate\n        start (datetime.datetime, optional): The first returned date will be on or\n            after this date. Defaults to None.  If a timezone-naive datetime is\n            provided, it is assumed to be in the schedule's timezone.\n        end (datetime.datetime, optional): The maximum scheduled date to return. If\n            a timezone-naive datetime is provided, it is assumed to be in the\n            schedule's timezone.\n\n    Returns:\n        List[pendulum.DateTime]: A list of dates\n    \"\"\"\n    return sorted(self._get_dates_generator(n=n, start=start, end=end))\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.IntervalSchedule","title":"IntervalSchedule","text":"

    Bases: PrefectBaseModel

    A schedule formed by adding interval increments to an anchor_date. If no anchor_date is supplied, the current UTC time is used. If a timezone-naive datetime is provided for anchor_date, it is assumed to be in the schedule's timezone (or UTC). Even if supplied with an IANA timezone, anchor dates are always stored as UTC offsets, so a timezone can be provided to determine localization behaviors like DST boundary handling. If none is provided it will be inferred from the anchor date.

    NOTE: If the IntervalSchedule anchor_date or timezone is provided in a DST-observing timezone, then the schedule will adjust itself appropriately. Intervals greater than 24 hours will follow DST conventions, while intervals of less than 24 hours will follow UTC intervals. For example, an hourly schedule will fire every UTC hour, even across DST boundaries. When clocks are set back, this will result in two runs that appear to both be scheduled for 1am local time, even though they are an hour apart in UTC time. For longer intervals, like a daily schedule, the interval schedule will adjust for DST boundaries so that the clock-hour remains constant. This means that a daily schedule that always fires at 9am will observe DST and continue to fire at 9am in the local time zone.

    Parameters:

    Name Type Description Default interval timedelta

    an interval to schedule on

    required anchor_date DateTimeTZ

    an anchor date to schedule increments against; if not provided, the current timestamp will be used

    required timezone str

    a valid timezone string

    required Source code in prefect/server/schemas/schedules.py
    class IntervalSchedule(PrefectBaseModel):\n    \"\"\"\n    A schedule formed by adding `interval` increments to an `anchor_date`. If no\n    `anchor_date` is supplied, the current UTC time is used.  If a\n    timezone-naive datetime is provided for `anchor_date`, it is assumed to be\n    in the schedule's timezone (or UTC). Even if supplied with an IANA timezone,\n    anchor dates are always stored as UTC offsets, so a `timezone` can be\n    provided to determine localization behaviors like DST boundary handling. If\n    none is provided it will be inferred from the anchor date.\n\n    NOTE: If the `IntervalSchedule` `anchor_date` or `timezone` is provided in a\n    DST-observing timezone, then the schedule will adjust itself appropriately.\n    Intervals greater than 24 hours will follow DST conventions, while intervals\n    of less than 24 hours will follow UTC intervals. For example, an hourly\n    schedule will fire every UTC hour, even across DST boundaries. When clocks\n    are set back, this will result in two runs that *appear* to both be\n    scheduled for 1am local time, even though they are an hour apart in UTC\n    time. For longer intervals, like a daily schedule, the interval schedule\n    will adjust for DST boundaries so that the clock-hour remains constant. This\n    means that a daily schedule that always fires at 9am will observe DST and\n    continue to fire at 9am in the local time zone.\n\n    Args:\n        interval (datetime.timedelta): an interval to schedule on\n        anchor_date (DateTimeTZ, optional): an anchor date to schedule increments against;\n            if not provided, the current timestamp will be used\n        timezone (str, optional): a valid timezone string\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n        exclude_none = True\n\n    interval: datetime.timedelta\n    anchor_date: DateTimeTZ = None\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n\n    @validator(\"interval\")\n    def interval_must_be_positive(cls, v):\n        if v.total_seconds() <= 0:\n            raise ValueError(\"The interval must be positive\")\n        return v\n\n    @validator(\"anchor_date\", always=True)\n    def default_anchor_date(cls, v):\n        if v is None:\n            return pendulum.now(\"UTC\")\n        return pendulum.instance(v)\n\n    @validator(\"timezone\", always=True)\n    def default_timezone(cls, v, *, values, **kwargs):\n        # if was provided, make sure its a valid IANA string\n        if v and v not in pendulum.tz.timezones:\n            raise ValueError(f'Invalid timezone: \"{v}\"')\n\n        # otherwise infer the timezone from the anchor date\n        elif v is None and values.get(\"anchor_date\"):\n            tz = values[\"anchor_date\"].tz.name\n            if tz in pendulum.tz.timezones:\n                return tz\n            # sometimes anchor dates have \"timezones\" that are UTC offsets\n            # like \"-04:00\". This happens when parsing ISO8601 strings.\n            # In this case we, the correct inferred localization is \"UTC\".\n            else:\n                return \"UTC\"\n\n        return v\n\n    async def get_dates(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> List[pendulum.DateTime]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to None.  If a timezone-naive datetime is\n                provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): The maximum scheduled date to return. If\n                a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: A list of dates\n        \"\"\"\n        return sorted(self._get_dates_generator(n=n, start=start, end=end))\n\n    def _get_dates_generator(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> Generator[pendulum.DateTime, None, None]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to None.  If a timezone-naive datetime is\n                provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): The maximum scheduled date to return. If\n                a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: a list of dates\n        \"\"\"\n        if n is None:\n            # if an end was supplied, we do our best to supply all matching dates (up to\n            # MAX_ITERATIONS)\n            if end is not None:\n                n = MAX_ITERATIONS\n            else:\n                n = 1\n\n        if start is None:\n            start = pendulum.now(\"UTC\")\n\n        anchor_tz = self.anchor_date.in_tz(self.timezone)\n        start, end = _prepare_scheduling_start_and_end(start, end, self.timezone)\n\n        # compute the offset between the anchor date and the start date to jump to the\n        # next date\n        offset = (start - anchor_tz).total_seconds() / self.interval.total_seconds()\n        next_date = anchor_tz.add(seconds=self.interval.total_seconds() * int(offset))\n\n        # break the interval into `days` and `seconds` because pendulum\n        # will handle DST boundaries properly if days are provided, but not\n        # if we add `total seconds`. Therefore, `next_date + self.interval`\n        # fails while `next_date.add(days=days, seconds=seconds)` works.\n        interval_days = self.interval.days\n        interval_seconds = self.interval.total_seconds() - (\n            interval_days * 24 * 60 * 60\n        )\n\n        # daylight saving time boundaries can create a situation where the next date is\n        # before the start date, so we advance it if necessary\n        while next_date < start:\n            next_date = next_date.add(days=interval_days, seconds=interval_seconds)\n\n        counter = 0\n        dates = set()\n\n        while True:\n            # if the end date was exceeded, exit\n            if end and next_date > end:\n                break\n\n            # ensure no duplicates; weird things can happen with DST\n            if next_date not in dates:\n                dates.add(next_date)\n                yield next_date\n\n            # if enough dates have been collected or enough attempts were made, exit\n            if len(dates) >= n or counter > MAX_ITERATIONS:\n                break\n\n            counter += 1\n\n            next_date = next_date.add(days=interval_days, seconds=interval_seconds)\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.IntervalSchedule.get_dates","title":"get_dates async","text":"

    Retrieves dates from the schedule. Up to 1,000 candidate dates are checked following the start date.

    Parameters:

    Name Type Description Default n int

    The number of dates to generate

    None start datetime

    The first returned date will be on or after this date. Defaults to None. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None end datetime

    The maximum scheduled date to return. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None

    Returns:

    Type Description List[DateTime]

    List[pendulum.DateTime]: A list of dates

    Source code in prefect/server/schemas/schedules.py
    async def get_dates(\n    self,\n    n: int = None,\n    start: datetime.datetime = None,\n    end: datetime.datetime = None,\n) -> List[pendulum.DateTime]:\n    \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n    following the start date.\n\n    Args:\n        n (int): The number of dates to generate\n        start (datetime.datetime, optional): The first returned date will be on or\n            after this date. Defaults to None.  If a timezone-naive datetime is\n            provided, it is assumed to be in the schedule's timezone.\n        end (datetime.datetime, optional): The maximum scheduled date to return. If\n            a timezone-naive datetime is provided, it is assumed to be in the\n            schedule's timezone.\n\n    Returns:\n        List[pendulum.DateTime]: A list of dates\n    \"\"\"\n    return sorted(self._get_dates_generator(n=n, start=start, end=end))\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.RRuleSchedule","title":"RRuleSchedule","text":"

    Bases: PrefectBaseModel

    RRule schedule, based on the iCalendar standard (RFC 5545) as implemented in dateutils.rrule.

    RRules are appropriate for any kind of calendar-date manipulation, including irregular intervals, repetition, exclusions, week day or day-of-month adjustments, and more.

    Note that as a calendar-oriented standard, RRuleSchedules are sensitive to to the initial timezone provided. A 9am daily schedule with a daylight saving time-aware start date will maintain a local 9am time through DST boundaries; a 9am daily schedule with a UTC start date will maintain a 9am UTC time.

    Parameters:

    Name Type Description Default rrule str

    a valid RRule string

    required timezone str

    a valid timezone string

    required Source code in prefect/server/schemas/schedules.py
    class RRuleSchedule(PrefectBaseModel):\n    \"\"\"\n    RRule schedule, based on the iCalendar standard\n    ([RFC 5545](https://datatracker.ietf.org/doc/html/rfc5545)) as\n    implemented in `dateutils.rrule`.\n\n    RRules are appropriate for any kind of calendar-date manipulation, including\n    irregular intervals, repetition, exclusions, week day or day-of-month\n    adjustments, and more.\n\n    Note that as a calendar-oriented standard, `RRuleSchedules` are sensitive to\n    to the initial timezone provided. A 9am daily schedule with a daylight saving\n    time-aware start date will maintain a local 9am time through DST boundaries;\n    a 9am daily schedule with a UTC start date will maintain a 9am UTC time.\n\n    Args:\n        rrule (str): a valid RRule string\n        timezone (str, optional): a valid timezone string\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    rrule: str\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n\n    @validator(\"rrule\")\n    def validate_rrule_str(cls, v):\n        # attempt to parse the rrule string as an rrule object\n        # this will error if the string is invalid\n        try:\n            dateutil.rrule.rrulestr(v, cache=True)\n        except ValueError as exc:\n            # rrules errors are a mix of cryptic and informative\n            # so reraise to be clear that the string was invalid\n            raise ValueError(f'Invalid RRule string \"{v}\": {exc}')\n        if len(v) > MAX_RRULE_LENGTH:\n            raise ValueError(\n                f'Invalid RRule string \"{v[:40]}...\"\\n'\n                f\"Max length is {MAX_RRULE_LENGTH}, got {len(v)}\"\n            )\n        return v\n\n    @classmethod\n    def from_rrule(cls, rrule: dateutil.rrule.rrule):\n        if isinstance(rrule, dateutil.rrule.rrule):\n            if rrule._dtstart.tzinfo is not None:\n                timezone = rrule._dtstart.tzinfo.name\n            else:\n                timezone = \"UTC\"\n            return RRuleSchedule(rrule=str(rrule), timezone=timezone)\n        elif isinstance(rrule, dateutil.rrule.rruleset):\n            dtstarts = [rr._dtstart for rr in rrule._rrule if rr._dtstart is not None]\n            unique_dstarts = set(pendulum.instance(d).in_tz(\"UTC\") for d in dtstarts)\n            unique_timezones = set(d.tzinfo for d in dtstarts if d.tzinfo is not None)\n\n            if len(unique_timezones) > 1:\n                raise ValueError(\n                    f\"rruleset has too many dtstart timezones: {unique_timezones}\"\n                )\n\n            if len(unique_dstarts) > 1:\n                raise ValueError(f\"rruleset has too many dtstarts: {unique_dstarts}\")\n\n            if unique_dstarts and unique_timezones:\n                timezone = dtstarts[0].tzinfo.name\n            else:\n                timezone = \"UTC\"\n\n            rruleset_string = \"\"\n            if rrule._rrule:\n                rruleset_string += \"\\n\".join(str(r) for r in rrule._rrule)\n            if rrule._exrule:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"\\n\".join(str(r) for r in rrule._exrule).replace(\n                    \"RRULE\", \"EXRULE\"\n                )\n            if rrule._rdate:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"RDATE:\" + \",\".join(\n                    rd.strftime(\"%Y%m%dT%H%M%SZ\") for rd in rrule._rdate\n                )\n            if rrule._exdate:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"EXDATE:\" + \",\".join(\n                    exd.strftime(\"%Y%m%dT%H%M%SZ\") for exd in rrule._exdate\n                )\n            return RRuleSchedule(rrule=rruleset_string, timezone=timezone)\n        else:\n            raise ValueError(f\"Invalid RRule object: {rrule}\")\n\n    def to_rrule(self) -> dateutil.rrule.rrule:\n        \"\"\"\n        Since rrule doesn't properly serialize/deserialize timezones, we localize dates\n        here\n        \"\"\"\n        rrule = dateutil.rrule.rrulestr(\n            self.rrule,\n            dtstart=DEFAULT_ANCHOR_DATE,\n            cache=True,\n        )\n        timezone = dateutil.tz.gettz(self.timezone)\n        if isinstance(rrule, dateutil.rrule.rrule):\n            kwargs = dict(dtstart=rrule._dtstart.replace(tzinfo=timezone))\n            if rrule._until:\n                kwargs.update(\n                    until=rrule._until.replace(tzinfo=timezone),\n                )\n            return rrule.replace(**kwargs)\n        elif isinstance(rrule, dateutil.rrule.rruleset):\n            # update rrules\n            localized_rrules = []\n            for rr in rrule._rrule:\n                kwargs = dict(dtstart=rr._dtstart.replace(tzinfo=timezone))\n                if rr._until:\n                    kwargs.update(\n                        until=rr._until.replace(tzinfo=timezone),\n                    )\n                localized_rrules.append(rr.replace(**kwargs))\n            rrule._rrule = localized_rrules\n\n            # update exrules\n            localized_exrules = []\n            for exr in rrule._exrule:\n                kwargs = dict(dtstart=exr._dtstart.replace(tzinfo=timezone))\n                if exr._until:\n                    kwargs.update(\n                        until=exr._until.replace(tzinfo=timezone),\n                    )\n                localized_exrules.append(exr.replace(**kwargs))\n            rrule._exrule = localized_exrules\n\n            # update rdates\n            localized_rdates = []\n            for rd in rrule._rdate:\n                localized_rdates.append(rd.replace(tzinfo=timezone))\n            rrule._rdate = localized_rdates\n\n            # update exdates\n            localized_exdates = []\n            for exd in rrule._exdate:\n                localized_exdates.append(exd.replace(tzinfo=timezone))\n            rrule._exdate = localized_exdates\n\n            return rrule\n\n    @validator(\"timezone\", always=True)\n    def valid_timezone(cls, v):\n        if v and v not in pytz.all_timezones_set:\n            raise ValueError(f'Invalid timezone: \"{v}\"')\n        elif v is None:\n            return \"UTC\"\n        return v\n\n    async def get_dates(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> List[pendulum.DateTime]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to None.  If a timezone-naive datetime is\n                provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): The maximum scheduled date to return. If\n                a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: A list of dates\n        \"\"\"\n        return sorted(self._get_dates_generator(n=n, start=start, end=end))\n\n    def _get_dates_generator(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> Generator[pendulum.DateTime, None, None]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to the current date. If a timezone-naive\n                datetime is provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): No returned date will exceed this date.\n                If a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: a list of dates\n        \"\"\"\n        if start is None:\n            start = pendulum.now(\"UTC\")\n\n        start, end = _prepare_scheduling_start_and_end(start, end, self.timezone)\n\n        if n is None:\n            # if an end was supplied, we do our best to supply all matching dates (up\n            # to MAX_ITERATIONS)\n            if end is not None:\n                n = MAX_ITERATIONS\n            else:\n                n = 1\n\n        dates = set()\n        counter = 0\n\n        # pass count = None to account for discrepancies with duplicates around DST\n        # boundaries\n        for next_date in self.to_rrule().xafter(start, count=None, inc=True):\n            next_date = pendulum.instance(next_date).in_tz(self.timezone)\n\n            # if the end date was exceeded, exit\n            if end and next_date > end:\n                break\n\n            # ensure no duplicates; weird things can happen with DST\n            if next_date not in dates:\n                dates.add(next_date)\n                yield next_date\n\n            # if enough dates have been collected or enough attempts were made, exit\n            if len(dates) >= n or counter > MAX_ITERATIONS:\n                break\n\n            counter += 1\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.RRuleSchedule.get_dates","title":"get_dates async","text":"

    Retrieves dates from the schedule. Up to 1,000 candidate dates are checked following the start date.

    Parameters:

    Name Type Description Default n int

    The number of dates to generate

    None start datetime

    The first returned date will be on or after this date. Defaults to None. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None end datetime

    The maximum scheduled date to return. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None

    Returns:

    Type Description List[DateTime]

    List[pendulum.DateTime]: A list of dates

    Source code in prefect/server/schemas/schedules.py
    async def get_dates(\n    self,\n    n: int = None,\n    start: datetime.datetime = None,\n    end: datetime.datetime = None,\n) -> List[pendulum.DateTime]:\n    \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n    following the start date.\n\n    Args:\n        n (int): The number of dates to generate\n        start (datetime.datetime, optional): The first returned date will be on or\n            after this date. Defaults to None.  If a timezone-naive datetime is\n            provided, it is assumed to be in the schedule's timezone.\n        end (datetime.datetime, optional): The maximum scheduled date to return. If\n            a timezone-naive datetime is provided, it is assumed to be in the\n            schedule's timezone.\n\n    Returns:\n        List[pendulum.DateTime]: A list of dates\n    \"\"\"\n    return sorted(self._get_dates_generator(n=n, start=start, end=end))\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.RRuleSchedule.to_rrule","title":"to_rrule","text":"

    Since rrule doesn't properly serialize/deserialize timezones, we localize dates here

    Source code in prefect/server/schemas/schedules.py
    def to_rrule(self) -> dateutil.rrule.rrule:\n    \"\"\"\n    Since rrule doesn't properly serialize/deserialize timezones, we localize dates\n    here\n    \"\"\"\n    rrule = dateutil.rrule.rrulestr(\n        self.rrule,\n        dtstart=DEFAULT_ANCHOR_DATE,\n        cache=True,\n    )\n    timezone = dateutil.tz.gettz(self.timezone)\n    if isinstance(rrule, dateutil.rrule.rrule):\n        kwargs = dict(dtstart=rrule._dtstart.replace(tzinfo=timezone))\n        if rrule._until:\n            kwargs.update(\n                until=rrule._until.replace(tzinfo=timezone),\n            )\n        return rrule.replace(**kwargs)\n    elif isinstance(rrule, dateutil.rrule.rruleset):\n        # update rrules\n        localized_rrules = []\n        for rr in rrule._rrule:\n            kwargs = dict(dtstart=rr._dtstart.replace(tzinfo=timezone))\n            if rr._until:\n                kwargs.update(\n                    until=rr._until.replace(tzinfo=timezone),\n                )\n            localized_rrules.append(rr.replace(**kwargs))\n        rrule._rrule = localized_rrules\n\n        # update exrules\n        localized_exrules = []\n        for exr in rrule._exrule:\n            kwargs = dict(dtstart=exr._dtstart.replace(tzinfo=timezone))\n            if exr._until:\n                kwargs.update(\n                    until=exr._until.replace(tzinfo=timezone),\n                )\n            localized_exrules.append(exr.replace(**kwargs))\n        rrule._exrule = localized_exrules\n\n        # update rdates\n        localized_rdates = []\n        for rd in rrule._rdate:\n            localized_rdates.append(rd.replace(tzinfo=timezone))\n        rrule._rdate = localized_rdates\n\n        # update exdates\n        localized_exdates = []\n        for exd in rrule._exdate:\n            localized_exdates.append(exd.replace(tzinfo=timezone))\n        rrule._exdate = localized_exdates\n\n        return rrule\n
    "},{"location":"api-ref/server/schemas/sorting/","title":"server.schemas.sorting","text":""},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting","title":"prefect.server.schemas.sorting","text":"

    Schemas for sorting Prefect REST API objects.

    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.ArtifactCollectionSort","title":"ArtifactCollectionSort","text":"

    Bases: AutoEnum

    Defines artifact collection sorting options.

    Source code in prefect/server/schemas/sorting.py
    class ArtifactCollectionSort(AutoEnum):\n    \"\"\"Defines artifact collection sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    ID_DESC = AutoEnum.auto()\n    KEY_DESC = AutoEnum.auto()\n    KEY_ASC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort artifact collections\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.ArtifactCollection.created.desc(),\n            \"UPDATED_DESC\": db.ArtifactCollection.updated.desc(),\n            \"ID_DESC\": db.ArtifactCollection.id.desc(),\n            \"KEY_DESC\": db.ArtifactCollection.key.desc(),\n            \"KEY_ASC\": db.ArtifactCollection.key.asc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.ArtifactCollectionSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort artifact collections

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort artifact collections\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.ArtifactCollection.created.desc(),\n        \"UPDATED_DESC\": db.ArtifactCollection.updated.desc(),\n        \"ID_DESC\": db.ArtifactCollection.id.desc(),\n        \"KEY_DESC\": db.ArtifactCollection.key.desc(),\n        \"KEY_ASC\": db.ArtifactCollection.key.asc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.ArtifactSort","title":"ArtifactSort","text":"

    Bases: AutoEnum

    Defines artifact sorting options.

    Source code in prefect/server/schemas/sorting.py
    class ArtifactSort(AutoEnum):\n    \"\"\"Defines artifact sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    ID_DESC = AutoEnum.auto()\n    KEY_DESC = AutoEnum.auto()\n    KEY_ASC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort artifacts\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.Artifact.created.desc(),\n            \"UPDATED_DESC\": db.Artifact.updated.desc(),\n            \"ID_DESC\": db.Artifact.id.desc(),\n            \"KEY_DESC\": db.Artifact.key.desc(),\n            \"KEY_ASC\": db.Artifact.key.asc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.ArtifactSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort artifacts

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort artifacts\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.Artifact.created.desc(),\n        \"UPDATED_DESC\": db.Artifact.updated.desc(),\n        \"ID_DESC\": db.Artifact.id.desc(),\n        \"KEY_DESC\": db.Artifact.key.desc(),\n        \"KEY_ASC\": db.Artifact.key.asc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.BlockDocumentSort","title":"BlockDocumentSort","text":"

    Bases: AutoEnum

    Defines block document sorting options.

    Source code in prefect/server/schemas/sorting.py
    class BlockDocumentSort(AutoEnum):\n    \"\"\"Defines block document sorting options.\"\"\"\n\n    NAME_DESC = \"NAME_DESC\"\n    NAME_ASC = \"NAME_ASC\"\n    BLOCK_TYPE_AND_NAME_ASC = \"BLOCK_TYPE_AND_NAME_ASC\"\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort block documents\"\"\"\n        sort_mapping = {\n            \"NAME_DESC\": db.BlockDocument.name.desc(),\n            \"NAME_ASC\": db.BlockDocument.name.asc(),\n            \"BLOCK_TYPE_AND_NAME_ASC\": sa.text(\"block_type_name asc, name asc\"),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.BlockDocumentSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort block documents

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort block documents\"\"\"\n    sort_mapping = {\n        \"NAME_DESC\": db.BlockDocument.name.desc(),\n        \"NAME_ASC\": db.BlockDocument.name.asc(),\n        \"BLOCK_TYPE_AND_NAME_ASC\": sa.text(\"block_type_name asc, name asc\"),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.DeploymentSort","title":"DeploymentSort","text":"

    Bases: AutoEnum

    Defines deployment sorting options.

    Source code in prefect/server/schemas/sorting.py
    class DeploymentSort(AutoEnum):\n    \"\"\"Defines deployment sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort deployments\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.Deployment.created.desc(),\n            \"UPDATED_DESC\": db.Deployment.updated.desc(),\n            \"NAME_ASC\": db.Deployment.name.asc(),\n            \"NAME_DESC\": db.Deployment.name.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.DeploymentSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort deployments

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort deployments\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.Deployment.created.desc(),\n        \"UPDATED_DESC\": db.Deployment.updated.desc(),\n        \"NAME_ASC\": db.Deployment.name.asc(),\n        \"NAME_DESC\": db.Deployment.name.desc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.FlowRunSort","title":"FlowRunSort","text":"

    Bases: AutoEnum

    Defines flow run sorting options.

    Source code in prefect/server/schemas/sorting.py
    class FlowRunSort(AutoEnum):\n    \"\"\"Defines flow run sorting options.\"\"\"\n\n    ID_DESC = AutoEnum.auto()\n    START_TIME_ASC = AutoEnum.auto()\n    START_TIME_DESC = AutoEnum.auto()\n    EXPECTED_START_TIME_ASC = AutoEnum.auto()\n    EXPECTED_START_TIME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n    NEXT_SCHEDULED_START_TIME_ASC = AutoEnum.auto()\n    END_TIME_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        from sqlalchemy.sql.functions import coalesce\n\n        \"\"\"Return an expression used to sort flow runs\"\"\"\n        sort_mapping = {\n            \"ID_DESC\": db.FlowRun.id.desc(),\n            \"START_TIME_ASC\": coalesce(\n                db.FlowRun.start_time, db.FlowRun.expected_start_time\n            ).asc(),\n            \"START_TIME_DESC\": coalesce(\n                db.FlowRun.start_time, db.FlowRun.expected_start_time\n            ).desc(),\n            \"EXPECTED_START_TIME_ASC\": db.FlowRun.expected_start_time.asc(),\n            \"EXPECTED_START_TIME_DESC\": db.FlowRun.expected_start_time.desc(),\n            \"NAME_ASC\": db.FlowRun.name.asc(),\n            \"NAME_DESC\": db.FlowRun.name.desc(),\n            \"NEXT_SCHEDULED_START_TIME_ASC\": db.FlowRun.next_scheduled_start_time.asc(),\n            \"END_TIME_DESC\": db.FlowRun.end_time.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.FlowSort","title":"FlowSort","text":"

    Bases: AutoEnum

    Defines flow sorting options.

    Source code in prefect/server/schemas/sorting.py
    class FlowSort(AutoEnum):\n    \"\"\"Defines flow sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort flows\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.Flow.created.desc(),\n            \"UPDATED_DESC\": db.Flow.updated.desc(),\n            \"NAME_ASC\": db.Flow.name.asc(),\n            \"NAME_DESC\": db.Flow.name.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.FlowSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort flows

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort flows\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.Flow.created.desc(),\n        \"UPDATED_DESC\": db.Flow.updated.desc(),\n        \"NAME_ASC\": db.Flow.name.asc(),\n        \"NAME_DESC\": db.Flow.name.desc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.LogSort","title":"LogSort","text":"

    Bases: AutoEnum

    Defines log sorting options.

    Source code in prefect/server/schemas/sorting.py
    class LogSort(AutoEnum):\n    \"\"\"Defines log sorting options.\"\"\"\n\n    TIMESTAMP_ASC = AutoEnum.auto()\n    TIMESTAMP_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort task runs\"\"\"\n        sort_mapping = {\n            \"TIMESTAMP_ASC\": db.Log.timestamp.asc(),\n            \"TIMESTAMP_DESC\": db.Log.timestamp.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.LogSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort task runs

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort task runs\"\"\"\n    sort_mapping = {\n        \"TIMESTAMP_ASC\": db.Log.timestamp.asc(),\n        \"TIMESTAMP_DESC\": db.Log.timestamp.desc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.TaskRunSort","title":"TaskRunSort","text":"

    Bases: AutoEnum

    Defines task run sorting options.

    Source code in prefect/server/schemas/sorting.py
    class TaskRunSort(AutoEnum):\n    \"\"\"Defines task run sorting options.\"\"\"\n\n    ID_DESC = AutoEnum.auto()\n    EXPECTED_START_TIME_ASC = AutoEnum.auto()\n    EXPECTED_START_TIME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n    NEXT_SCHEDULED_START_TIME_ASC = AutoEnum.auto()\n    END_TIME_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort task runs\"\"\"\n        sort_mapping = {\n            \"ID_DESC\": db.TaskRun.id.desc(),\n            \"EXPECTED_START_TIME_ASC\": db.TaskRun.expected_start_time.asc(),\n            \"EXPECTED_START_TIME_DESC\": db.TaskRun.expected_start_time.desc(),\n            \"NAME_ASC\": db.TaskRun.name.asc(),\n            \"NAME_DESC\": db.TaskRun.name.desc(),\n            \"NEXT_SCHEDULED_START_TIME_ASC\": db.TaskRun.next_scheduled_start_time.asc(),\n            \"END_TIME_DESC\": db.TaskRun.end_time.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.TaskRunSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort task runs

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort task runs\"\"\"\n    sort_mapping = {\n        \"ID_DESC\": db.TaskRun.id.desc(),\n        \"EXPECTED_START_TIME_ASC\": db.TaskRun.expected_start_time.asc(),\n        \"EXPECTED_START_TIME_DESC\": db.TaskRun.expected_start_time.desc(),\n        \"NAME_ASC\": db.TaskRun.name.asc(),\n        \"NAME_DESC\": db.TaskRun.name.desc(),\n        \"NEXT_SCHEDULED_START_TIME_ASC\": db.TaskRun.next_scheduled_start_time.asc(),\n        \"END_TIME_DESC\": db.TaskRun.end_time.desc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.VariableSort","title":"VariableSort","text":"

    Bases: AutoEnum

    Defines variables sorting options.

    Source code in prefect/server/schemas/sorting.py
    class VariableSort(AutoEnum):\n    \"\"\"Defines variables sorting options.\"\"\"\n\n    CREATED_DESC = \"CREATED_DESC\"\n    UPDATED_DESC = \"UPDATED_DESC\"\n    NAME_DESC = \"NAME_DESC\"\n    NAME_ASC = \"NAME_ASC\"\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort variables\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.Variable.created.desc(),\n            \"UPDATED_DESC\": db.Variable.updated.desc(),\n            \"NAME_DESC\": db.Variable.name.desc(),\n            \"NAME_ASC\": db.Variable.name.asc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.VariableSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort variables

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort variables\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.Variable.created.desc(),\n        \"UPDATED_DESC\": db.Variable.updated.desc(),\n        \"NAME_DESC\": db.Variable.name.desc(),\n        \"NAME_ASC\": db.Variable.name.asc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/states/","title":"server.schemas.states","text":""},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states","title":"prefect.server.schemas.states","text":"

    State schemas.

    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.State","title":"State","text":"

    Bases: StateBaseModel, Generic[R]

    Represents the state of a run.

    Source code in prefect/server/schemas/states.py
    class State(StateBaseModel, Generic[R]):\n    \"\"\"Represents the state of a run.\"\"\"\n\n    class Config:\n        orm_mode = True\n\n    type: StateType\n    name: Optional[str] = Field(default=None)\n    timestamp: DateTimeTZ = Field(default_factory=lambda: pendulum.now(\"UTC\"))\n    message: Optional[str] = Field(default=None, example=\"Run started\")\n    data: Optional[Any] = Field(\n        default=None,\n        description=(\n            \"Data associated with the state, e.g. a result. \"\n            \"Content must be storable as JSON.\"\n        ),\n    )\n    state_details: StateDetails = Field(default_factory=StateDetails)\n\n    @classmethod\n    def from_orm_without_result(\n        cls,\n        orm_state: Union[\n            \"prefect.server.database.orm_models.ORMFlowRunState\",\n            \"prefect.server.database.orm_models.ORMTaskRunState\",\n        ],\n        with_data: Optional[Any] = None,\n    ):\n        \"\"\"\n        During orchestration, ORM states can be instantiated prior to inserting results\n        into the artifact table and the `data` field will not be eagerly loaded. In\n        these cases, sqlalchemy will attempt to lazily load the the relationship, which\n        will fail when called within a synchronous pydantic method.\n\n        This method will construct a `State` object from an ORM model without a loaded\n        artifact and attach data passed using the `with_data` argument to the `data`\n        field.\n        \"\"\"\n\n        field_keys = cls.schema()[\"properties\"].keys()\n        state_data = {\n            field: getattr(orm_state, field, None)\n            for field in field_keys\n            if field != \"data\"\n        }\n        state_data[\"data\"] = with_data\n        return cls(**state_data)\n\n    @validator(\"name\", always=True)\n    def default_name_from_type(cls, v, *, values, **kwargs):\n        \"\"\"If a name is not provided, use the type\"\"\"\n\n        # if `type` is not in `values` it means the `type` didn't pass its own\n        # validation check and an error will be raised after this function is called\n        if v is None and values.get(\"type\"):\n            v = \" \".join([v.capitalize() for v in values.get(\"type\").value.split(\"_\")])\n        return v\n\n    @root_validator\n    def default_scheduled_start_time(cls, values):\n        \"\"\"\n        TODO: This should throw an error instead of setting a default but is out of\n              scope for https://github.com/PrefectHQ/orion/pull/174/ and can be rolled\n              into work refactoring state initialization\n        \"\"\"\n        if values.get(\"type\") == StateType.SCHEDULED:\n            state_details = values.setdefault(\n                \"state_details\", cls.__fields__[\"state_details\"].get_default()\n            )\n            if not state_details.scheduled_time:\n                state_details.scheduled_time = pendulum.now(\"utc\")\n        return values\n\n    def is_scheduled(self) -> bool:\n        return self.type == StateType.SCHEDULED\n\n    def is_pending(self) -> bool:\n        return self.type == StateType.PENDING\n\n    def is_running(self) -> bool:\n        return self.type == StateType.RUNNING\n\n    def is_completed(self) -> bool:\n        return self.type == StateType.COMPLETED\n\n    def is_failed(self) -> bool:\n        return self.type == StateType.FAILED\n\n    def is_crashed(self) -> bool:\n        return self.type == StateType.CRASHED\n\n    def is_cancelled(self) -> bool:\n        return self.type == StateType.CANCELLED\n\n    def is_cancelling(self) -> bool:\n        return self.type == StateType.CANCELLING\n\n    def is_final(self) -> bool:\n        return self.type in TERMINAL_STATES\n\n    def is_paused(self) -> bool:\n        return self.type == StateType.PAUSED\n\n    def copy(self, *, update: dict = None, reset_fields: bool = False, **kwargs):\n        \"\"\"\n        Copying API models should return an object that could be inserted into the\n        database again. The 'timestamp' is reset using the default factory.\n        \"\"\"\n        update = update or {}\n        update.setdefault(\"timestamp\", self.__fields__[\"timestamp\"].get_default())\n        return super().copy(reset_fields=reset_fields, update=update, **kwargs)\n\n    def result(self, raise_on_failure: bool = True, fetch: Optional[bool] = None):\n        # Backwards compatible `result` handling on the server-side schema\n        from prefect.states import State\n\n        warnings.warn(\n            (\n                \"`result` is no longer supported by\"\n                \" `prefect.server.schemas.states.State` and will be removed in a future\"\n                \" release. When result retrieval is needed, use `prefect.states.State`.\"\n            ),\n            DeprecationWarning,\n            stacklevel=2,\n        )\n\n        state = State.parse_obj(self)\n        return state.result(raise_on_failure=raise_on_failure, fetch=fetch)\n\n    def to_state_create(self):\n        # Backwards compatibility for `to_state_create`\n        from prefect.client.schemas import State\n\n        warnings.warn(\n            (\n                \"Use of `prefect.server.schemas.states.State` from the client is\"\n                \" deprecated and support will be removed in a future release. Use\"\n                \" `prefect.states.State` instead.\"\n            ),\n            DeprecationWarning,\n            stacklevel=2,\n        )\n\n        state = State.parse_obj(self)\n        return state.to_state_create()\n\n    def __repr__(self) -> str:\n        \"\"\"\n        Generates a complete state representation appropriate for introspection\n        and debugging, including the result:\n\n        `MyCompletedState(message=\"my message\", type=COMPLETED, result=...)`\n        \"\"\"\n        from prefect.deprecated.data_documents import DataDocument\n\n        if isinstance(self.data, DataDocument):\n            result = self.data.decode()\n        else:\n            result = self.data\n\n        display = dict(\n            message=repr(self.message),\n            type=str(self.type.value),\n            result=repr(result),\n        )\n\n        return f\"{self.name}({', '.join(f'{k}={v}' for k, v in display.items())})\"\n\n    def __str__(self) -> str:\n        \"\"\"\n        Generates a simple state representation appropriate for logging:\n\n        `MyCompletedState(\"my message\", type=COMPLETED)`\n        \"\"\"\n\n        display = []\n\n        if self.message:\n            display.append(repr(self.message))\n\n        if self.type.value.lower() != self.name.lower():\n            display.append(f\"type={self.type.value}\")\n\n        return f\"{self.name}({', '.join(display)})\"\n\n    def __hash__(self) -> int:\n        return hash(\n            (\n                getattr(self.state_details, \"flow_run_id\", None),\n                getattr(self.state_details, \"task_run_id\", None),\n                self.timestamp,\n                self.type,\n            )\n        )\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.State.from_orm_without_result","title":"from_orm_without_result classmethod","text":"

    During orchestration, ORM states can be instantiated prior to inserting results into the artifact table and the data field will not be eagerly loaded. In these cases, sqlalchemy will attempt to lazily load the the relationship, which will fail when called within a synchronous pydantic method.

    This method will construct a State object from an ORM model without a loaded artifact and attach data passed using the with_data argument to the data field.

    Source code in prefect/server/schemas/states.py
    @classmethod\ndef from_orm_without_result(\n    cls,\n    orm_state: Union[\n        \"prefect.server.database.orm_models.ORMFlowRunState\",\n        \"prefect.server.database.orm_models.ORMTaskRunState\",\n    ],\n    with_data: Optional[Any] = None,\n):\n    \"\"\"\n    During orchestration, ORM states can be instantiated prior to inserting results\n    into the artifact table and the `data` field will not be eagerly loaded. In\n    these cases, sqlalchemy will attempt to lazily load the the relationship, which\n    will fail when called within a synchronous pydantic method.\n\n    This method will construct a `State` object from an ORM model without a loaded\n    artifact and attach data passed using the `with_data` argument to the `data`\n    field.\n    \"\"\"\n\n    field_keys = cls.schema()[\"properties\"].keys()\n    state_data = {\n        field: getattr(orm_state, field, None)\n        for field in field_keys\n        if field != \"data\"\n    }\n    state_data[\"data\"] = with_data\n    return cls(**state_data)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.State.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.StateBaseModel","title":"StateBaseModel","text":"

    Bases: IDBaseModel

    Source code in prefect/server/schemas/states.py
    class StateBaseModel(IDBaseModel):\n    def orm_dict(\n        self, *args, shallow: bool = False, json_compatible: bool = False, **kwargs\n    ) -> dict:\n        \"\"\"\n        This method is used as a convenience method for constructing fixtues by first\n        building a `State` schema object and converting it into an ORM-compatible\n        format. Because the `data` field is not writable on ORM states, this method\n        omits the `data` field entirely for the purposes of constructing an ORM model.\n        If state data is required, an artifact must be created separately.\n        \"\"\"\n\n        schema_dict = self.dict(\n            *args, shallow=shallow, json_compatible=json_compatible, **kwargs\n        )\n        # remove the data field in order to construct a state ORM model\n        schema_dict.pop(\"data\", None)\n        return schema_dict\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.StateBaseModel.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.StateType","title":"StateType","text":"

    Bases: AutoEnum

    Enumeration of state types.

    Source code in prefect/server/schemas/states.py
    class StateType(AutoEnum):\n    \"\"\"Enumeration of state types.\"\"\"\n\n    SCHEDULED = AutoEnum.auto()\n    PENDING = AutoEnum.auto()\n    RUNNING = AutoEnum.auto()\n    COMPLETED = AutoEnum.auto()\n    FAILED = AutoEnum.auto()\n    CANCELLED = AutoEnum.auto()\n    CRASHED = AutoEnum.auto()\n    PAUSED = AutoEnum.auto()\n    CANCELLING = AutoEnum.auto()\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.StateType.auto","title":"auto staticmethod","text":"

    Exposes enum.auto() to avoid requiring a second import to use AutoEnum

    Source code in prefect/utilities/collections.py
    @staticmethod\ndef auto():\n    \"\"\"\n    Exposes `enum.auto()` to avoid requiring a second import to use `AutoEnum`\n    \"\"\"\n    return auto()\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.AwaitingRetry","title":"AwaitingRetry","text":"

    Convenience function for creating AwaitingRetry states.

    Returns:

    Name Type Description State State

    a AwaitingRetry state

    Source code in prefect/server/schemas/states.py
    def AwaitingRetry(\n    scheduled_time: datetime.datetime = None, cls: Type[State] = State, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `AwaitingRetry` states.\n\n    Returns:\n        State: a AwaitingRetry state\n    \"\"\"\n    return Scheduled(\n        cls=cls, scheduled_time=scheduled_time, name=\"AwaitingRetry\", **kwargs\n    )\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Cancelled","title":"Cancelled","text":"

    Convenience function for creating Cancelled states.

    Returns:

    Name Type Description State State

    a Cancelled state

    Source code in prefect/server/schemas/states.py
    def Cancelled(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Cancelled` states.\n\n    Returns:\n        State: a Cancelled state\n    \"\"\"\n    return cls(type=StateType.CANCELLED, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Cancelling","title":"Cancelling","text":"

    Convenience function for creating Cancelling states.

    Returns:

    Name Type Description State State

    a Cancelling state

    Source code in prefect/server/schemas/states.py
    def Cancelling(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Cancelling` states.\n\n    Returns:\n        State: a Cancelling state\n    \"\"\"\n    return cls(type=StateType.CANCELLING, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Completed","title":"Completed","text":"

    Convenience function for creating Completed states.

    Returns:

    Name Type Description State State

    a Completed state

    Source code in prefect/server/schemas/states.py
    def Completed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Completed` states.\n\n    Returns:\n        State: a Completed state\n    \"\"\"\n    return cls(type=StateType.COMPLETED, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Crashed","title":"Crashed","text":"

    Convenience function for creating Crashed states.

    Returns:

    Name Type Description State State

    a Crashed state

    Source code in prefect/server/schemas/states.py
    def Crashed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Crashed` states.\n\n    Returns:\n        State: a Crashed state\n    \"\"\"\n    return cls(type=StateType.CRASHED, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Failed","title":"Failed","text":"

    Convenience function for creating Failed states.

    Returns:

    Name Type Description State State

    a Failed state

    Source code in prefect/server/schemas/states.py
    def Failed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Failed` states.\n\n    Returns:\n        State: a Failed state\n    \"\"\"\n    return cls(type=StateType.FAILED, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Late","title":"Late","text":"

    Convenience function for creating Late states.

    Returns:

    Name Type Description State State

    a Late state

    Source code in prefect/server/schemas/states.py
    def Late(\n    scheduled_time: datetime.datetime = None, cls: Type[State] = State, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `Late` states.\n\n    Returns:\n        State: a Late state\n    \"\"\"\n    return Scheduled(cls=cls, scheduled_time=scheduled_time, name=\"Late\", **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Paused","title":"Paused","text":"

    Convenience function for creating Paused states.

    Returns:

    Name Type Description State State

    a Paused state

    Source code in prefect/server/schemas/states.py
    def Paused(\n    cls: Type[State] = State,\n    timeout_seconds: int = None,\n    pause_expiration_time: datetime.datetime = None,\n    reschedule: bool = False,\n    pause_key: str = None,\n    **kwargs,\n) -> State:\n    \"\"\"Convenience function for creating `Paused` states.\n\n    Returns:\n        State: a Paused state\n    \"\"\"\n    state_details = StateDetails.parse_obj(kwargs.pop(\"state_details\", {}))\n\n    if state_details.pause_timeout:\n        raise ValueError(\"An extra pause timeout was provided in state_details\")\n\n    if pause_expiration_time is not None and timeout_seconds is not None:\n        raise ValueError(\n            \"Cannot supply both a pause_expiration_time and timeout_seconds\"\n        )\n\n    if pause_expiration_time is None and timeout_seconds is None:\n        pass\n    else:\n        state_details.pause_timeout = pause_expiration_time or (\n            pendulum.now(\"UTC\") + pendulum.Duration(seconds=timeout_seconds)\n        )\n\n    state_details.pause_reschedule = reschedule\n    state_details.pause_key = pause_key\n\n    return cls(type=StateType.PAUSED, state_details=state_details, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Pending","title":"Pending","text":"

    Convenience function for creating Pending states.

    Returns:

    Name Type Description State State

    a Pending state

    Source code in prefect/server/schemas/states.py
    def Pending(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Pending` states.\n\n    Returns:\n        State: a Pending state\n    \"\"\"\n    return cls(type=StateType.PENDING, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Retrying","title":"Retrying","text":"

    Convenience function for creating Retrying states.

    Returns:

    Name Type Description State State

    a Retrying state

    Source code in prefect/server/schemas/states.py
    def Retrying(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Retrying` states.\n\n    Returns:\n        State: a Retrying state\n    \"\"\"\n    return cls(type=StateType.RUNNING, name=\"Retrying\", **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Running","title":"Running","text":"

    Convenience function for creating Running states.

    Returns:

    Name Type Description State State

    a Running state

    Source code in prefect/server/schemas/states.py
    def Running(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Running` states.\n\n    Returns:\n        State: a Running state\n    \"\"\"\n    return cls(type=StateType.RUNNING, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Scheduled","title":"Scheduled","text":"

    Convenience function for creating Scheduled states.

    Returns:

    Name Type Description State State

    a Scheduled state

    Source code in prefect/server/schemas/states.py
    def Scheduled(\n    scheduled_time: datetime.datetime = None, cls: Type[State] = State, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `Scheduled` states.\n\n    Returns:\n        State: a Scheduled state\n    \"\"\"\n    # NOTE: `scheduled_time` must come first for backwards compatibility\n\n    state_details = StateDetails.parse_obj(kwargs.pop(\"state_details\", {}))\n    if scheduled_time is None:\n        scheduled_time = pendulum.now(\"UTC\")\n    elif state_details.scheduled_time:\n        raise ValueError(\"An extra scheduled_time was provided in state_details\")\n    state_details.scheduled_time = scheduled_time\n\n    return cls(type=StateType.SCHEDULED, state_details=state_details, **kwargs)\n
    "},{"location":"api-ref/server/services/late_runs/","title":"server.services.late_runs","text":""},{"location":"api-ref/server/services/late_runs/#prefect.server.services.late_runs","title":"prefect.server.services.late_runs","text":"

    The MarkLateRuns service. Responsible for putting flow runs in a Late state if they are not started on time. The threshold for a late run can be configured by changing PREFECT_API_SERVICES_LATE_RUNS_AFTER_SECONDS.

    "},{"location":"api-ref/server/services/late_runs/#prefect.server.services.late_runs.MarkLateRuns","title":"MarkLateRuns","text":"

    Bases: LoopService

    A simple loop service responsible for identifying flow runs that are \"late\".

    A flow run is defined as \"late\" if has not scheduled within a certain amount of time after its scheduled start time. The exact amount is configurable in Prefect REST API Settings.

    Source code in prefect/server/services/late_runs.py
    class MarkLateRuns(LoopService):\n    \"\"\"\n    A simple loop service responsible for identifying flow runs that are \"late\".\n\n    A flow run is defined as \"late\" if has not scheduled within a certain amount\n    of time after its scheduled start time. The exact amount is configurable in\n    Prefect REST API Settings.\n    \"\"\"\n\n    def __init__(self, loop_seconds: float = None, **kwargs):\n        super().__init__(\n            loop_seconds=loop_seconds\n            or PREFECT_API_SERVICES_LATE_RUNS_LOOP_SECONDS.value(),\n            **kwargs,\n        )\n\n        # mark runs late if they are this far past their expected start time\n        self.mark_late_after: datetime.timedelta = (\n            PREFECT_API_SERVICES_LATE_RUNS_AFTER_SECONDS.value()\n        )\n\n        # query for this many runs to mark as late at once\n        self.batch_size = 400\n\n    @inject_db\n    async def run_once(self, db: PrefectDBInterface):\n        \"\"\"\n        Mark flow runs as late by:\n\n        - Querying for flow runs in a scheduled state that are Scheduled to start in the past\n        - For any runs past the \"late\" threshold, setting the flow run state to a new `Late` state\n        \"\"\"\n        scheduled_to_start_before = pendulum.now(\"UTC\").subtract(\n            seconds=self.mark_late_after.total_seconds()\n        )\n\n        while True:\n            async with db.session_context(begin_transaction=True) as session:\n                query = self._get_select_late_flow_runs_query(\n                    scheduled_to_start_before=scheduled_to_start_before, db=db\n                )\n\n                result = await session.execute(query)\n                runs = result.all()\n\n                # mark each run as late\n                for run in runs:\n                    await self._mark_flow_run_as_late(session=session, flow_run=run)\n\n                # if no runs were found, exit the loop\n                if len(runs) < self.batch_size:\n                    break\n\n        self.logger.info(\"Finished monitoring for late runs.\")\n\n    @inject_db\n    def _get_select_late_flow_runs_query(\n        self, scheduled_to_start_before: datetime.datetime, db: PrefectDBInterface\n    ):\n        \"\"\"\n        Returns a sqlalchemy query for late flow runs.\n\n        Args:\n            scheduled_to_start_before: the maximum next scheduled start time of\n                scheduled flow runs to consider in the returned query\n        \"\"\"\n        query = (\n            sa.select(\n                db.FlowRun.id,\n                db.FlowRun.next_scheduled_start_time,\n            )\n            .where(\n                # The next scheduled start time is in the past, including the mark late\n                # after buffer\n                (db.FlowRun.next_scheduled_start_time <= scheduled_to_start_before),\n                db.FlowRun.state_type == states.StateType.SCHEDULED,\n                db.FlowRun.state_name == \"Scheduled\",\n            )\n            .limit(self.batch_size)\n        )\n        return query\n\n    async def _mark_flow_run_as_late(\n        self, session: AsyncSession, flow_run: PrefectDBInterface.FlowRun\n    ) -> None:\n        \"\"\"\n        Mark a flow run as late.\n\n        Pass-through method for overrides.\n        \"\"\"\n        await models.flow_runs.set_flow_run_state(\n            session=session,\n            flow_run_id=flow_run.id,\n            state=states.Late(scheduled_time=flow_run.next_scheduled_start_time),\n            force=True,\n        )\n
    "},{"location":"api-ref/server/services/late_runs/#prefect.server.services.late_runs.MarkLateRuns.run_once","title":"run_once async","text":"

    Mark flow runs as late by:

    • Querying for flow runs in a scheduled state that are Scheduled to start in the past
    • For any runs past the \"late\" threshold, setting the flow run state to a new Late state
    Source code in prefect/server/services/late_runs.py
    @inject_db\nasync def run_once(self, db: PrefectDBInterface):\n    \"\"\"\n    Mark flow runs as late by:\n\n    - Querying for flow runs in a scheduled state that are Scheduled to start in the past\n    - For any runs past the \"late\" threshold, setting the flow run state to a new `Late` state\n    \"\"\"\n    scheduled_to_start_before = pendulum.now(\"UTC\").subtract(\n        seconds=self.mark_late_after.total_seconds()\n    )\n\n    while True:\n        async with db.session_context(begin_transaction=True) as session:\n            query = self._get_select_late_flow_runs_query(\n                scheduled_to_start_before=scheduled_to_start_before, db=db\n            )\n\n            result = await session.execute(query)\n            runs = result.all()\n\n            # mark each run as late\n            for run in runs:\n                await self._mark_flow_run_as_late(session=session, flow_run=run)\n\n            # if no runs were found, exit the loop\n            if len(runs) < self.batch_size:\n                break\n\n    self.logger.info(\"Finished monitoring for late runs.\")\n
    "},{"location":"api-ref/server/services/loop_service/","title":"server.services.loop_service","text":""},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service","title":"prefect.server.services.loop_service","text":"

    The base class for all Prefect REST API loop services.

    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.LoopService","title":"LoopService","text":"

    Loop services are relatively lightweight maintenance routines that need to run periodically.

    This class makes it straightforward to design and integrate them. Users only need to define the run_once coroutine to describe the behavior of the service on each loop.

    Source code in prefect/server/services/loop_service.py
    class LoopService:\n    \"\"\"\n    Loop services are relatively lightweight maintenance routines that need to run periodically.\n\n    This class makes it straightforward to design and integrate them. Users only need to\n    define the `run_once` coroutine to describe the behavior of the service on each loop.\n    \"\"\"\n\n    loop_seconds = 60\n\n    def __init__(self, loop_seconds: float = None, handle_signals: bool = True):\n        \"\"\"\n        Args:\n            loop_seconds (float): if provided, overrides the loop interval\n                otherwise specified as a class variable\n            handle_signals (bool): if True (default), SIGINT and SIGTERM are\n                gracefully intercepted and shut down the running service.\n        \"\"\"\n        if loop_seconds:\n            self.loop_seconds = loop_seconds  # seconds between runs\n        self._should_stop = False  # flag for whether the service should stop running\n        self._is_running = False  # flag for whether the service is running\n        self.name = type(self).__name__\n        self.logger = get_logger(f\"server.services.{self.name.lower()}\")\n\n        if handle_signals:\n            _register_signal(signal.SIGINT, self._stop)\n            _register_signal(signal.SIGTERM, self._stop)\n\n    @inject_db\n    async def _on_start(self, db: PrefectDBInterface) -> None:\n        \"\"\"\n        Called prior to running the service\n        \"\"\"\n        # reset the _should_stop flag\n        self._should_stop = False\n        # set the _is_running flag\n        self._is_running = True\n\n    async def _on_stop(self) -> None:\n        \"\"\"\n        Called after running the service\n        \"\"\"\n        # reset the _is_running flag\n        self._is_running = False\n\n    async def start(self, loops=None) -> None:\n        \"\"\"\n        Run the service `loops` time. Pass loops=None to run forever.\n\n        Args:\n            loops (int, optional): the number of loops to run before exiting.\n        \"\"\"\n\n        await self._on_start()\n\n        i = 0\n        while not self._should_stop:\n            start_time = pendulum.now(\"UTC\")\n\n            try:\n                self.logger.debug(f\"About to run {self.name}...\")\n                await self.run_once()\n\n            except NotImplementedError as exc:\n                raise exc from None\n\n            # if an error is raised, log and continue\n            except Exception as exc:\n                self.logger.error(f\"Unexpected error in: {repr(exc)}\", exc_info=True)\n\n            end_time = pendulum.now(\"UTC\")\n\n            # if service took longer than its loop interval, log a warning\n            # that the interval might be too short\n            if (end_time - start_time).total_seconds() > self.loop_seconds:\n                self.logger.warning(\n                    f\"{self.name} took {(end_time-start_time).total_seconds()} seconds\"\n                    \" to run, which is longer than its loop interval of\"\n                    f\" {self.loop_seconds} seconds.\"\n                )\n\n            # check if early stopping was requested\n            i += 1\n            if loops is not None and i == loops:\n                self.logger.debug(f\"{self.name} exiting after {loops} loop(s).\")\n                await self.stop(block=False)\n\n            # next run is every \"loop seconds\" after each previous run *started*.\n            # note that if the loop took unexpectedly long, the \"next_run\" time\n            # might be in the past, which will result in an instant start\n            next_run = max(\n                start_time.add(seconds=self.loop_seconds), pendulum.now(\"UTC\")\n            )\n            self.logger.debug(f\"Finished running {self.name}. Next run at {next_run}\")\n\n            # check the `_should_stop` flag every 1 seconds until the next run time is reached\n            while pendulum.now(\"UTC\") < next_run and not self._should_stop:\n                await asyncio.sleep(\n                    min(1, (next_run - pendulum.now(\"UTC\")).total_seconds())\n                )\n\n        await self._on_stop()\n\n    async def stop(self, block=True) -> None:\n        \"\"\"\n        Gracefully stops a running LoopService and optionally blocks until the\n        service stops.\n\n        Args:\n            block (bool): if True, blocks until the service is\n                finished running. Otherwise it requests a stop and returns but\n                the service may still be running a final loop.\n\n        \"\"\"\n        self._stop()\n\n        if block:\n            # if block=True, sleep until the service stops running,\n            # but no more than `loop_seconds` to avoid a deadlock\n            with anyio.move_on_after(self.loop_seconds):\n                while self._is_running:\n                    await asyncio.sleep(0.1)\n\n            # if the service is still running after `loop_seconds`, something's wrong\n            if self._is_running:\n                self.logger.warning(\n                    f\"`stop(block=True)` was called on {self.name} but more than one\"\n                    f\" loop interval ({self.loop_seconds} seconds) has passed. This\"\n                    \" usually means something is wrong. If `stop()` was called from\"\n                    \" inside the loop service, use `stop(block=False)` instead.\"\n                )\n\n    def _stop(self, *_) -> None:\n        \"\"\"\n        Private, synchronous method for setting the `_should_stop` flag. Takes arbitrary\n        arguments so it can be used as a signal handler.\n        \"\"\"\n        self._should_stop = True\n\n    async def run_once(self) -> None:\n        \"\"\"\n        Represents one loop of the service.\n\n        Users should override this method.\n\n        To actually run the service once, call `LoopService().start(loops=1)`\n        instead of `LoopService().run_once()`, because this method will not invoke setup\n        and teardown methods properly.\n        \"\"\"\n        raise NotImplementedError(\"LoopService subclasses must implement this method.\")\n
    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.LoopService.run_once","title":"run_once async","text":"

    Represents one loop of the service.

    Users should override this method.

    To actually run the service once, call LoopService().start(loops=1) instead of LoopService().run_once(), because this method will not invoke setup and teardown methods properly.

    Source code in prefect/server/services/loop_service.py
    async def run_once(self) -> None:\n    \"\"\"\n    Represents one loop of the service.\n\n    Users should override this method.\n\n    To actually run the service once, call `LoopService().start(loops=1)`\n    instead of `LoopService().run_once()`, because this method will not invoke setup\n    and teardown methods properly.\n    \"\"\"\n    raise NotImplementedError(\"LoopService subclasses must implement this method.\")\n
    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.LoopService.start","title":"start async","text":"

    Run the service loops time. Pass loops=None to run forever.

    Parameters:

    Name Type Description Default loops int

    the number of loops to run before exiting.

    None Source code in prefect/server/services/loop_service.py
    async def start(self, loops=None) -> None:\n    \"\"\"\n    Run the service `loops` time. Pass loops=None to run forever.\n\n    Args:\n        loops (int, optional): the number of loops to run before exiting.\n    \"\"\"\n\n    await self._on_start()\n\n    i = 0\n    while not self._should_stop:\n        start_time = pendulum.now(\"UTC\")\n\n        try:\n            self.logger.debug(f\"About to run {self.name}...\")\n            await self.run_once()\n\n        except NotImplementedError as exc:\n            raise exc from None\n\n        # if an error is raised, log and continue\n        except Exception as exc:\n            self.logger.error(f\"Unexpected error in: {repr(exc)}\", exc_info=True)\n\n        end_time = pendulum.now(\"UTC\")\n\n        # if service took longer than its loop interval, log a warning\n        # that the interval might be too short\n        if (end_time - start_time).total_seconds() > self.loop_seconds:\n            self.logger.warning(\n                f\"{self.name} took {(end_time-start_time).total_seconds()} seconds\"\n                \" to run, which is longer than its loop interval of\"\n                f\" {self.loop_seconds} seconds.\"\n            )\n\n        # check if early stopping was requested\n        i += 1\n        if loops is not None and i == loops:\n            self.logger.debug(f\"{self.name} exiting after {loops} loop(s).\")\n            await self.stop(block=False)\n\n        # next run is every \"loop seconds\" after each previous run *started*.\n        # note that if the loop took unexpectedly long, the \"next_run\" time\n        # might be in the past, which will result in an instant start\n        next_run = max(\n            start_time.add(seconds=self.loop_seconds), pendulum.now(\"UTC\")\n        )\n        self.logger.debug(f\"Finished running {self.name}. Next run at {next_run}\")\n\n        # check the `_should_stop` flag every 1 seconds until the next run time is reached\n        while pendulum.now(\"UTC\") < next_run and not self._should_stop:\n            await asyncio.sleep(\n                min(1, (next_run - pendulum.now(\"UTC\")).total_seconds())\n            )\n\n    await self._on_stop()\n
    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.LoopService.stop","title":"stop async","text":"

    Gracefully stops a running LoopService and optionally blocks until the service stops.

    Parameters:

    Name Type Description Default block bool

    if True, blocks until the service is finished running. Otherwise it requests a stop and returns but the service may still be running a final loop.

    True Source code in prefect/server/services/loop_service.py
    async def stop(self, block=True) -> None:\n    \"\"\"\n    Gracefully stops a running LoopService and optionally blocks until the\n    service stops.\n\n    Args:\n        block (bool): if True, blocks until the service is\n            finished running. Otherwise it requests a stop and returns but\n            the service may still be running a final loop.\n\n    \"\"\"\n    self._stop()\n\n    if block:\n        # if block=True, sleep until the service stops running,\n        # but no more than `loop_seconds` to avoid a deadlock\n        with anyio.move_on_after(self.loop_seconds):\n            while self._is_running:\n                await asyncio.sleep(0.1)\n\n        # if the service is still running after `loop_seconds`, something's wrong\n        if self._is_running:\n            self.logger.warning(\n                f\"`stop(block=True)` was called on {self.name} but more than one\"\n                f\" loop interval ({self.loop_seconds} seconds) has passed. This\"\n                \" usually means something is wrong. If `stop()` was called from\"\n                \" inside the loop service, use `stop(block=False)` instead.\"\n            )\n
    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.run_multiple_services","title":"run_multiple_services async","text":"

    Only one signal handler can be active at a time, so this function takes a list of loop services and runs all of them with a global signal handler.

    Source code in prefect/server/services/loop_service.py
    async def run_multiple_services(loop_services: List[LoopService]):\n    \"\"\"\n    Only one signal handler can be active at a time, so this function takes a list\n    of loop services and runs all of them with a global signal handler.\n    \"\"\"\n\n    def stop_all_services(self, *_):\n        for service in loop_services:\n            service._stop()\n\n    signal.signal(signal.SIGINT, stop_all_services)\n    signal.signal(signal.SIGTERM, stop_all_services)\n    await asyncio.gather(*[service.start() for service in loop_services])\n
    "},{"location":"api-ref/server/services/scheduler/","title":"server.services.scheduler","text":""},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler","title":"prefect.server.services.scheduler","text":"

    The Scheduler service.

    "},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler.RecentDeploymentsScheduler","title":"RecentDeploymentsScheduler","text":"

    Bases: Scheduler

    A scheduler that only schedules deployments that were updated very recently. This scheduler can run on a tight loop and ensure that runs from newly-created or updated deployments are rapidly scheduled without having to wait for the \"main\" scheduler to complete its loop.

    Note that scheduling is idempotent, so its ok for this scheduler to attempt to schedule the same deployments as the main scheduler. It's purpose is to accelerate scheduling for any deployments that users are interacting with.

    Source code in prefect/server/services/scheduler.py
    class RecentDeploymentsScheduler(Scheduler):\n    \"\"\"\n    A scheduler that only schedules deployments that were updated very recently.\n    This scheduler can run on a tight loop and ensure that runs from\n    newly-created or updated deployments are rapidly scheduled without having to\n    wait for the \"main\" scheduler to complete its loop.\n\n    Note that scheduling is idempotent, so its ok for this scheduler to attempt\n    to schedule the same deployments as the main scheduler. It's purpose is to\n    accelerate scheduling for any deployments that users are interacting with.\n    \"\"\"\n\n    # this scheduler runs on a tight loop\n    loop_seconds = 5\n\n    @inject_db\n    def _get_select_deployments_to_schedule_query(self, db: PrefectDBInterface):\n        \"\"\"\n        Returns a sqlalchemy query for selecting deployments to schedule\n        \"\"\"\n        query = (\n            sa.select(db.Deployment.id)\n            .where(\n                db.Deployment.is_schedule_active.is_(True),\n                db.Deployment.schedule.is_not(None),\n                # use a slightly larger window than the loop interval to pick up\n                # any deployments that were created *while* the scheduler was\n                # last running (assuming the scheduler takes less than one\n                # second to run). Scheduling is idempotent so picking up schedules\n                # multiple times is not a concern.\n                db.Deployment.updated\n                >= pendulum.now(\"UTC\").subtract(seconds=self.loop_seconds + 1),\n            )\n            .order_by(db.Deployment.id)\n            .limit(self.deployment_batch_size)\n        )\n        return query\n
    "},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler.Scheduler","title":"Scheduler","text":"

    Bases: LoopService

    A loop service that schedules flow runs from deployments.

    Source code in prefect/server/services/scheduler.py
    class Scheduler(LoopService):\n    \"\"\"\n    A loop service that schedules flow runs from deployments.\n    \"\"\"\n\n    # the main scheduler takes its loop interval from\n    # PREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS\n    loop_seconds = None\n\n    def __init__(self, loop_seconds: float = None, **kwargs):\n        super().__init__(\n            loop_seconds=(\n                loop_seconds\n                or self.loop_seconds\n                or PREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS.value()\n            ),\n            **kwargs,\n        )\n        self.deployment_batch_size: int = (\n            PREFECT_API_SERVICES_SCHEDULER_DEPLOYMENT_BATCH_SIZE.value()\n        )\n        self.max_runs: int = PREFECT_API_SERVICES_SCHEDULER_MAX_RUNS.value()\n        self.min_runs: int = PREFECT_API_SERVICES_SCHEDULER_MIN_RUNS.value()\n        self.max_scheduled_time: datetime.timedelta = (\n            PREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME.value()\n        )\n        self.min_scheduled_time: datetime.timedelta = (\n            PREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME.value()\n        )\n        self.insert_batch_size = (\n            PREFECT_API_SERVICES_SCHEDULER_INSERT_BATCH_SIZE.value()\n        )\n\n    @inject_db\n    async def run_once(self, db: PrefectDBInterface):\n        \"\"\"\n        Schedule flow runs by:\n\n        - Querying for deployments with active schedules\n        - Generating the next set of flow runs based on each deployments schedule\n        - Inserting all scheduled flow runs into the database\n\n        All inserted flow runs are committed to the database at the termination of the\n        loop.\n        \"\"\"\n        total_inserted_runs = 0\n\n        last_id = None\n        while True:\n            async with db.session_context(begin_transaction=False) as session:\n                query = self._get_select_deployments_to_schedule_query()\n\n                # use cursor based pagination\n                if last_id:\n                    query = query.where(db.Deployment.id > last_id)\n\n                result = await session.execute(query)\n                deployment_ids = result.scalars().unique().all()\n\n                # collect runs across all deployments\n                try:\n                    runs_to_insert = await self._collect_flow_runs(\n                        session=session, deployment_ids=deployment_ids\n                    )\n                except TryAgain:\n                    continue\n\n            # bulk insert the runs based on batch size setting\n            for batch in batched_iterable(runs_to_insert, self.insert_batch_size):\n                async with db.session_context(begin_transaction=True) as session:\n                    inserted_runs = await self._insert_scheduled_flow_runs(\n                        session=session, runs=batch\n                    )\n                    total_inserted_runs += len(inserted_runs)\n\n            # if this is the last page of deployments, exit the loop\n            if len(deployment_ids) < self.deployment_batch_size:\n                break\n            else:\n                # record the last deployment ID\n                last_id = deployment_ids[-1]\n\n        self.logger.info(f\"Scheduled {total_inserted_runs} runs.\")\n\n    @inject_db\n    def _get_select_deployments_to_schedule_query(self, db: PrefectDBInterface):\n        \"\"\"\n        Returns a sqlalchemy query for selecting deployments to schedule.\n\n        The query gets the IDs of any deployments with:\n\n            - an active schedule\n            - EITHER:\n                - fewer than `min_runs` auto-scheduled runs\n                - OR the max scheduled time is less than `max_scheduled_time` in the future\n        \"\"\"\n        now = pendulum.now(\"UTC\")\n        query = (\n            sa.select(db.Deployment.id)\n            .select_from(db.Deployment)\n            # TODO: on Postgres, this could be replaced with a lateral join that\n            # sorts by `next_scheduled_start_time desc` and limits by\n            # `self.min_runs` for a ~ 50% speedup. At the time of writing,\n            # performance of this universal query appears to be fast enough that\n            # this optimization is not worth maintaining db-specific queries\n            .join(\n                db.FlowRun,\n                # join on matching deployments, only picking up future scheduled runs\n                sa.and_(\n                    db.Deployment.id == db.FlowRun.deployment_id,\n                    db.FlowRun.state_type == StateType.SCHEDULED,\n                    db.FlowRun.next_scheduled_start_time >= now,\n                    db.FlowRun.auto_scheduled.is_(True),\n                ),\n                isouter=True,\n            )\n            .where(\n                db.Deployment.is_schedule_active.is_(True),\n                db.Deployment.schedule.is_not(None),\n            )\n            .group_by(db.Deployment.id)\n            # having EITHER fewer than three runs OR runs not scheduled far enough out\n            .having(\n                sa.or_(\n                    sa.func.count(db.FlowRun.next_scheduled_start_time) < self.min_runs,\n                    sa.func.max(db.FlowRun.next_scheduled_start_time)\n                    < now + self.min_scheduled_time,\n                )\n            )\n            .order_by(db.Deployment.id)\n            .limit(self.deployment_batch_size)\n        )\n        return query\n\n    async def _collect_flow_runs(\n        self,\n        session: sa.orm.Session,\n        deployment_ids: List[UUID],\n    ) -> List[Dict]:\n        runs_to_insert = []\n        for deployment_id in deployment_ids:\n            now = pendulum.now(\"UTC\")\n            # guard against erroneously configured schedules\n            try:\n                runs_to_insert.extend(\n                    await self._generate_scheduled_flow_runs(\n                        session=session,\n                        deployment_id=deployment_id,\n                        start_time=now,\n                        end_time=now + self.max_scheduled_time,\n                        min_time=self.min_scheduled_time,\n                        min_runs=self.min_runs,\n                        max_runs=self.max_runs,\n                    )\n                )\n            except Exception:\n                self.logger.exception(\n                    f\"Error scheduling deployment {deployment_id!r}.\",\n                )\n            finally:\n                connection = await session.connection()\n                if connection.invalidated:\n                    # If the error we handled above was the kind of database error that\n                    # causes underlying transaction to rollback and the connection to\n                    # become invalidated, rollback this session.  Errors that may cause\n                    # this are connection drops, database restarts, and things of the\n                    # sort.\n                    #\n                    # This rollback _does not rollback a transaction_, since that has\n                    # actually already happened due to the error above.  It brings the\n                    # Python session in sync with underlying connection so that when we\n                    # exec the outer with block, the context manager will not attempt to\n                    # commit the session.\n                    #\n                    # Then, raise TryAgain to break out of these nested loops, back to\n                    # the outer loop, where we'll begin a new transaction with\n                    # session.begin() in the next loop iteration.\n                    await session.rollback()\n                    raise TryAgain()\n        return runs_to_insert\n\n    @inject_db\n    async def _generate_scheduled_flow_runs(\n        self,\n        session: sa.orm.Session,\n        deployment_id: UUID,\n        start_time: datetime.datetime,\n        end_time: datetime.datetime,\n        min_time: datetime.timedelta,\n        min_runs: int,\n        max_runs: int,\n        db: PrefectDBInterface,\n    ) -> List[Dict]:\n        \"\"\"\n        Given a `deployment_id` and schedule params, generates a list of flow run\n        objects and associated scheduled states that represent scheduled flow runs.\n\n        Pass-through method for overrides.\n\n\n        Args:\n            session: a database session\n            deployment_id: the id of the deployment to schedule\n            start_time: the time from which to start scheduling runs\n            end_time: runs will be scheduled until at most this time\n            min_time: runs will be scheduled until at least this far in the future\n            min_runs: a minimum amount of runs to schedule\n            max_runs: a maximum amount of runs to schedule\n\n        This function will generate the minimum number of runs that satisfy the min\n        and max times, and the min and max counts. Specifically, the following order\n        will be respected:\n\n            - Runs will be generated starting on or after the `start_time`\n            - No more than `max_runs` runs will be generated\n            - No runs will be generated after `end_time` is reached\n            - At least `min_runs` runs will be generated\n            - Runs will be generated until at least `start_time + min_time` is reached\n\n        \"\"\"\n        return await models.deployments._generate_scheduled_flow_runs(\n            session=session,\n            deployment_id=deployment_id,\n            start_time=start_time,\n            end_time=end_time,\n            min_time=min_time,\n            min_runs=min_runs,\n            max_runs=max_runs,\n        )\n\n    @inject_db\n    async def _insert_scheduled_flow_runs(\n        self,\n        session: sa.orm.Session,\n        runs: List[Dict],\n        db: PrefectDBInterface,\n    ) -> List[UUID]:\n        \"\"\"\n        Given a list of flow runs to schedule, as generated by\n        `_generate_scheduled_flow_runs`, inserts them into the database. Note this is a\n        separate method to facilitate batch operations on many scheduled runs.\n\n        Pass-through method for overrides.\n        \"\"\"\n        return await models.deployments._insert_scheduled_flow_runs(\n            session=session, runs=runs\n        )\n
    "},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler.Scheduler.run_once","title":"run_once async","text":"

    Schedule flow runs by:

    • Querying for deployments with active schedules
    • Generating the next set of flow runs based on each deployments schedule
    • Inserting all scheduled flow runs into the database

    All inserted flow runs are committed to the database at the termination of the loop.

    Source code in prefect/server/services/scheduler.py
    @inject_db\nasync def run_once(self, db: PrefectDBInterface):\n    \"\"\"\n    Schedule flow runs by:\n\n    - Querying for deployments with active schedules\n    - Generating the next set of flow runs based on each deployments schedule\n    - Inserting all scheduled flow runs into the database\n\n    All inserted flow runs are committed to the database at the termination of the\n    loop.\n    \"\"\"\n    total_inserted_runs = 0\n\n    last_id = None\n    while True:\n        async with db.session_context(begin_transaction=False) as session:\n            query = self._get_select_deployments_to_schedule_query()\n\n            # use cursor based pagination\n            if last_id:\n                query = query.where(db.Deployment.id > last_id)\n\n            result = await session.execute(query)\n            deployment_ids = result.scalars().unique().all()\n\n            # collect runs across all deployments\n            try:\n                runs_to_insert = await self._collect_flow_runs(\n                    session=session, deployment_ids=deployment_ids\n                )\n            except TryAgain:\n                continue\n\n        # bulk insert the runs based on batch size setting\n        for batch in batched_iterable(runs_to_insert, self.insert_batch_size):\n            async with db.session_context(begin_transaction=True) as session:\n                inserted_runs = await self._insert_scheduled_flow_runs(\n                    session=session, runs=batch\n                )\n                total_inserted_runs += len(inserted_runs)\n\n        # if this is the last page of deployments, exit the loop\n        if len(deployment_ids) < self.deployment_batch_size:\n            break\n        else:\n            # record the last deployment ID\n            last_id = deployment_ids[-1]\n\n    self.logger.info(f\"Scheduled {total_inserted_runs} runs.\")\n
    "},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler.TryAgain","title":"TryAgain","text":"

    Bases: Exception

    Internal control-flow exception used to retry the Scheduler's main loop

    Source code in prefect/server/services/scheduler.py
    class TryAgain(Exception):\n    \"\"\"Internal control-flow exception used to retry the Scheduler's main loop\"\"\"\n
    "},{"location":"api-ref/server/utilities/database/","title":"server.utilities.database","text":""},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database","title":"prefect.server.utilities.database","text":"

    Utilities for interacting with Prefect REST API database and ORM layer.

    Prefect supports both SQLite and Postgres. Many of these utilities allow the Prefect REST API to seamlessly switch between the two.

    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.GenerateUUID","title":"GenerateUUID","text":"

    Bases: FunctionElement

    Platform-independent UUID default generator. Note the actual functionality for this class is specified in the compiles-decorated functions below

    Source code in prefect/server/utilities/database.py
    class GenerateUUID(FunctionElement):\n    \"\"\"\n    Platform-independent UUID default generator.\n    Note the actual functionality for this class is specified in the\n    `compiles`-decorated functions below\n    \"\"\"\n\n    name = \"uuid_default\"\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.JSON","title":"JSON","text":"

    Bases: TypeDecorator

    JSON type that returns SQLAlchemy's dialect-specific JSON types, where possible. Uses generic JSON otherwise.

    The \"base\" type is postgresql.JSONB to expose useful methods prior to SQL compilation

    Source code in prefect/server/utilities/database.py
    class JSON(TypeDecorator):\n    \"\"\"\n    JSON type that returns SQLAlchemy's dialect-specific JSON types, where\n    possible. Uses generic JSON otherwise.\n\n    The \"base\" type is postgresql.JSONB to expose useful methods prior\n    to SQL compilation\n    \"\"\"\n\n    impl = postgresql.JSONB\n    cache_ok = True\n\n    def load_dialect_impl(self, dialect):\n        if dialect.name == \"postgresql\":\n            return dialect.type_descriptor(postgresql.JSONB(none_as_null=True))\n        elif dialect.name == \"sqlite\":\n            return dialect.type_descriptor(sqlite.JSON(none_as_null=True))\n        else:\n            return dialect.type_descriptor(sa.JSON(none_as_null=True))\n\n    def process_bind_param(self, value, dialect):\n        \"\"\"Prepares the given value to be used as a JSON field in a parameter binding\"\"\"\n        if not value:\n            return value\n\n        # PostgreSQL does not support the floating point extrema values `NaN`,\n        # `-Infinity`, or `Infinity`\n        # https://www.postgresql.org/docs/current/datatype-json.html#JSON-TYPE-MAPPING-TABLE\n        #\n        # SQLite supports storing and retrieving full JSON values that include\n        # `NaN`, `-Infinity`, or `Infinity`, but any query that requires SQLite to parse\n        # the value (like `json_extract`) will fail.\n        #\n        # Replace any `NaN`, `-Infinity`, or `Infinity` values with `None` in the\n        # returned value.  See more about `parse_constant` at\n        # https://docs.python.org/3/library/json.html#json.load.\n        return json.loads(json.dumps(value), parse_constant=lambda c: None)\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.Pydantic","title":"Pydantic","text":"

    Bases: TypeDecorator

    A pydantic type that converts inserted parameters to json and converts read values to the pydantic type.

    Source code in prefect/server/utilities/database.py
    class Pydantic(TypeDecorator):\n    \"\"\"\n    A pydantic type that converts inserted parameters to\n    json and converts read values to the pydantic type.\n    \"\"\"\n\n    impl = JSON\n    cache_ok = True\n\n    def __init__(self, pydantic_type, sa_column_type=None):\n        super().__init__()\n        self._pydantic_type = pydantic_type\n        if sa_column_type is not None:\n            self.impl = sa_column_type\n\n    def process_bind_param(self, value, dialect):\n        if value is None:\n            return None\n        # parse the value to ensure it complies with the schema\n        # (this will raise validation errors if not)\n        value = pydantic.parse_obj_as(self._pydantic_type, value)\n        # sqlalchemy requires the bind parameter's value to be a python-native\n        # collection of JSON-compatible objects. we achieve that by dumping the\n        # value to a json string using the pydantic JSON encoder and re-parsing\n        # it into a python-native form.\n        return json.loads(json.dumps(value, default=pydantic.json.pydantic_encoder))\n\n    def process_result_value(self, value, dialect):\n        if value is not None:\n            # load the json object into a fully hydrated typed object\n            return pydantic.parse_obj_as(self._pydantic_type, value)\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.Timestamp","title":"Timestamp","text":"

    Bases: TypeDecorator

    TypeDecorator that ensures that timestamps have a timezone.

    For SQLite, all timestamps are converted to UTC (since they are stored as naive timestamps without timezones) and recovered as UTC.

    Source code in prefect/server/utilities/database.py
    class Timestamp(TypeDecorator):\n    \"\"\"TypeDecorator that ensures that timestamps have a timezone.\n\n    For SQLite, all timestamps are converted to UTC (since they are stored\n    as naive timestamps without timezones) and recovered as UTC.\n    \"\"\"\n\n    impl = sa.TIMESTAMP(timezone=True)\n    cache_ok = True\n\n    def load_dialect_impl(self, dialect):\n        if dialect.name == \"postgresql\":\n            return dialect.type_descriptor(postgresql.TIMESTAMP(timezone=True))\n        elif dialect.name == \"sqlite\":\n            return dialect.type_descriptor(\n                sqlite.DATETIME(\n                    # SQLite is very particular about datetimes, and performs all comparisons\n                    # as alphanumeric comparisons without regard for actual timestamp\n                    # semantics or timezones. Therefore, it's important to have uniform\n                    # and sortable datetime representations. The default is an ISO8601-compatible\n                    # string with NO time zone and a space (\" \") delimiter between the date\n                    # and the time. The below settings can be used to add a \"T\" delimiter but\n                    # will require all other sqlite datetimes to be set similarly, including\n                    # the custom default value for datetime columns and any handwritten SQL\n                    # formed with `strftime()`.\n                    #\n                    # store with \"T\" separator for time\n                    # storage_format=(\n                    #     \"%(year)04d-%(month)02d-%(day)02d\"\n                    #     \"T%(hour)02d:%(minute)02d:%(second)02d.%(microsecond)06d\"\n                    # ),\n                    # handle ISO 8601 with \"T\" or \" \" as the time separator\n                    # regexp=r\"(\\d+)-(\\d+)-(\\d+)[T ](\\d+):(\\d+):(\\d+).(\\d+)\",\n                )\n            )\n        else:\n            return dialect.type_descriptor(sa.TIMESTAMP(timezone=True))\n\n    def process_bind_param(self, value, dialect):\n        if value is None:\n            return None\n        else:\n            if value.tzinfo is None:\n                raise ValueError(\"Timestamps must have a timezone.\")\n            elif dialect.name == \"sqlite\":\n                return pendulum.instance(value).in_timezone(\"UTC\")\n            else:\n                return value\n\n    def process_result_value(self, value, dialect):\n        # retrieve timestamps in their native timezone (or UTC)\n        if value is not None:\n            return pendulum.instance(value).in_timezone(\"utc\")\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.UUID","title":"UUID","text":"

    Bases: TypeDecorator

    Platform-independent UUID type.

    Uses PostgreSQL's UUID type, otherwise uses CHAR(36), storing as stringified hex values with hyphens.

    Source code in prefect/server/utilities/database.py
    class UUID(TypeDecorator):\n    \"\"\"\n    Platform-independent UUID type.\n\n    Uses PostgreSQL's UUID type, otherwise uses\n    CHAR(36), storing as stringified hex values with\n    hyphens.\n    \"\"\"\n\n    impl = TypeEngine\n    cache_ok = True\n\n    def load_dialect_impl(self, dialect):\n        if dialect.name == \"postgresql\":\n            return dialect.type_descriptor(postgresql.UUID())\n        else:\n            return dialect.type_descriptor(CHAR(36))\n\n    def process_bind_param(self, value, dialect):\n        if value is None:\n            return None\n        elif dialect.name == \"postgresql\":\n            return str(value)\n        elif isinstance(value, uuid.UUID):\n            return str(value)\n        else:\n            return str(uuid.UUID(value))\n\n    def process_result_value(self, value, dialect):\n        if value is None:\n            return value\n        else:\n            if not isinstance(value, uuid.UUID):\n                value = uuid.UUID(value)\n            return value\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.date_add","title":"date_add","text":"

    Bases: FunctionElement

    Platform-independent way to add a date and an interval.

    Source code in prefect/server/utilities/database.py
    class date_add(FunctionElement):\n    \"\"\"\n    Platform-independent way to add a date and an interval.\n    \"\"\"\n\n    type = Timestamp()\n    name = \"date_add\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, dt, interval):\n        self.dt = dt\n        self.interval = interval\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.date_diff","title":"date_diff","text":"

    Bases: FunctionElement

    Platform-independent difference of dates. Computes d1 - d2.

    Source code in prefect/server/utilities/database.py
    class date_diff(FunctionElement):\n    \"\"\"\n    Platform-independent difference of dates. Computes d1 - d2.\n    \"\"\"\n\n    type = sa.Interval()\n    name = \"date_diff\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, d1, d2):\n        self.d1 = d1\n        self.d2 = d2\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.interval_add","title":"interval_add","text":"

    Bases: FunctionElement

    Platform-independent way to add two intervals.

    Source code in prefect/server/utilities/database.py
    class interval_add(FunctionElement):\n    \"\"\"\n    Platform-independent way to add two intervals.\n    \"\"\"\n\n    type = sa.Interval()\n    name = \"interval_add\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, i1, i2):\n        self.i1 = i1\n        self.i2 = i2\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.json_contains","title":"json_contains","text":"

    Bases: FunctionElement

    Platform independent json_contains operator, tests if the left expression contains the right expression.

    On postgres this is equivalent to the @> containment operator. https://www.postgresql.org/docs/current/functions-json.html

    Source code in prefect/server/utilities/database.py
    class json_contains(FunctionElement):\n    \"\"\"\n    Platform independent json_contains operator, tests if the\n    `left` expression contains the `right` expression.\n\n    On postgres this is equivalent to the @> containment operator.\n    https://www.postgresql.org/docs/current/functions-json.html\n    \"\"\"\n\n    type = BOOLEAN\n    name = \"json_contains\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, left, right):\n        self.left = left\n        self.right = right\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.json_has_all_keys","title":"json_has_all_keys","text":"

    Bases: FunctionElement

    Platform independent json_has_all_keys operator.

    On postgres this is equivalent to the ?& existence operator. https://www.postgresql.org/docs/current/functions-json.html

    Source code in prefect/server/utilities/database.py
    class json_has_all_keys(FunctionElement):\n    \"\"\"Platform independent json_has_all_keys operator.\n\n    On postgres this is equivalent to the ?& existence operator.\n    https://www.postgresql.org/docs/current/functions-json.html\n    \"\"\"\n\n    type = BOOLEAN\n    name = \"json_has_all_keys\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, json_expr, values: List):\n        self.json_expr = json_expr\n        if isinstance(values, list) and not all(isinstance(v, str) for v in values):\n            raise ValueError(\n                \"json_has_all_key values must be strings if provided as a literal list\"\n            )\n        self.values = values\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.json_has_any_key","title":"json_has_any_key","text":"

    Bases: FunctionElement

    Platform independent json_has_any_key operator.

    On postgres this is equivalent to the ?| existence operator. https://www.postgresql.org/docs/current/functions-json.html

    Source code in prefect/server/utilities/database.py
    class json_has_any_key(FunctionElement):\n    \"\"\"\n    Platform independent json_has_any_key operator.\n\n    On postgres this is equivalent to the ?| existence operator.\n    https://www.postgresql.org/docs/current/functions-json.html\n    \"\"\"\n\n    type = BOOLEAN\n    name = \"json_has_any_key\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, json_expr, values: List):\n        self.json_expr = json_expr\n        if not all(isinstance(v, str) for v in values):\n            raise ValueError(\"json_has_any_key values must be strings\")\n        self.values = values\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.now","title":"now","text":"

    Bases: FunctionElement

    Platform-independent \"now\" generator.

    Source code in prefect/server/utilities/database.py
    class now(FunctionElement):\n    \"\"\"\n    Platform-independent \"now\" generator.\n    \"\"\"\n\n    type = Timestamp()\n    name = \"now\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = True\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.get_dialect","title":"get_dialect","text":"

    Get the dialect of a session, engine, or connection url.

    Primary use case is figuring out whether the Prefect REST API is communicating with SQLite or Postgres.

    Example
    import prefect.settings\nfrom prefect.server.utilities.database import get_dialect\n\ndialect = get_dialect(PREFECT_API_DATABASE_CONNECTION_URL.value())\nif dialect == \"sqlite\":\n    print(\"Using SQLite!\")\nelse:\n    print(\"Using Postgres!\")\n
    Source code in prefect/server/utilities/database.py
    def get_dialect(\n    obj: Union[str, sa.orm.Session, sa.engine.Engine],\n) -> sa.engine.Dialect:\n    \"\"\"\n    Get the dialect of a session, engine, or connection url.\n\n    Primary use case is figuring out whether the Prefect REST API is communicating with\n    SQLite or Postgres.\n\n    Example:\n        ```python\n        import prefect.settings\n        from prefect.server.utilities.database import get_dialect\n\n        dialect = get_dialect(PREFECT_API_DATABASE_CONNECTION_URL.value())\n        if dialect == \"sqlite\":\n            print(\"Using SQLite!\")\n        else:\n            print(\"Using Postgres!\")\n        ```\n    \"\"\"\n    if isinstance(obj, sa.orm.Session):\n        url = obj.bind.url\n    elif isinstance(obj, sa.engine.Engine):\n        url = obj.url\n    else:\n        url = sa.engine.url.make_url(obj)\n\n    return url.get_dialect()\n
    "},{"location":"api-ref/server/utilities/schemas/","title":"server.utilities.schemas","text":""},{"location":"api-ref/server/utilities/schemas/#prefect.server.utilities.schemas","title":"prefect.server.utilities.schemas","text":""},{"location":"api-ref/server/utilities/server/","title":"server.utilities.server","text":""},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server","title":"prefect.server.utilities.server","text":"

    Utilities for the Prefect REST API server.

    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.PrefectAPIRoute","title":"PrefectAPIRoute","text":"

    Bases: APIRoute

    A FastAPIRoute class which attaches an async stack to requests that exits before a response is returned.

    Requests already have request.scope['fastapi_astack'] which is an async stack for the full scope of the request. This stack is used for managing contexts of FastAPI dependencies. If we want to close a dependency before the request is complete (i.e. before returning a response to the user), we need a stack with a different scope. This extension adds this stack at request.state.response_scoped_stack.

    Source code in prefect/server/utilities/server.py
    class PrefectAPIRoute(APIRoute):\n    \"\"\"\n    A FastAPIRoute class which attaches an async stack to requests that exits before\n    a response is returned.\n\n    Requests already have `request.scope['fastapi_astack']` which is an async stack for\n    the full scope of the request. This stack is used for managing contexts of FastAPI\n    dependencies. If we want to close a dependency before the request is complete\n    (i.e. before returning a response to the user), we need a stack with a different\n    scope. This extension adds this stack at `request.state.response_scoped_stack`.\n    \"\"\"\n\n    def get_route_handler(self) -> Callable[[Request], Coroutine[Any, Any, Response]]:\n        default_handler = super().get_route_handler()\n\n        async def handle_response_scoped_depends(request: Request) -> Response:\n            # Create a new stack scoped to exit before the response is returned\n            async with AsyncExitStack() as stack:\n                request.state.response_scoped_stack = stack\n                response = await default_handler(request)\n\n            return response\n\n        return handle_response_scoped_depends\n
    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.PrefectRouter","title":"PrefectRouter","text":"

    Bases: APIRouter

    A base class for Prefect REST API routers.

    Source code in prefect/server/utilities/server.py
    class PrefectRouter(APIRouter):\n    \"\"\"\n    A base class for Prefect REST API routers.\n    \"\"\"\n\n    def __init__(self, **kwargs: Any) -> None:\n        kwargs.setdefault(\"route_class\", PrefectAPIRoute)\n        super().__init__(**kwargs)\n\n    def add_api_route(\n        self, path: str, endpoint: Callable[..., Any], **kwargs: Any\n    ) -> None:\n        \"\"\"\n        Add an API route.\n\n        For routes that return content and have not specified a `response_model`,\n        use return type annotation to infer the response model.\n\n        For routes that return No-Content status codes, explicitly set\n        a `response_class` to ensure nothing is returned in the response body.\n        \"\"\"\n        if kwargs.get(\"status_code\") == status.HTTP_204_NO_CONTENT:\n            # any routes that return No-Content status codes must\n            # explicitly set a response_class that will handle status codes\n            # and not return anything in the body\n            kwargs[\"response_class\"] = Response\n        if kwargs.get(\"response_model\") is None:\n            kwargs[\"response_model\"] = get_type_hints(endpoint).get(\"return\")\n        return super().add_api_route(path, endpoint, **kwargs)\n
    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.PrefectRouter.add_api_route","title":"add_api_route","text":"

    Add an API route.

    For routes that return content and have not specified a response_model, use return type annotation to infer the response model.

    For routes that return No-Content status codes, explicitly set a response_class to ensure nothing is returned in the response body.

    Source code in prefect/server/utilities/server.py
    def add_api_route(\n    self, path: str, endpoint: Callable[..., Any], **kwargs: Any\n) -> None:\n    \"\"\"\n    Add an API route.\n\n    For routes that return content and have not specified a `response_model`,\n    use return type annotation to infer the response model.\n\n    For routes that return No-Content status codes, explicitly set\n    a `response_class` to ensure nothing is returned in the response body.\n    \"\"\"\n    if kwargs.get(\"status_code\") == status.HTTP_204_NO_CONTENT:\n        # any routes that return No-Content status codes must\n        # explicitly set a response_class that will handle status codes\n        # and not return anything in the body\n        kwargs[\"response_class\"] = Response\n    if kwargs.get(\"response_model\") is None:\n        kwargs[\"response_model\"] = get_type_hints(endpoint).get(\"return\")\n    return super().add_api_route(path, endpoint, **kwargs)\n
    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.method_paths_from_routes","title":"method_paths_from_routes","text":"

    Generate a set of strings describing the given routes in the format:

    For example, \"GET /logs/\"

    Source code in prefect/server/utilities/server.py
    def method_paths_from_routes(routes: Iterable[APIRoute]) -> Set[str]:\n    \"\"\"\n    Generate a set of strings describing the given routes in the format: <method> <path>\n\n    For example, \"GET /logs/\"\n    \"\"\"\n    method_paths = set()\n    for route in routes:\n        for method in route.methods:\n            method_paths.add(f\"{method} {route.path}\")\n\n    return method_paths\n
    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.response_scoped_dependency","title":"response_scoped_dependency","text":"

    Ensure that this dependency closes before the response is returned to the client. By default, FastAPI closes dependencies after sending the response.

    Uses an async stack that is exited before the response is returned. This is particularly useful for database sessions which must be committed before the client can do more work.

    Do not use a response-scoped dependency within a FastAPI background task.

    Background tasks run after FastAPI sends the response, so a response-scoped dependency will already be closed. Use a normal FastAPI dependency instead.

    Parameters:

    Name Type Description Default dependency Callable

    An async callable. FastAPI dependencies may still be used.

    required

    Returns:

    Type Description

    A wrapped dependency which will push the dependency context manager onto

    a stack when called.

    Source code in prefect/server/utilities/server.py
    def response_scoped_dependency(dependency: Callable):\n    \"\"\"\n    Ensure that this dependency closes before the response is returned to the client. By\n    default, FastAPI closes dependencies after sending the response.\n\n    Uses an async stack that is exited before the response is returned. This is\n    particularly useful for database sessions which must be committed before the client\n    can do more work.\n\n    NOTE: Do not use a response-scoped dependency within a FastAPI background task.\n          Background tasks run after FastAPI sends the response, so a response-scoped\n          dependency will already be closed. Use a normal FastAPI dependency instead.\n\n    Args:\n        dependency: An async callable. FastAPI dependencies may still be used.\n\n    Returns:\n        A wrapped `dependency` which will push the `dependency` context manager onto\n        a stack when called.\n    \"\"\"\n    signature = inspect.signature(dependency)\n\n    async def wrapper(*args, request: Request, **kwargs):\n        # Replicate FastAPI behavior of auto-creating a context manager\n        if inspect.isasyncgenfunction(dependency):\n            context_manager = asynccontextmanager(dependency)\n        else:\n            context_manager = dependency\n\n        # Ensure request is provided if requested\n        if \"request\" in signature.parameters:\n            kwargs[\"request\"] = request\n\n        # Enter the route handler provided stack that is closed before responding,\n        # return the value yielded by the wrapped dependency\n        return await request.state.response_scoped_stack.enter_async_context(\n            context_manager(*args, **kwargs)\n        )\n\n    # Ensure that the signature includes `request: Request` to ensure that FastAPI will\n    # inject the request as a dependency; maintain the old signature so those depends\n    # work\n    request_parameter = inspect.signature(wrapper).parameters[\"request\"]\n    functools.update_wrapper(wrapper, dependency)\n\n    if \"request\" not in signature.parameters:\n        new_parameters = signature.parameters.copy()\n        new_parameters[\"request\"] = request_parameter\n        wrapper.__signature__ = signature.replace(\n            parameters=tuple(new_parameters.values())\n        )\n\n    return wrapper\n
    "},{"location":"cloud/","title":"Welcome to Prefect Cloud","text":"

    Prefect Cloud is a hosted workflow application framework that provides all the capabilities of Prefect server plus additional features, such as:

    • automations, events, and webhooks so you can create event-driven workflows
    • workspaces, RBAC, SSO, audit logs and related user management tools for collaboration
    • push work pools for running flows on serverless infrastructure without a worker
    • error summaries powered by Marvin AI to help you resolve errors faster

    Getting Started with Prefect Cloud

    Ready to jump right in and start running with Prefect Cloud? See the Quickstart and follow the instructions on the Cloud tabs to write and deploy your first Prefect Cloud-monitored flow run.

    Prefect Cloud includes all the features in the open-source Prefect server plus the following:

    Prefect Cloud features

    • User accounts \u2014 personal accounts for working in Prefect Cloud.
    • Workspaces \u2014 isolated environments to organize your flows, deployments, and flow runs.
    • Automations \u2014 configure triggers, actions, and notifications in response to real-time monitoring events.
    • Email notifications \u2014 send email alerts from Prefect's server based on automation triggers.
    • Service accounts \u2014 configure API access for running workers or executing flow runs on remote infrastructure.
    • Custom role-based access controls (RBAC) \u2014 assign users granular permissions to perform certain activities within an account or a workspace.
    • Single Sign-on (SSO) \u2014 authentication using your identity provider.
    • Audit Log \u2014 a record of user activities to monitor security and compliance.
    • Collaboration \u2014 invite other people to your account.
    • Error summaries \u2014 (enabled by Marvin AI) distill the error logs of Failed and Crashed flow runs into actionable information.
    • Push work pools \u2014 run flows on your serverless infrastructure without running a worker.
    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#user-accounts","title":"User accounts","text":"

    When you sign up for Prefect Cloud, an account and a user profile are automatically provisioned for you.

    Your profile is the place where you'll manage settings related to yourself as a user, including:

    • Profile, including profile handle and image
    • API keys
    • Preferences, including timezone and color mode

    As an account Admin, you will also have access to account settings from the Account Settings page, such as:

    • Members
    • Workspaces
    • Roles

    As an account Admin you can create a workspace and invite other individuals to your workspace.

    Upgrading from a Prefect Cloud Free tier plan to a Pro or Enterprise tier plan enables additional functionality for adding workspaces, managing teams, and running higher volume workloads.

    Workspace Admins have the ability to use single sign-on (SSO), set role-based access controls (RBAC), view Audit Logs, and configure service accounts.

    Enterprise add custom roles, object-level access control lists, teams, and Directory Sync/SCIM provisioning for SSO.

    Prefect Cloud plans for teams of every size

    See the Prefect Cloud plans for details on Pro and Enterprise account tiers.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#workspaces","title":"Workspaces","text":"

    A workspace is an isolated environment within Prefect Cloud for your flows, deployments, and block configuration. See the Workspaces documentation for more information about configuring and using workspaces.

    Each workspace keeps track of its own:

    • Flow runs and task runs executed in an environment that is syncing with the workspace
    • Flows associated with flow runs and deployments observed by the Prefect Cloud API
    • Deployments
    • Work pools
    • Blocks and storage
    • Automations

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#events","title":"Events","text":"

    Prefect Cloud allows you to see your events. Events provide information about the state of your workflows, and can be used to trigger automations.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#automations","title":"Automations","text":"

    Prefect Cloud automations provide additional notification capabilities beyond those in a self-hosted open-source Prefect server. Automations also enable you to create event-driven workflows.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#incidents","title":"Incidents","text":"

    Prefect Cloud's incidents help teams identify, rectify, and document issues in mission-critical workflows. Incidents are formal declarations of disruptions to a workspace. With automations), activity in that workspace can be paused when an incident is created and resumed when it is resolved.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#error-summaries","title":"Error summaries","text":"

    Prefect Cloud error summaries, enabled by Marvin AI, distill the error logs of Failed and Crashed flow runs into actionable information. To enable this feature and others powered by Marvin AI, visit the Settings page for your account.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#service-accounts","title":"Service accounts","text":"

    Service accounts enable you to create Prefect Cloud API keys that are not associated with a user account. Service accounts are typically used to configure API access for running workers or executing flow runs on remote infrastructure. See the service accounts documentation for more information about creating and managing service accounts.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#roles-and-custom-permissions","title":"Roles and custom permissions","text":"

    Role-based access controls (RBAC) enable you to assign users a role with permissions to perform certain activities within an account or a workspace. See the role-based access controls (RBAC) documentation for more information about managing user roles in a Prefect Cloud account.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#single-sign-on-sso","title":"Single Sign-on (SSO)","text":"

    Prefect Cloud's Pro and Enterprise plans offer single sign-on (SSO) authentication integration with your team\u2019s identity provider. SSO integration can bet set up with identity providers that support OIDC and SAML. Directory Sync and SCIM provisioning is also available with Enterprise plans.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#audit-log","title":"Audit log","text":"

    Prefect Cloud's Pro and Enterprise plans offer Audit Logs for compliance and security. Audit logs provide a chronological record of activities performed by users in an account.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#prefect-cloud-rest-api","title":"Prefect Cloud REST API","text":"

    The Prefect REST API is used for communicating data from Prefect clients to Prefect Cloud or a local Prefect server for orchestration and monitoring. This API is mainly consumed by Prefect clients like the Prefect Python Client or the Prefect UI.

    Prefect Cloud REST API interactive documentation

    Prefect Cloud REST API documentation is available at https://app.prefect.cloud/api/docs.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#start-using-prefect-cloud","title":"Start using Prefect Cloud","text":"

    To create an account or sign in with an existing Prefect Cloud account, go to https://app.prefect.cloud/.

    Then follow the steps in the UI to deploy your first Prefect Cloud-monitored flow run. For more details, see the Prefect Quickstart and follow the instructions on the Cloud tabs.

    Need help?

    Get your questions answered by a Prefect Product Advocate! Book a Meeting

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/cloud-quickstart/","title":"Getting Started with Prefect Cloud","text":"

    Get started with Prefect Cloud in just a few steps:

    1. Sign in or register a Prefect Cloud account.
    2. Create a workspace for your account.
    3. Install Prefect in your local environment.
    4. Log into Prefect Cloud from a local terminal session.
    5. Run a flow locally and view flow run execution in Prefect Cloud.
    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#sign-in-or-register","title":"Sign in or register","text":"

    To sign in with an existing account or register an account, go to https://app.prefect.cloud/.

    You can create an account with any of the following:

    • Google account
    • Microsoft account
    • GitHub account
    • Email

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#create-a-workspace","title":"Create a workspace","text":"

    A workspace is an isolated environment within Prefect Cloud for your flows and deployments. You can use workspaces to organize or compartmentalize your workflows.

    When you register a new account, you'll be prompted to provide a name and description for your workspace.

    Note that the Owner setting applies only to users who are members of Prefect Cloud accounts and have permission to create workspaces within account.

    Select Create to create the workspace. If you change your mind, select Edit from the options menu to modify the workspace details or to delete it.

    The Workspace Settings page for your new workspace displays the commands that enable you to install Prefect and log into Prefect Cloud in a local execution environment.

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#install-prefect","title":"Install Prefect","text":"

    Configure a local execution environment to use Prefect Cloud as the API server for flow runs. In other words, \"log in\" to Prefect Cloud from a local environment where you want to run a flow.

    Open a new terminal session.

    Install Prefect in the environment in which you want to execute flow runs.

    pip install -U prefect\n

    Installation requirements

    Prefect requires Python 3.8 or later. If you have any questions about Prefect installations requirements or dependencies in your preferred development environment, check out the Installation documentation.

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#log-into-prefect-cloud-from-a-terminal","title":"Log into Prefect Cloud from a terminal","text":"

    Use the prefect cloud login Prefect CLI command to log into Prefect Cloud from your environment.

    prefect cloud login\n

    The prefect cloud login command, used on its own, provides an interactive login experience. Using this command, you may log in with either an API key or through a browser.

    ? How would you like to authenticate? [Use arrows to move; enter to select]\n> Log in with a webb browser                                                \n  Paste an API key                                                         \nOpening browser...\nWaiting for response...\nAuthenticated with Prefect Cloud! Using workspace 'jeffdc/prod'.\n

    If you choose to log in via the browser, Prefect opens a new tab in your default browser and enables you to log in and authenticate the session.

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#run-a-flow-with-prefect-cloud","title":"Run a flow with Prefect Cloud","text":"

    You're all set to run a flow locally, orchestrated with Prefect Cloud.

    In your local environment, where you configured the previous steps, create a file named quickstart_flow.py with the following contents:

    from prefect import flow\n\n@flow(log_prints=True)\ndef quickstart_flow():\n    print(\"Local quickstart flow is running!\")\n\nif __name__ == \"__main__\":\n    quickstart_flow()\n

    Now run quickstart_flow.py. You'll see log messages like this in your terminal, indicating that the flow is running correctly:

    17:18:09.863 | INFO    | prefect.engine - Created flow run 'fragrant-quetzal' for flow 'quickstart-flow'\n17:18:09.864 | INFO    | Flow run 'fragrant-quetzal' - View at https://app.prefect.cloud/account/my_workspace_id/workspace/my_flow_id/flow-runs/flow-run/my_flow_run_id\n17:18:10.010 | INFO    | Flow run 'fragrant-quetzal' - Local quickstart flow is running!\n17:18:10.144 | INFO    | Flow run 'fragrant-quetzal' - Finished in state Completed()\n

    Go to the Flow Runs pages in your workspace in Prefect Cloud. You'll see the flow run results right there in Prefect Cloud!

    Prefect Cloud automatically tracks any flow runs in a local execution environment logged into Prefect Cloud.

    Select the name of the flow run to see details about this run.

    Congratulations! You successfully ran a local flow and, because you're logged into Prefect Cloud, the local flow run results were captured by Prefect Cloud.

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#next-steps","title":"Next steps","text":"

    If you're new to Prefect, learn more about writing and running flows in the Prefect Flows First Steps tutorial. If you're already familiar with flows, try creating a deployment and triggering flow runs with Prefect Cloud by following the Deployments tutorial.

    Want to learn more about the features available in Prefect Cloud? Start with the Prefect Cloud Overview.

    If you ran into any issues getting your first flow run with Prefect Cloud working, please join our community to ask questions or provide feedback:

    Prefect's Slack Community is helpful, friendly, and fast growing - come say hi!

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/connecting/","title":"Connecting & Troubleshooting Prefect Cloud","text":"

    To create flow runs in a local or remote execution environment and use either Prefect Cloud or a Prefect server as the backend API server, you need to

    • Configure the execution environment with the location of the API.
    • Authenticate with the API, either by logging in or providing a valid API key (Prefect Cloud only).
    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#log-into-prefect-cloud-from-a-terminal","title":"Log into Prefect Cloud from a terminal","text":"

    Configure a local execution environment to use Prefect Cloud as the API server for flow runs. In other words, \"log in\" to Prefect Cloud from a local environment where you want to run a flow.

    1. Open a new terminal session.
    2. Install Prefect in the environment in which you want to execute flow runs.
    $ pip install -U prefect\n
    1. Use the prefect cloud login Prefect CLI command to log into Prefect Cloud from your environment.
    $ prefect cloud login\n

    The prefect cloud login command, used on its own, provides an interactive login experience. Using this command, you can log in with either an API key or through a browser.

    $ prefect cloud login\n? How would you like to authenticate? [Use arrows to move; enter to select]\n> Log in with a web browser\n    Paste an API key\nPaste your authentication key:\n? Which workspace would you like to use? [Use arrows to move; enter to select]\n> prefect/terry-prefect-workspace\n    g-gadflow/g-workspace\nAuthenticated with Prefect Cloud! Using workspace 'prefect/terry-prefect-workspace'.\n

    You can also log in by providing a Prefect Cloud API key that you create.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#change-workspaces","title":"Change workspaces","text":"

    If you need to change which workspace you're syncing with, use the prefect cloud workspace set Prefect CLI command while logged in, passing the account handle and workspace name.

    $ prefect cloud workspace set --workspace \"prefect/my-workspace\"\n

    If no workspace is provided, you will be prompted to select one.

    Workspace Settings also shows you the prefect cloud workspace set Prefect CLI command you can use to sync a local execution environment with a given workspace.

    You may also use the prefect cloud login command with the --workspace or -w option to set the current workspace.

    $ prefect cloud login --workspace \"prefect/my-workspace\"\n
    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#manually-configure-prefect-api-settings","title":"Manually configure Prefect API settings","text":"

    You can also manually configure the PREFECT_API_URL setting to specify the Prefect Cloud API.

    For Prefect Cloud, you can configure the PREFECT_API_URL and PREFECT_API_KEY settings to authenticate with Prefect Cloud by using an account ID, workspace ID, and API key.

    $ prefect config set PREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]\"\n$ prefect config set PREFECT_API_KEY=\"[API-KEY]\"\n

    When you're in a Prefect Cloud workspace, you can copy the PREFECT_API_URL value directly from the page URL.

    In this example, we configured PREFECT_API_URL and PREFECT_API_KEY in the default profile. You can use prefect profile CLI commands to create settings profiles for different configurations. For example, you could have a \"cloud\" profile configured to use the Prefect Cloud API URL and API key, and another \"local\" profile for local development using a local Prefect API server started with prefect server start. See Settings for details.

    Environment variables

    You can also set PREFECT_API_URL and PREFECT_API_KEY as you would any other environment variable. See Overriding defaults with environment variables for more information.

    See the Flow orchestration with Prefect tutorial for examples.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#install-requirements-in-execution-environments","title":"Install requirements in execution environments","text":"

    In local and remote execution environments \u2014 such as VMs and containers \u2014 you must make sure any flow requirements or dependencies have been installed before creating a flow run.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#troubleshooting-prefect-cloud","title":"Troubleshooting Prefect Cloud","text":"

    This section provides tips that may be helpful if you run into problems using Prefect Cloud.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#prefect-cloud-and-proxies","title":"Prefect Cloud and proxies","text":"

    Proxies intermediate network requests between a server and a client.

    To communicate with Prefect Cloud, the Prefect client library makes HTTPS requests. These requests are made using the httpx Python library. httpx respects accepted proxy environment variables, so the Prefect client is able to communicate through proxies.

    To enable communication via proxies, simply set the HTTPS_PROXY and SSL_CERT_FILE environment variables as appropriate in your execution environment and things should \u201cjust work.\u201d

    See the Using Prefect Cloud with proxies topic in Prefect Discourse for examples of proxy configuration.

    URLs that should be whitelisted for outbound-communication in a secure environment include the UI, the API, Authentication, and the current OCSP server:

    • app.prefect.cloud
    • api.prefect.cloud
    • auth.workos.com
    • api.github.com
    • github.com
    • ocsp.pki.goog/s/gts1d4/OxYEb8XcYmo
    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#prefect-cloud-access-via-api","title":"Prefect Cloud access via API","text":"

    If the Prefect Cloud API key, environment variable settings, or account login for your execution environment are not configured correctly, you may experience errors or unexpected flow run results when using Prefect CLI commands, running flows, or observing flow run results in Prefect Cloud.

    Use the prefect config view CLI command to make sure your execution environment is correctly configured to access Prefect Cloud.

    $ prefect config view\nPREFECT_PROFILE='cloud'\nPREFECT_API_KEY='pnu_xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx' (from profile)\nPREFECT_API_URL='https://api.prefect.cloud/api/accounts/...' (from profile)\n

    Make sure PREFECT_API_URL is configured to use https://api.prefect.cloud/api/....

    Make sure PREFECT_API_KEY is configured to use a valid API key.

    You can use the prefect cloud workspace ls CLI command to view or set the active workspace.

    $ prefect cloud workspace ls\n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503   Available Workspaces: \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502   g-gadflow/g-workspace \u2502\n\u2502    * prefect/workinonit \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n    * active workspace\n

    You can also check that the account and workspace IDs specified in the URL for PREFECT_API_URL match those shown in the URL bar for your Prefect Cloud workspace.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#prefect-cloud-login-errors","title":"Prefect Cloud login errors","text":"

    If you're having difficulty logging in to Prefect Cloud, the following troubleshooting steps may resolve the issue, or will provide more information when sharing your case to the support channel.

    • Are you logging into Prefect Cloud 2? Prefect Cloud 1 and Prefect Cloud 2 use separate accounts. Make sure to use the right Prefect Cloud 2 URL: https://app.prefect.cloud/
    • Do you already have a Prefect Cloud account? If you\u2019re having difficulty accepting an invitation, try creating an account first using the email associated with the invitation, then accept the invitation.
    • Are you using a single sign-on (SSO) provider, social authentication (Google, Microsoft, or GitHub) or just using an emailed link?

    Other tips to help with login difficulties:

    • Hard refresh your browser with Cmd+Shift+R.
    • Try in a different browser. We actively test against the following browsers:
    • Chrome
    • Edge
    • Firefox
    • Safari
    • Clear recent browser history/cookies

    None of this worked?

    Email us at help@prefect.io and provide answers to the questions above in your email to make it faster to troubleshoot and unblock you. Make sure you add the email address with which you were trying to log in, your Prefect Cloud account name, and, if applicable, the organization to which it belongs.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/incidents/","title":"Incidents","text":"","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#overview","title":"Overview","text":"

    Incidents in Prefect Cloud is an advanced feature designed to optimize the management of workflow disruptions. It serves as a proactive tool for data-driven teams, helping them identify, rectify, and document issues in mission-critical workflows. This system enhances operational efficiency by automating the incident management process and providing a centralized platform for collaboration and compliance.

    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#what-are-incidents","title":"What are incidents?","text":"

    Incidents are formal declarations of disruptions to a workspace. With automations, activity in that workspace can be paused when an incident is created and resumed when it is resolved.

    Incidents vary in nature and severity, ranging from minor glitches to critical system failures. Prefect Cloud now enables users to effectively and automatically track and manage these incidents, ensuring minimal impact on operational continuity.

    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#why-use-incident-management","title":"Why use incident management?","text":"
    1. Automated detection and reporting: Incidents can be automatically identified based on specific triggers or manually reported by team members, facilitating prompt response.

    2. Collaborative problem-solving: The platform fosters collaboration, allowing team members to share insights, discuss resolutions, and track contributions.

    3. Comprehensive impact assessment: Users gain insights into the incident's influence on workflows, helping in prioritizing response efforts.

    4. Compliance with incident management processes: Detailed documentation and reporting features support compliance with incident management systems.

    5. Enhanced operational transparency: The system provides a transparent view of both ongoing and resolved incidents, promoting accountability and continuous improvement.

    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#how-to-use-incident-management-in-prefect-cloud","title":"How to use incident management in Prefect Cloud","text":"","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#creating-an-incident","title":"Creating an incident","text":"

    There are several ways to create an incident:

    1. From the Incidents page:

      • Click on the + button.
      • Fill in required fields and attach any Prefect resources related to your incident.
    2. From a flow run, work pool, or block:

      • Initiate an incident directly from a failed flow run, automatically linking it as a resource, by clicking on the menu button and selecting \"Declare an incident\".
    3. Via an automation:

      • Set up incident creation as an automated response to selected triggers.
    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#incident-automations","title":"Incident automations","text":"

    Automations can be used for triggering an incident and for selecting actions to take when an incident is triggered. For example, a work pool status change could trigger the declaration of an incident, or a critical level incident could trigger a notification action.

    To automatically take action when an incident is declared, set up a custom trigger that listens for declaration events.

    {\n  \"match\": {\n    \"prefect.resource.id\": \"prefect-cloud.incident.*\"\n  },\n  \"expect\": [\n    \"prefect-cloud.incident.declared\"\n  ],\n  \"posture\": \"Reactive\",\n  \"threshold\": 1,\n  \"within\": 0\n}\n

    Building custom triggers

    To get started with incident automations, you only need to specify two fields in your trigger:

    • match: The resource emitting your event of interest. You can match on specific resource IDs, use wildcards to match on all resources of a given type, and even match on other resource attributes, like prefect.resource.name.

    • expect: The event type to listen for. For example, you could listen for any (or all) of the following event types:

      • prefect-cloud.incident.declared
      • prefect-cloud.incident.resolved
      • prefect-cloud.incident.updated.severity

    See Event Triggers for more information on custom triggers, and check out your Event Feed to see the event types emitted by your incidents and other resources (i.e. events that you can react to).

    When an incident is declared, any actions you configure such as pausing work pools or sending notifications, will execute immediately.

    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#managing-an-incident","title":"Managing an incident","text":"
    • Monitor active incidents: View real-time status, severity, and impact.
    • Adjust incident details: Update status, severity, and other relevant information.
    • Collaborate: Add comments and insights; these will display with user identifiers and timestamps.
    • Impact assessment: Evaluate how the incident affects ongoing and future workflows.
    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#resolving-and-documenting-incidents","title":"Resolving and documenting incidents","text":"
    • Resolution: Update the incident status to reflect resolution steps taken.
    • Documentation: Ensure all actions, comments, and changes are logged for future reference.
    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#incident-reporting","title":"Incident reporting","text":"
    • Generate a detailed timeline of the incident: actions taken, updates to severity and resolution - suitable for compliance and retrospective analysis.
    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/rate-limits/","title":"API Rate Limits & Retention Periods","text":"

    API rate limits restrict the number of requests that a single client can make in a given time period. They ensure Prefect Cloud's stability, so that when you make an API call, you always get a response.

    Prefect Cloud rate limits are subject to change

    The following rate limits are in effect currently, but are subject to change. Contact Prefect support at help@prefect.io if you have questions about current rate limits.

    Prefect Cloud enforces the following rate limits:

    • Flow and task creation rate limits
    • Log service rate limits
    ","tags":["API","Prefect Cloud","rate limits"],"boost":2},{"location":"cloud/rate-limits/#flow-flow-run-and-task-run-rate-limits","title":"Flow, flow run, and task run rate limits","text":"

    Prefect Cloud limits the flow_runs, task_runs, and flows endpoints and their subroutes at the following levels:

    • 400 per minute for personal accounts
    • 2,000 per minute for Pro accounts

    The Prefect Cloud API will return a 429 response with an appropriate Retry-After header if these limits are triggered.

    ","tags":["API","Prefect Cloud","rate limits"],"boost":2},{"location":"cloud/rate-limits/#log-service-rate-limits","title":"Log service rate limits","text":"

    Prefect Cloud limits the number of logs accepted:

    • 700 logs per minute for personal accounts
    • 10,000 logs per minute for Pro accounts

    The Prefect Cloud API will return a 429 response if these limits are triggered.

    ","tags":["API","Prefect Cloud","rate limits"],"boost":2},{"location":"cloud/rate-limits/#flow-run-retention","title":"Flow run retention","text":"

    Prefect Cloud feature

    The Flow Run Retention Policy setting is only applicable in Prefect Cloud.

    Flow runs in Prefect Cloud are retained according to the Flow Run Retention Policy set by your account tier. The policy setting applies to all workspaces owned by the account.

    The flow run retention policy represents the number of days each flow run is available in the Prefect Cloud UI, and via the Prefect CLI and API after it ends. Once a flow run reaches a terminal state (detailed in the chart here), it will be retained until the end of the flow run retention period.

    Flow Run Retention Policy keys on terminal state

    Note that, because Flow Run Retention Policy keys on terminal state, if two flows start at the same time, but reach a terminal state at different times, they will be removed at different times according to when they each reached their respective terminal states.

    This retention policy applies to all details about a flow run, including its task runs. Subflow runs follow the retention policy independently from their parent flow runs, and are removed based on the time each subflow run reaches a terminal state.

    If you or your organization have needs that require a tailored retention period, contact the Prefect Sales team.

    ","tags":["API","Prefect Cloud","rate limits"],"boost":2},{"location":"cloud/workspaces/","title":"Workspaces","text":"

    A workspace is a discrete environment within Prefect Cloud for your workflows and blocks. Workspaces are available to Prefect Cloud accounts only.

    Workspaces can be used to organize and compartmentalize your workflows. For example, you can use separate workspaces to isolate dev, staging, and prod environments, or to provide separation between different teams.

    When you first log into Prefect Cloud, you will be prompted to create your own initial workspace. After creating your workspace, you'll be able to view flow runs, flows, deployments, and other workspace-specific features in the Prefect Cloud UI.

    Select a workspace name in the navigation menu to see all workspaces you can access.

    Your list of available workspaces may include:

    • Your own account's workspace.
    • Workspaces in an account to which you've been invited and have been given access as an Admin or Member.

    Workspace-specific features

    Each workspace keeps track of its own:

    • Flow runs and task runs executed in an environment that is syncing with the workspace
    • Flows associated with flow runs or deployments observed by the Prefect Cloud API
    • Deployments
    • Work pools
    • Blocks and Storage
    • Automations

    Your user permissions within workspaces may vary. Account admins can assign roles and permissions at the workspace level.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#create-a-workspace","title":"Create a workspace","text":"

    On the Account Workspaces dropdown or the Workspaces page select the + icon to create a new workspace.

    You'll be prompted to configure:

    • The Workspace Owner from the dropdown account menu options.
    • The Workspace Name must be unique within the account.
    • An optional description for the workspace.

    Select Create to create the new workspace. The number of available workspaces varies by Prefect Cloud plan. See Pricing if you need additional workspaces or users.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#workspace-settings","title":"Workspace settings","text":"

    Within a workspace, select Settings -> General to view or edit workspace details.

    On this page you can edit workspace details or delete the workspace.

    Deleting a workspace

    Deleting a workspace deletes all deployments, flow run history, work pools, and notifications configured in workspace.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#workspace-access","title":"Workspace access","text":"

    Within a Prefect Cloud Pro or Enterprise tier account, Workspace Owners can invite other people to be members and provision service accounts to a workspace. In addition to giving the user access to the workspace, a Workspace Owner assigns a workspace role to the user. The role specifies the scope of permissions for the user within the workspace.

    As a Workspace Owner, select Workspaces -> Sharing to manage members and service accounts for the workspace.

    If you've previously invited individuals to your account or provisioned service accounts, you'll see them listed here.

    To invite someone to an account, select the Members + icon. You can select from a list of existing account members.

    Select a Role for the user. This will be the initial role for the user within the workspace. A workspace Owner can change this role at any time.

    Select Send to initiate the invitation.

    To add a service account to a workspace, select the Service Accounts + icon. You can select from a list of configured service accounts. Select a Workspace Role for the service account. This will be the initial role for the service account within the workspace. A workspace Owner can change this role at any time. Select Share to finalize adding the service account.

    To remove a workspace member or service account, select Remove from the menu on the right side of the user or service account information on this page.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#workspace-transfer","title":"Workspace transfer","text":"

    Workspace transfer enables you to move an existing workspace from one account to another.

    Workspace transfer retains existing workspace configuration and flow run history, including blocks, deployments, notifications, work pools, and logs.

    Workspace transfer permissions

    Workspace transfer must be initiated or approved by a user with admin privileges for the workspace to be transferred.

    To initiate a workspace transfer between personal accounts, contact support@prefect.io.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#transfer-a-workspace","title":"Transfer a workspace","text":"

    To transfer a workspace, select Settings -> General within the workspace. Then, from the three dot menu in the upper right of the page, select Transfer.

    The Transfer Workspace page shows the workspace to be transferred on the left. Select the target account for the workspace on the right.

    Workspace transfer impact on accounts

    Workspace transfer may impact resource usage and costs for source and target accounts.

    When you transfer a workspace, users, API keys, and service accounts may lose access to the workspace. Audit log will no longer track activity on the workspace. Flow runs ending outside of the destination account\u2019s flow run retention period will be removed. You may also need to update Prefect CLI profiles and execution environment settings to access the workspace's new location.

    You may also incur new charges in the target account to accommodate the transferred workspace.

    The Transfer Workspace page outlines the impacts of transferring the selected workspace to the selected target. Please review these notes carefully before selecting Transfer to transfer the workspace.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/users/","title":"User accounts","text":"

    Sign up for a Prefect Cloud account at app.prefect.cloud.

    An individual user can be invited to become a member of other accounts.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#user-settings","title":"User settings","text":"

    Users can access their personal settings in the profile menu, including:

    • Profile: View and editing basic information, such as name.
    • API keys: Create and view API keys for connecting to Prefect Cloud from the CLI or other environments.
    • Preferences: Manage settings, such as color mode and default time zone.
    • Feature previews: Enable or disable feature previews.
    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#account-roles","title":"Account roles","text":"

    Users who are part of an account can hold the role of Admin or Member. Admins can invite other users to join the account and manage the account's workspaces and teams.

    Admins on Pro and Enterprise tier Prefect Cloud accounts can grant members of the account roles in a workspace, such as Runner or Viewer. Custom roles are available on Enterprise tier accounts.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#api-keys","title":"API keys","text":"

    API keys enable you to authenticate an environment to work with Prefect Cloud.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#service-accounts","title":"Service accounts","text":"

    Service accounts enable you to create a Prefect Cloud API key that is not associated with a user account.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#single-sign-on-sso","title":"Single sign-on (SSO)","text":"

    Pro and Enterprise plans offer single sign-on (SSO) integration with your team\u2019s identity provider. Enterprise tier accounts provide additional options with directory sync and SCIM provisioning.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#audit-log","title":"Audit log","text":"

    Audit logs provide a chronological record of activities performed by Prefect Cloud users who are members of an account.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#object-level-access-control-lists-acls","title":"Object-level access control lists (ACLs)","text":"

    Prefect Cloud's Enterprise plan offers object-level access control lists to restrict access to specific users and service accounts within a workspace.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#teams","title":"Teams","text":"

    Users of Enterprise tier Prefect Cloud accounts can be added to Teams to simplify access control governance.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/api-keys/","title":"Manage Prefect Cloud API Keys","text":"

    API keys enable you to authenticate a local environment to work with Prefect Cloud.

    If you run prefect cloud login from your CLI, you'll have the choice to authenticate through your browser or by pasting an API key.

    If you choose to authenticate through your browser, you'll be directed to an authorization page. After you grant approval to connect, you'll be redirected to the CLI and the API key will be saved to your local Prefect profile.

    If you choose to authenticate by pasting an API key, you'll need to create an API key in the Prefect Cloud UI first.

    ","tags":["Prefect Cloud","API keys","configuration"],"boost":2},{"location":"cloud/users/api-keys/#create-an-api-key","title":"Create an API key","text":"

    To create an API key, select the account icon at the bottom-left corner of the UI.

    Select API Keys. The page displays a list of previously generated keys and lets you create new API keys or delete keys.

    Select the + button to create a new API key. Provide a name for the key and an expiration date.

    Note that API keys cannot be revealed again in the UI after you generate them, so copy the key to a secure location.

    ","tags":["Prefect Cloud","API keys","configuration"],"boost":2},{"location":"cloud/users/api-keys/#log-into-prefect-cloud-with-an-api-key","title":"Log into Prefect Cloud with an API Key","text":"
    prefect cloud login -k '<my-api-key>'\n
    ","tags":["Prefect Cloud","API keys","configuration"],"boost":2},{"location":"cloud/users/api-keys/#service-account-api-keys","title":"Service account API keys","text":"

    Service accounts are a feature of Prefect Cloud Pro and Enterprise tier plans that enable you to create a Prefect Cloud API key that is not associated with a user account.

    Service accounts are typically used to configure API access for running workers or executing flow runs on remote infrastructure. Events and logs for flow runs in those environments are then associated with the service account rather than a user, and API access may be managed or revoked by configuring or removing the service account without disrupting user access.

    See the service accounts documentation for more information about creating and managing service accounts in Prefect Cloud.

    ","tags":["Prefect Cloud","API keys","configuration"],"boost":2},{"location":"cloud/users/audit-log/","title":"Audit Log","text":"

    Prefect Cloud's Pro and Enterprise plans offer enhanced compliance and transparency tools with Audit Log. Audit logs provide a chronological record of activities performed by members in your account, allowing you to monitor detailed Prefect Cloud actions for security and compliance purposes.

    Audit logs enable you to identify who took what action, when, and using what resources within your Prefect Cloud account. In conjunction with appropriate tools and procedures, audit logs can assist in detecting potential security violations and investigating application errors.

    Audit logs can be used to identify changes in:

    • Access to workspaces
    • User login activity
    • User API key creation and removal
    • Workspace creation and removal
    • Account member invitations and removal
    • Service account creation, API key rotation, and removal
    • Billing payment method for self-serve pricing tiers

    See the Prefect Cloud plan information to learn more about options for supporting audit logs.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","audit logs","compliance"],"boost":2},{"location":"cloud/users/audit-log/#viewing-audit-logs","title":"Viewing audit logs","text":"

    From your Pro or Enterprise account settings page, select the Audit Log page to view audit logs.

    Pro and Enterprise account tier admins can view audit logs for:

    • Account-level events in Prefect Cloud, such as:
    • Member invites
    • Changing a member\u2019s role
    • Member login and logout of Prefect Cloud
    • Creating or deleting a service account
    • Workspace-level events in Prefect Cloud, such as:
    • Adding a member to a workspace
    • Changing a member\u2019s workspace role
    • Creating or deleting a workspace

    Admins can filter audit logs on multiple dimensions to restrict the results they see by workspace, user, or event type. Available audit log events are displayed in the Events drop-down menu.

    Audit logs may also be filtered by date range. Audit log retention period varies by Prefect Cloud plan.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","audit logs","compliance"],"boost":2},{"location":"cloud/users/object-access-control-lists/","title":"Object Access Control Lists","text":"

    Prefect Cloud's Enterprise plan offers object-level access control lists to restrict access to specific users and service accounts within a workspace. ACLs are supported for blocks and deployments.

    Organization Admins and Workspace Owners can configure access control lists by navigating to an object and clicking manage access. When an ACL is added, all users and service accounts with access to an object via their workspace role will lose access if not explicitly added to the ACL.

    ACLs and visibility

    Objects not governed by access control lists such as flow runs, flows, and artifacts will be visible to a user within a workspace even if an associated block or deployment has been restricted for that user.

    See the Prefect Cloud plans to learn more about options for supporting object-level access control.

    ","tags":["UI","Permissions","Access","Prefect Cloud","enterprise","teams","workspaces","organizations","audit logs","compliance"],"boost":2},{"location":"cloud/users/roles/","title":"User and Service Account Roles","text":"

    Prefect Cloud's Pro and Enterprise tiers allow you to set team member access to the appropriate level within specific workspaces.

    Role-based access controls (RBAC) enable you to assign users granular permissions to perform certain activities.

    To give users access to functionality beyond the scope of Prefect\u2019s built-in workspace roles, Enterprise account Admins can create custom roles for users.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#built-in-roles","title":"Built-in roles","text":"

    Roles give users abilities at either the account level or at the individual workspace level.

    • An account-level role defines a user's default permissions within an account.
    • A workspace-level role defines a user's permissions within a specific workspace.

    The following sections outline the abilities of the built-in, Prefect-defined ac and workspace roles.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#account-level-roles","title":"Account-level roles","text":"

    The following built-in roles have permissions across an account in Prefect Cloud.

    Role Abilities Admin \u2022 Set/change all account profile settings allowed to be set/changed by a Prefect user. \u2022 Add and remove account members, and their account roles. \u2022 Create and delete service accounts in the account. \u2022 Create workspaces in the account. \u2022 Implicit workspace owner access on all workspaces in the account. Member \u2022 View account profile settings. \u2022 View workspaces I have access to in the account. \u2022 View account members and their roles. \u2022 View service accounts in the account.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#workspace-level-roles","title":"Workspace-level roles","text":"

    The following built-in roles have permissions within a given workspace in Prefect Cloud.

    Role Abilities Viewer \u2022 View flow runs within a workspace. \u2022 View deployments within a workspace. \u2022 View all work pools within a workspace. \u2022 View all blocks within a workspace. \u2022 View all automations within a workspace. \u2022 View workspace handle and description. Runner All Viewer abilities, plus: \u2022 Run deployments within a workspace. Developer All Runner abilities, plus: \u2022 Run flows within a workspace. \u2022 Delete flow runs within a workspace. \u2022 Create, edit, and delete deployments within a workspace. \u2022 Create, edit, and delete work pools within a workspace. \u2022 Create, edit, and delete all blocks and their secrets within a workspace. \u2022 Create, edit, and delete automations within a workspace. \u2022 View all workspace settings. Owner All Developer abilities, plus: \u2022 Add and remove account members, and set their role within a workspace. \u2022 Set the workspace\u2019s default workspace role for all users in the account. \u2022 Set, view, edit workspace settings. Worker The minimum scopes required for a worker to poll for and submit work.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#custom-workspace-roles","title":"Custom workspace roles","text":"

    The built-in roles will serve the needs of most users, but your team may need to configure custom roles, giving users access to specific permissions within a workspace.

    Custom roles can inherit permissions from a built-in role. This enables tweaks to the role to meet your team\u2019s needs, while ensuring users can still benefit from Prefect\u2019s default workspace role permission curation as new functionality becomes available.

    Custom workspace roles can also be created independent of Prefect\u2019s built-in roles. This option gives workspace admins full control of user access to workspace functionality. However, for non-inherited custom roles, the workspace admin takes on the responsibility for monitoring and setting permissions for new functionality as it is released.

    See Role permissions for details of permissions you may set for custom roles.

    After you create a new role, it become available in the account Members page and the Workspace Sharing page for you to apply to users.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#inherited-roles","title":"Inherited roles","text":"

    A custom role may be configured as an Inherited Role. Using an inherited role allows you to create a custom role using a set of initial permissions associated with a built-in Prefect role. Additional permissions can be added to the custom role. Permissions included in the inherited role cannot be removed.

    Custom roles created using an inherited role will follow Prefect's default workspace role permission curation as new functionality becomes available.

    To configure an inherited role when configuring a custom role, select the Inherit permission from a default role check box, then select the role from which the new role should inherit permissions.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#workspace-role-permissions","title":"Workspace role permissions","text":"

    The following permissions are available for custom roles.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#automations","title":"Automations","text":"Permission Description View automations User can see configured automations within a workspace. Create, edit, and delete automations User can create, edit, and delete automations within a workspace. Includes permissions of View automations.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#blocks","title":"Blocks","text":"Permission Description View blocks User can see configured blocks within a workspace. View secret block data User can see configured blocks and their secrets within a workspace. Includes permissions of\u00a0View blocks. Create, edit, and delete blocks User can create, edit, and delete blocks within a workspace. Includes permissions of View blocks and View secret block data.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#deployments","title":"Deployments","text":"Permission Description View deployments User can see configured deployments within a workspace. Run deployments User can run deployments within a workspace. This does not give a user permission to execute the flow associated with the deployment. This only gives a user (via their key) the ability to run a deployment \u2014 another user/key must actually execute that flow, such as a service account with an appropriate role. Includes permissions of View deployments. Create and edit deployments User can create and edit deployments within a workspace. Includes permissions of View deployments and Run deployments. Delete deployments User can delete deployments within a workspace. Includes permissions of View deployments, Run deployments, and Create and edit deployments.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#flows","title":"Flows","text":"Permission Description View flows and flow runs User can see flows and flow runs within a workspace. Create, update, and delete saved search filters User can create, update, and delete saved flow run search filters configured within a workspace. Includes permissions of View flows and flow runs. Create, update, and run flows User can create, update, and run flows within a workspace. Includes permissions of View flows and flow runs. Delete flows User can delete flows within a workspace. Includes permissions of View flows and flow runs and Create, update, and run flows.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#notifications","title":"Notifications","text":"Permission Description View notification policies User can see notification policies configured within a workspace. Create and edit notification policies User can create and edit notification policies configured within a workspace. Includes permissions of View notification policies. Delete notification policies User can delete notification policies configured within a workspace. Includes permissions of View notification policies and Create and edit notification policies.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#task-run-concurrency","title":"Task run concurrency","text":"Permission Description View concurrency limits User can see configured task run concurrency limits within a workspace. Create, edit, and delete concurrency limits User can create, edit, and delete task run concurrency limits within a workspace. Includes permissions of View concurrency limits.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#work-pools","title":"Work pools","text":"Permission Description View work pools User can see work pools configured within a workspace. Create, edit, and pause work pools User can create, edit, and pause work pools configured within a workspace. Includes permissions of View work pools. Delete work pools User can delete work pools configured within a workspace. Includes permissions of View work pools and Create, edit, and pause work pools.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#workspace-management","title":"Workspace management","text":"Permission Description View information about workspace service accounts User can see service accounts configured within a workspace. View information about workspace users User can see user accounts for users invited to the workspace. View workspace settings User can see settings configured within a workspace. Edit workspace settings User can edit settings for a workspace. Includes permissions of View workspace settings. Delete the workspace User can delete a workspace. Includes permissions of View workspace settings and Edit workspace settings.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/service-accounts/","title":"Service Accounts","text":"

    Service accounts enable you to create a Prefect Cloud API key that is not associated with a user account. Service accounts are typically used to configure API access for running workers or executing deployment flow runs on remote infrastructure.

    Service accounts are non-user accounts that have the following features:

    • Prefect Cloud API keys
    • Roles and permissions

    Using service account credentials, you can configure an execution environment to interact with your Prefect Cloud workspaces without a user having to manually log in from that environment. Service accounts may be created, added to workspaces, have their roles changed, or deleted without affecting other user accounts.

    Select Service Accounts to view, create, or edit service accounts.

    Service accounts are created at the account level, but individual workspaces may be shared with the service account. See workspace sharing for more information.

    Service account credentials

    When you create a service account, Prefect Cloud creates a new API key for the account and provides the API configuration command for the execution environment. Save these to a safe location for future use. If the access credentials are lost or compromised, you should regenerate the credentials from the service account page.

    Service account roles

    Service accounts are created at the account level, and can then be added to workspaces within the account.

    A service account may only be a Member of an account. It can never be an account Admin. You may apply any valid workspace-level role to a service account.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/users/service-accounts/#create-a-service-account","title":"Create a service account","text":"

    Within your account, on the Service Accounts page, select the + icon to create a new service account. You'll be prompted to configure:

    • The service account name. This name must be unique within your account.
    • An expiration date, or the Never Expire option.

    Service account roles

    A service account may only be a Member of an account. You may apply any valid workspace-level role to a service account when it is added to a workspace.

    Select Create to create the new service account.

    Note that API keys cannot be revealed again in the UI after you generate them, so copy the key to a secure location.

    You can change the API key and expiration for a service account by rotating the API key. Select Rotate API Key from the menu on the left side of the service account's information on this page.

    To delete a service account, select Remove from the menu on the left side of the service account's information.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/users/sso/","title":"Single Sign-on (SSO)","text":"

    Prefect Cloud's Pro and Enterprise plans offer single sign-on (SSO) integration with your team\u2019s identity provider. SSO integration can bet set up with any identity provider that supports:

    • OIDC
    • SAML 2.0

    When using SSO, Prefect Cloud won't store passwords for any accounts managed by your identity provider. Members of your Prefect Cloud account will instead log in and authenticate using your identity provider.

    Once your SSO integration has been set up, non-admins will be required to authenticate through the SSO provider when accessing account resources.

    See the Prefect Cloud plans to learn more about options for supporting more users and workspaces, service accounts, and SSO.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","single sign-on","SSO","authentication"],"boost":2},{"location":"cloud/users/sso/#configuring-sso","title":"Configuring SSO","text":"

    Within your account, select the SSO page to enable SSO for users.

    If you haven't enabled SSO for a domain yet, enter the email domains for which you want to configure SSO in Prefect Cloud and save it.

    Under Enabled Domains, select the domains from the Domains list, then select Generate Link. This step creates a link you can use to configure SSO with your identity provider.

    Using the provided link navigate to the Identity Provider Configuration dashboard and select your identity provider to continue configuration. If your provider isn't listed, you can continue with the SAML or Open ID Connect choices instead.

    Once you complete SSO configuration your users will be required to authenticate via your identity provider when accessing account resources, giving you full control over application access.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","single sign-on","SSO","authentication"],"boost":2},{"location":"cloud/users/sso/#directory-sync","title":"Directory sync","text":"

    Directory sync automatically provisions and de-provisions users for your account.

    Provisioned users are given basic \u201cMember\u201d roles and will have access to any resources that role entails.

    When a user is unassigned from the Prefect Cloud application in your identity provider, they will automatically lose access to Prefect Cloud resources, allowing your IT team to control access to Prefect Cloud without ever signing into the app.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","single sign-on","SSO","authentication"],"boost":2},{"location":"cloud/users/sso/#scim-provisioning","title":"SCIM Provisioning","text":"

    Enterprise accounts have access to SCIM for user provisioning. The SSO tab provides access to enable SCIM provisioning.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","single sign-on","SSO","authentication"],"boost":2},{"location":"cloud/users/teams/","title":"Teams","text":"

    Prefect Cloud's Enterprise plan offers team management to simplify access control governance.

    Account Admins can configure teams and team membership from the account settings menu by clicking Teams. Teams are composed of users and service accounts. Teams can be added to workspaces or object access control lists just like users and service accounts.

    If SCIM is enabled on your account, the set of teams and the users within them is governed by your IDP. Prefect Cloud service accounts, which are not governed by your IDP, can be still be added to your existing set of teams.

    See the Prefect Cloud plans to learn more about options for supporting teams.

    ","tags":["UI","Permissions","Access","Prefect Cloud","enterprise","teams","workspaces","organizations","audit logs","compliance"],"boost":2},{"location":"community/","title":"Community","text":"

    There are many ways to get involved with the Prefect community

    • Join over 26,000 engineers in the Prefect Slack community
    • Get help in Prefect Discourse - the community-driven knowledge base
    • Give Prefect a \u2b50\ufe0f on GitHub
    • Contribute to Prefect's open source libraries
    • Become a Prefect Ambassador by joining Club 42
    ","tags":["community","Slack","Discourse"],"boost":2},{"location":"concepts/","title":"Explore Prefect concepts","text":"Concept Description Flows A Prefect workflow, defined as a Python function. Tasks Discrete units of work in a Prefect workflow. Deployments A server-side concept that encapsulates flow metadata, allowing it to be scheduled and triggered via API. Work Pools & Workers Use Prefect to dynamically provision and configure infrastructure in your execution environment. Schedules Tell the Prefect API how to create new flow runs for you automatically on a specified cadence. Results The data returned by a flow or a task. Artifacts Formatted outputs rendered in the Prefect UI, such as markdown, tables, or links. Incidents Identify, rectify and document issues in mission-critical workflows. States Rich objects that capture the status of a particular task run or flow run. Blocks Prefect primitives that enable the storage of configuration and provide a UI interface. Task Runners Configure how tasks are run - concurrently, in parallel, or in a distributed environment. Automations Configure actions that Prefect executes automatically based on trigger conditions. Block and Agent-Based Deployments Description Block-based Deployments Create deployments that rely on blocks. Infrastructure Blocks that specify infrastructure for flow runs created by a deployment. Storage Lets you configure how flow code for deployments is persisted and retrieved. Agents Like untyped workers.

    Many features specific to Prefect Cloud are in their own subheading.

    ","tags":["concepts","features","overview"],"boost":2},{"location":"concepts/agents/","title":"Agents","text":"

    Workers are recommended

    Agents are part of the block-based deployment model. Work Pools and Workers simplify the specification of a flow's infrastructure and runtime environment. If you have existing agents, you can upgrade from agents to workers to significantly enhance the experience of deploying flows.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#agent-overview","title":"Agent overview","text":"

    Agent processes are lightweight polling services that get scheduled work from a work pool and deploy the corresponding flow runs.

    Agents poll for work every 15 seconds by default. This interval is configurable in your profile settings with the PREFECT_AGENT_QUERY_INTERVAL setting.

    It is possible for multiple agent processes to be started for a single work pool. Each agent process sends a unique ID to the server to help disambiguate themselves and let users know how many agents are active.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#agent-options","title":"Agent options","text":"

    Agents are configured to pull work from one or more work pool queues. If the agent references a work queue that doesn't exist, it will be created automatically.

    Configuration parameters you can specify when starting an agent include:

    Option Description --api The API URL for the Prefect server. Default is the value of PREFECT_API_URL. --hide-welcome Do not display the startup ASCII art for the agent process. --limit Maximum number of flow runs to start simultaneously. [default: None] --match, -m Dynamically matches work queue names with the specified prefix for the agent to pull from,for example dev- will match all work queues with a name that starts with dev-. [default: None] --pool, -p A work pool name for the agent to pull from. [default: None] --prefetch-seconds The amount of time before a flow run's scheduled start time to begin submission. Default is the value of PREFECT_AGENT_PREFETCH_SECONDS. --run-once Only run agent polling once. By default, the agent runs forever. [default: no-run-once] --work-queue, -q One or more work queue names for the agent to pull from. [default: None]

    You must start an agent within an environment that can access or create the infrastructure needed to execute flow runs. Your agent will deploy flow runs to the infrastructure specified by the deployment.

    Prefect must be installed in execution environments

    Prefect must be installed in any environment in which you intend to run the agent or execute a flow run.

    PREFECT_API_URL and PREFECT_API_KEY settings for agents

    PREFECT_API_URL must be set for the environment in which your agent is running or specified when starting the agent with the --api flag. You must also have a user or service account with the Worker role, which can be configured by setting the PREFECT_API_KEY.

    If you want an agent to communicate with Prefect Cloud or a Prefect server from a remote execution environment such as a VM or Docker container, you must configure PREFECT_API_URL in that environment.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#starting-an-agent","title":"Starting an agent","text":"

    Use the prefect agent start CLI command to start an agent. You must pass at least one work pool name or match string that the agent will poll for work. If the work pool does not exist, it will be created.

    prefect agent start -p [work pool name]\n

    For example:

    Starting agent with ephemeral API...\n\u00a0 ___ ___ ___ ___ ___ ___ _____ \u00a0 \u00a0 _ \u00a0 ___ ___ _\u00a0 _ _____\n\u00a0| _ \\ _ \\ __| __| __/ __|_ \u00a0 _| \u00a0 /_\\ / __| __| \\| |_ \u00a0 _|\n\u00a0|\u00a0 _/ \u00a0 / _|| _|| _| (__\u00a0 | |\u00a0 \u00a0 / _ \\ (_ | _|| .` | | |\n\u00a0|_| |_|_\\___|_| |___\\___| |_| \u00a0 /_/ \\_\\___|___|_|\\_| |_|\n\nAgent started! Looking for work from work pool 'my-pool'...\n

    By default, the agent polls the API specified by the PREFECT_API_URL environment variable. To configure the agent to poll from a different server location, use the --api flag, specifying the URL of the server.

    In addition, agents can match multiple queues in a work pool by providing a --match string instead of specifying all of the queues. The agent will poll every queue with a name that starts with the given string. New queues matching this prefix will be found by the agent without needing to restart it.

    For example:

    prefect agent start --match \"foo-\"\n

    This example will poll every work queue that starts with \"foo-\".

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#configuring-prefetch","title":"Configuring prefetch","text":"

    By default, the agent begins submission of flow runs a short time (10 seconds) before they are scheduled to run. This allows time for the infrastructure to be created, so the flow run can start on time. In some cases, infrastructure will take longer than this to actually start the flow run. In these cases, the prefetch can be increased using the --prefetch-seconds option or the PREFECT_AGENT_PREFETCH_SECONDS setting.

    Submission can begin an arbitrary amount of time before the flow run is scheduled to start. If this value is larger than the amount of time it takes for the infrastructure to start, the flow run will wait until its scheduled start time. This allows flow runs to start exactly on time.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#troubleshooting","title":"Troubleshooting","text":"","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#agent-crash-or-keyboard-interrupt","title":"Agent crash or keyboard interrupt","text":"

    If the agent process is ended abruptly, you can sometimes have left over flows that were destined for the agent whose process was ended. In the UI, these will show up as pending. You will need to delete these flows in order for the restarted agent to begin processing the work queue again. Take note of the flows you deleted, you might need to set them to run manually.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/artifacts/","title":"Artifacts","text":"

    Artifacts are persisted outputs such as tables, Markdown, or links. They are stored on Prefect Cloud or a Prefect server instance and rendered in the Prefect UI. Artifacts make it easy to track and monitor the objects that your flows produce and update over time.

    Published artifacts may be associated with a particular task run or flow run. Artifacts can also be created outside of any flow run context.

    Whether you're publishing links, Markdown, or tables, artifacts provide a powerful and flexible way to showcase data within your workflows.

    With artifacts, you can easily manage and share information with your team, providing valuable insights and context.

    Common use cases for artifacts include:

    • Debugging: By publishing data that you care about in the UI, you can easily see when and where your results were written. If an artifact doesn't look the way you expect, you can find out which flow run last updated it, and you can click through a link in the artifact to a storage location (such as an S3 bucket).
    • Data quality checks: Artifacts can be used to publish data quality checks from in-progress tasks. This can help ensure that data quality is maintained throughout the pipeline. During long-running tasks such as ML model training, you might use artifacts to publish performance graphs. This can help you visualize how well your models are performing and make adjustments as needed. You can also track the versions of these artifacts over time, making it easier to identify changes in your data.
    • Documentation: Artifacts can be used to publish documentation and sample data to help you keep track of your work and share information with your colleagues. For instance, artifacts allow you to add a description to let your colleagues know why this piece of data is important.
    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#creating-artifacts","title":"Creating artifacts","text":"

    Creating artifacts allows you to publish data from task and flow runs or outside of a flow run context. Currently, you can render three artifact types: links, Markdown, and tables.

    Artifacts render individually

    Please note that every artifact created within a task will be displayed as an individual artifact in the Prefect UI. This means that each call to create_link_artifact() or create_markdown_artifact() generates a distinct artifact.

    Unlike the print() command, where you can concatenate multiple calls to include additional items in a report, within a task, these commands must be used multiple times if necessary.

    To create artifacts like reports or summaries using create_markdown_artifact(), compile your message string separately and then pass it to create_markdown_artifact() to create the complete artifact.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#creating-link-artifacts","title":"Creating link artifacts","text":"

    To create a link artifact, use the create_link_artifact() function. To create multiple versions of the same artifact and/or view them on the Artifacts page of the Prefect UI, provide a key argument to the create_link_artifact() function to track an artifact's history over time. Without a key, the artifact will only be visible in the Artifacts tab of the associated flow run or task run.\"

    from prefect import flow, task\nfrom prefect.artifacts import create_link_artifact\n\n@task\ndef my_first_task():\n        create_link_artifact(\n            key=\"irregular-data\",\n            link=\"https://nyc3.digitaloceanspaces.com/my-bucket-name/highly_variable_data.csv\",\n            description=\"## Highly variable data\",\n        )\n\n@task\ndef my_second_task():\n        create_link_artifact(\n            key=\"irregular-data\",\n            link=\"https://nyc3.digitaloceanspaces.com/my-bucket-name/low_pred_data.csv\",\n            description=\"# Low prediction accuracy\",\n        )\n\n@flow\ndef my_flow():\n    my_first_task()\n    my_second_task()\n\nif __name__ == \"__main__\":\n    my_flow()\n

    Tip

    You can specify multiple artifacts with the same key to more easily track something very specific that you care about, such as irregularities in your data pipeline.

    After running the above flows, you can find your new artifacts in the Artifacts page of the UI. Click into the \"irregular-data\" artifact and see all versions of it, along with custom descriptions and links to the relevant data.

    Here, you'll also be able to view information about your artifact such as its associated flow run or task run id, previous and future versions of the artifact (multiple artifacts can have the same key in order to show lineage), the data you've stored (in this case a Markdown-rendered link), an optional Markdown description, and when the artifact was created or updated.

    To make the links more readable for you and your collaborators, you can pass in a link_text argument for your link artifacts:

    from prefect import flow\nfrom prefect.artifacts import create_link_artifact\n\n@flow\ndef my_flow():\n    create_link_artifact(\n        key=\"my-important-link\",\n        link=\"https://www.prefect.io/\",\n        link_text=\"Prefect\",\n    )\n\nif __name__ == \"__main__\":\n    my_flow()\n

    In the above example, the create_link_artifact method is used within a flow to create a link artifact with a key of my-important-link. The link parameter is used to specify the external resource to be linked to, and link_text is used to specify the text to be displayed for the link. An optional description could also be added for context.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#creating-markdown-artifacts","title":"Creating Markdown artifacts","text":"

    To create a Markdown artifact, you can use the create_markdown_artifact() function. To create multiple versions of the same artifact and/or view them on the Artifacts page of the Prefect UI, provide a key argument to the create_markdown_artifact() function to track an artifact's history over time. Without a key, the artifact will only be visible in the Artifacts tab of the associated flow run or task run.\"

    Don't indent Markdown

    Markdown in mult-line strings must be unindented to be interpreted correctly.

    from prefect import flow, task\nfrom prefect.artifacts import create_markdown_artifact\n\n@task\ndef markdown_task():\n    na_revenue = 500000\n    markdown_report = f\"\"\"# Sales Report\n\n## Summary\n\nIn the past quarter, our company saw a significant increase in sales, with a total revenue of $1,000,000. \nThis represents a 20% increase over the same period last year.\n\n## Sales by Region\n\n| Region        | Revenue |\n|:--------------|-------:|\n| North America | ${na_revenue:,} |\n| Europe        | $250,000 |\n| Asia          | $150,000 |\n| South America | $75,000 |\n| Africa        | $25,000 |\n\n## Top Products\n\n1. Product A - $300,000 in revenue\n2. Product B - $200,000 in revenue\n3. Product C - $150,000 in revenue\n\n## Conclusion\n\nOverall, these results are very encouraging and demonstrate the success of our sales team in increasing revenue \nacross all regions. However, we still have room for improvement and should focus on further increasing sales in \nthe coming quarter.\n\"\"\"\n    create_markdown_artifact(\n        key=\"gtm-report\",\n        markdown=markdown_report,\n        description=\"Quarterly Sales Report\",\n    )\n\n@flow()\ndef my_flow():\n    markdown_task()\n\n\nif __name__ == \"__main__\":\n    my_flow()\n

    After running the above flow, you should see your \"gtm-report\" artifact in the Artifacts page of the UI.

    As with all artifacts, you'll be able to view the associated flow run or task run id, previous and future versions of the artifact, your rendered Markdown data, and your optional Markdown description.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#create-table-artifacts","title":"Create table artifacts","text":"

    You can create a table artifact by calling create_table_artifact(). To create multiple versions of the same artifact and/or view them on the Artifacts page of the Prefect UI, provide a key argument to the create_table_artifact() function to track an artifact's history over time. Without a key, the artifact will only be visible in the artifacts tab of the associated flow run or task run.\"

    Note

    The create_table_artifact() function accepts a table argument, which can be provided as either a list of lists, a list of dictionaries, or a dictionary of lists.

    from prefect.artifacts import create_table_artifact\n\ndef my_fn():\n    highest_churn_possibility = [\n       {'customer_id':'12345', 'name': 'John Smith', 'churn_probability': 0.85 }, \n       {'customer_id':'56789', 'name': 'Jane Jones', 'churn_probability': 0.65 } \n    ]\n\n    create_table_artifact(\n        key=\"personalized-reachout\",\n        table=highest_churn_possibility,\n        description= \"# Marvin, please reach out to these customers today!\"\n    )\n\nif __name__ == \"__main__\":\n    my_fn()\n

    As you can see, you don't need to create an artifact in a flow run context. You can create one anywhere in a Python script and see it in the Prefect UI.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#managing-artifacts","title":"Managing artifacts","text":"","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#reading-artifacts","title":"Reading artifacts","text":"

    In the Prefect UI, you can view all of the latest versions of your artifacts and click into a specific artifact to see its lineage over time. Additionally, you can inspect all versions of an artifact with a given key by running:

    prefect artifact inspect <my_key>\n

    or view all artifacts by running:

    prefect artifact ls\n

    You can also use the Prefect REST API to programmatically filter your results.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#deleting-artifacts","title":"Deleting artifacts","text":"

    You can delete an artifact directly using the CLI to delete specific artifacts with a given key or id:

    prefect artifact delete <my_key>\n
    prefect artifact delete --id <my_id>\n

    Alternatively, you can delete artifacts using the Prefect REST API.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#artifacts-api","title":"Artifacts API","text":"

    Prefect provides the Prefect REST API to allow you to create, read, and delete artifacts programmatically. With the Artifacts API, you can automate the creation and management of artifacts as part of your workflow.

    For example, to read the five most recently created Markdown, table, and link artifacts, you can run the following:

    import requests\n\nPREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/abc/workspaces/xyz\"\nPREFECT_API_KEY=\"pnu_ghijk\"\ndata = {\n    \"sort\": \"CREATED_DESC\",\n    \"limit\": 5,\n    \"artifacts\": {\n        \"key\": {\n            \"exists_\": True\n        }\n    }\n}\n\nheaders = {\"Authorization\": f\"Bearer {PREFECT_API_KEY}\"}\nendpoint = f\"{PREFECT_API_URL}/artifacts/filter\"\n\nresponse = requests.post(endpoint, headers=headers, json=data)\nassert response.status_code == 200\nfor artifact in response.json():\n    print(artifact)\n

    If you don't specify a key or that a key must exist, you will also return results (which are a type of key-less artifact).

    See the rest of the Prefect REST API documentation on artifacts for more information!

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/automations/","title":"Automations","text":"

    Automations in Prefect Cloud enable you to configure actions that Prefect executes automatically based on trigger conditions related to your flows and work pools.

    Using triggers and actions you can automatically kick off flow runs, pause deployments, or send custom notifications in response to real-time monitoring events.

    Automations are only available in Prefect Cloud

    Notifications in an open-source Prefect server provide a subset of the notification message-sending features available in Automations.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#automations-overview","title":"Automations overview","text":"

    The Automations page provides an overview of all configured automations for your workspace.

    Selecting the toggle next to an automation pauses execution of the automation.

    The button next to the toggle provides commands to copy the automation ID, edit the automation, or delete the automation.

    Select the name of an automation to view Details about it and relevant Events.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#create-an-automation","title":"Create an automation","text":"

    On the Automations page, select the + icon to create a new automation. You'll be prompted to configure:

    • A trigger condition that causes the automation to execute.
    • One or more actions carried out by the automation.
    • Details about the automation, such as a name and description.
    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#triggers","title":"Triggers","text":"

    Triggers specify the conditions under which your action should be performed. Triggers can be of several types, including triggers based on:

    • Flow run state change
    • Note - Flow Run Tags currently are only evaluated with OR criteria
    • Work pool status
    • Custom event triggers

    Automations API

    The automations API enables further programmatic customization of trigger and action policies based on arbitrary events.

    Importantly, triggers can be configured not only in reaction to events, but also proactively: to trigger in the absence of an event you expect to see.

    For example, in the case of flow run state change triggers, you might expect production flows to finish in no longer than thirty minutes. But transient infrastructure or network issues could cause your flow to get \u201cstuck\u201d in a running state. A trigger could kick off an action if the flow stays in a running state for more than 30 minutes. This action could be on the flow itself, such as canceling or restarting it, or it could take the form of a notification so someone can take manual remediation steps.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#actions","title":"Actions","text":"

    Actions specify what your automation does when its trigger criteria are met. Current action types include:

    • Cancel a flow run
    • Pause a flow run
    • Run a deployment
    • Pause or resume a deployment schedule
    • Pause or resume a work queue
    • Pause or resume an automation
    • Send a notification
    • Call a webhook

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#selected-and-inferred-action-targets","title":"Selected and inferred action targets","text":"

    Some actions require you to either select the target of the action, or specify that the target of the action should be inferred.

    Selected targets are simple, and useful for when you know exactly what object your action should act on \u2014 for example, the case of a cleanup flow you want to run or a specific notification you\u2019d like to send.

    Inferred targets are deduced from the trigger itself.

    For example, if a trigger fires on a flow run that is stuck in a running state, and the action is to cancel an inferred flow run, the flow run to cancel is inferred as the stuck run that caused the trigger to fire.

    Similarly, if a trigger fires on a work queue event and the corresponding action is to pause an inferred work queue, the inferred work queue is the one that emitted the event.

    Prefect tries to infer the relevant event whenever possible, but sometimes one does not exist.

    Specify a name and, optionally, a description for the automation.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#custom-triggers","title":"Custom triggers","text":"

    Custom triggers allow advanced configuration of the conditions on which an automation executes its actions. Several custom trigger fields accept values that end with trailing wildcards, like \"prefect.flow-run.*\".

    The schema that defines a trigger is as follows:

    Name Type Supports trailing wildcards Description match object Labels for resources which this Automation will match. match_related object Labels for related resources which this Automation will match. after array of strings Event(s), one of which must have first been seen to start this automation. expect array of strings The event(s) this automation is expecting to see. If empty, this automation will evaluate any matched event. for_each array of strings Evaluate the Automation separately for each distinct value of these labels on the resource. By default, labels refer to the primary resource of the triggering event. You may also refer to labels from related resources by specifying related:<role>:<label>. This will use the value of that label for the first related resource in that role. posture string enum N/A The posture of this Automation, either Reactive or Proactive. Reactive automations respond to the presence of the expected events, while Proactive automations respond to the absence of those expected events. threshold integer N/A The number of events required for this Automation to trigger (for Reactive automations), or the number of events expected (for Proactive automations) within number N/A The time period over which the events must occur. For Reactive triggers, this may be as low as 0 seconds, but must be at least 10 seconds for Proactive triggers","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#resource-matching","title":"Resource matching","text":"

    match and match_related control which events a trigger considers for evaluation by filtering on the contents of their resource and related fields, respectively. Each label added to a match filter is ANDed with the other labels, and can accept a single value or a list of multiple values that are ORed together.

    Consider the resource and related fields on the following prefect.flow-run.Completed event, truncated for the sake of example. Its primary resource is a flow run, and since that flow run was started via a deployment, it is related to both its flow and its deployment:

    \"resource\": {\n  \"prefect.resource.id\": \"prefect.flow-run.925eacce-7fe5-4753-8f02-77f1511543db\",\n  \"prefect.resource.name\": \"cute-kittiwake\"\n}\n\"related\": [\n  {\n    \"prefect.resource.id\": \"prefect.flow.cb6126db-d528-402f-b439-96637187a8ca\",\n    \"prefect.resource.role\": \"flow\",\n    \"prefect.resource.name\": \"hello\"\n  },\n  {\n    \"prefect.resource.id\": \"prefect.deployment.37ca4a08-e2d9-4628-a310-cc15a323378e\",\n    \"prefect.resource.role\": \"deployment\",\n    \"prefect.resource.name\": \"example\"\n  }\n]\n

    There are a number of valid ways to select the above event for evaluation, and the approach depends on the purpose of the automation.

    The following configuration will filter for any events whose primary resource is a flow run, and that flow run has a name starting with cute- or radical-.

    \"match\": {\n  \"prefect.resource.id\": \"prefect.flow-run.*\",\n  \"prefect.resource.name\": [\"cute-*\", \"radical-*\"]\n},\n\"match_related\": {},\n...\n

    This configuration, on the other hand, will filter for any events for which this specific deployment is a related resource.

    \"match\": {},\n\"match_related\": {\n  \"prefect.resource.id\": \"prefect.deployment.37ca4a08-e2d9-4628-a310-cc15a323378e\"\n},\n...\n

    Both of the above approaches will select the example prefect.flow-run.Completed event, but will permit additional, possibly undesired events through the filter as well. match and match_related can be combined for more restrictive filtering:

    \"match\": {\n  \"prefect.resource.id\": \"prefect.flow-run.*\",\n  \"prefect.resource.name\": [\"cute-*\", \"radical-*\"]\n},\n\"match_related\": {\n  \"prefect.resource.id\": \"prefect.deployment.37ca4a08-e2d9-4628-a310-cc15a323378e\"\n},\n...\n

    Now this trigger will filter only for events whose primary resource is a flow run started by a specific deployment, and that flow run has a name starting with cute- or radical-.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#expected-events","title":"Expected events","text":"

    Once an event has passed through the match filters, it must be decided if this event should be counted toward the trigger's threshold. Whether that is the case is determined by the event names present in expect.

    This configuration informs the trigger to evaluate only prefect.flow-run.Completed events that have passed the match filters.

    \"expect\": [\n  \"prefect.flow-run.Completed\"\n],\n...\n

    threshold decides the quantity of expected events needed to satisfy the trigger. Increasing the threshold above 1 will also require use of within to define a range of time in which multiple events are seen. The following configuration will expect two occurrences of prefect.flow-run.Completed within 60 seconds.

    \"expect\": [\n  \"prefect.flow-run.Completed\"\n],\n\"threshold\": 2,\n\"within\": 60,\n...\n

    after can be used to handle scenarios that require more complex event reactivity.

    Take, for example, this flow which emits an event indicating the table it operates on is missing or empty:

    from prefect import flow\nfrom prefect.events import emit_event\nfrom db import Table\n\n\n@flow\ndef transform(table_name: str):\n  table = Table(table_name)\n\n  if not table.exists():\n    emit_event(\n        event=\"table-missing\",\n        resource={\"prefect.resource.id\": \"etl-events.transform\"}\n    )\n  elif table.is_empty():\n    emit_event(\n        event=\"table-empty\",\n        resource={\"prefect.resource.id\": \"etl-events.transform\"}\n    )\n  else:\n    # transform data\n

    The following configuration uses after to prevent this automation from firing unless either a table-missing or a table-empty event has occurred before a flow run of this deployment completes.

    Tip

    Note how match and match_related are used to ensure the trigger only evaluates events that are relevant to its purpose.

    \"match\": {\n  \"prefect.resource.id\": [\n    \"prefect.flow-run.*\",\n    \"etl-events.transform\"\n  ]\n},\n\"match_related\": {\n  \"prefect.resource.id\": \"prefect.deployment.37ca4a08-e2d9-4628-a310-cc15a323378e\"\n}\n\"after\": [\n  \"table-missing\",\n  \"table-empty\"\n]\n\"expect\": [\n  \"prefect.flow-run.Completed\"\n],\n...\n
    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#evaluation-strategy","title":"Evaluation strategy","text":"

    All of the previous examples were designed around a reactive posture - that is, count up events toward the threshold until it is met, then execute actions. To respond to the absence of events, use a proactive posture. A proactive trigger will fire when its threshold has not been met by the end of the window of time defined by within. Proactive triggers must have a within of at least 10 seconds.

    The following trigger will fire if a prefect.flow-run.Completed event is not seen within 60 seconds after a prefect.flow-run.Running event is seen.

    {\n  \"match\": {\n    \"prefect.resource.id\": \"prefect.flow-run.*\"\n  },\n  \"match_related\": {},\n  \"after\": [\n    \"prefect.flow-run.Running\"\n  ],\n  \"expect\": [\n    \"prefect.flow-run.Completed\"\n  ],\n  \"for_each\": [],\n  \"posture\": \"Proactive\",\n  \"threshold\": 1,\n  \"within\": 60\n}\n
    However, without for_each, a prefect.flow-run.Completed event from a different flow run than the one that started this trigger with its prefect.flow-run.Running event could satisfy the condition. Adding a for_each of prefect.resource.id will cause this trigger to be evaluated separately for each flow run id associated with these events.

    {\n  \"match\": {\n    \"prefect.resource.id\": \"prefect.flow-run.*\"\n  },\n  \"match_related\": {},\n  \"after\": [\n    \"prefect.flow-run.Running\"\n  ],\n  \"expect\": [\n    \"prefect.flow-run.Completed\"\n  ],\n  \"for_each\": [\n    \"prefect.resource.id\"\n  ],\n  \"posture\": \"Proactive\",\n  \"threshold\": 1,\n  \"within\": 60\n}\n
    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#create-an-automation-via-deployment-triggers","title":"Create an automation via deployment triggers","text":"

    To enable the simple configuration of event-driven deployments, Prefect provides deployment triggers - a shorthand for creating automations that are linked to specific deployments to run them based on the presence or absence of events.

    # prefect.yaml\ndeployments:\n  - name: my-deployment\n    entrypoint: path/to/flow.py:decorated_fn\n    work_pool:\n      name: my-process-pool\n    triggers:\n      - enabled: true\n        match:\n          prefect.resource.id: my.external.resource\n        expect:\n          - external.resource.pinged\n        parameters:\n          param_1: \"{{ event }}\"\n

    At deployment time, this will create a linked automation that is triggered by events matching your chosen grammar, which will pass the templatable event as a parameter to the deployment's flow run.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#pass-triggers-to-prefect-deploy","title":"Pass triggers to prefect deploy","text":"

    You can pass one or more --trigger arguments to prefect deploy, which can be either a JSON string or a path to a .yaml or .json file.

    # Pass a trigger as a JSON string\nprefect deploy -n test-deployment \\\n  --trigger '{\n    \"enabled\": true, \n    \"match\": {\n      \"prefect.resource.id\": \"prefect.flow-run.*\"\n    }, \n    \"expect\": [\"prefect.flow-run.Completed\"]\n  }'\n\n# Pass a trigger using a JSON/YAML file\nprefect deploy -n test-deployment --trigger triggers.yaml\nprefect deploy -n test-deployment --trigger my_stuff/triggers.json\n

    For example, a triggers.yaml file could have many triggers defined:

    triggers:\n  - enabled: true\n    match:\n      prefect.resource.id: my.external.resource\n    expect:\n      - external.resource.pinged\n    parameters:\n      param_1: \"{{ event }}\"\n  - enabled: true\n    match:\n      prefect.resource.id: my.other.external.resource\n    expect:\n      - some.other.event\n    parameters:\n      param_1: \"{{ event }}\"\n
    Both of the above triggers would be attached to test-deployment after running prefect deploy.

    Triggers passed to prefect deploy will override any triggers defined in prefect.yaml

    While you can define triggers in prefect.yaml for a given deployment, triggers passed to prefect deploy will take precedence over those defined in prefect.yaml.

    Note that deployment triggers contribute to the total number of automations in your workspace.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#automation-notifications","title":"Automation notifications","text":"

    Notifications enable you to set up automation actions that send a message.

    Automation notifications support sending notifications via any predefined block that is capable of and configured to send a message. That includes, for example:

    • Slack message to a channel
    • Microsoft Teams message to a channel
    • Email to a configured email address

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#templating-notifications-with-jinja","title":"Templating notifications with Jinja","text":"

    The notification body can include templated variables using Jinja syntax. Templated variable enable you to include details relevant to automation trigger, such as a flow or pool name.

    Jinja templated variable syntax wraps the variable name in double curly brackets, like {{ variable }}.

    You can access properties of the underlying flow run objects including:

    • flow_run
    • flow
    • deployment
    • work_queue
    • work_pool

    In addition to its native properties, each object includes an id along with created and updated timestamps.

    The flow_run|ui_url token returns the URL for viewing the flow run in Prefect Cloud.

    Here\u2019s an example for something that would be relevant to a flow run state-based notification:

    Flow run {{ flow_run.name }} entered state {{ flow_run.state.name }}. \n\n    Timestamp: {{ flow_run.state.timestamp }}\n    Flow ID: {{ flow_run.flow_id }}\n    Flow Run ID: {{ flow_run.id }}\n    State message: {{ flow_run.state.message }}\n

    The resulting Slack webhook notification would look something like this:

    You could include flow and deployment properties.

    Flow run {{ flow_run.name }} for flow {{ flow.name }}\nentered state {{ flow_run.state.name }}\nwith message {{ flow_run.state.message }}\n\nFlow tags: {{ flow_run.tags }}\nDeployment name: {{ deployment.name }}\nDeployment version: {{ deployment.version }}\nDeployment parameters: {{ deployment.parameters }}\n

    An automation that reports on work pool status might include notifications using work_pool properties.

    Work pool status alert!\n\nName: {{ work_pool.name }}\nLast polled: {{ work_pool.last_polled }}\n

    In addition to those shortcuts for flows, deployments, and work pools, you have access to the automation and the event that triggered the automation. See the Automations API for additional details.

    Automation: {{ automation.name }}\nDescription: {{ automation.description }}\n\nEvent: {{ event.id }}\nResource:\n{% for label, value in event.resource %}\n{{ label }}: {{ value }}\n{% endfor %}\nRelated Resources:\n{% for related in event.related %}\n    Role: {{ related.role }}\n    {% for label, value in event.resource %}\n    {{ label }}: {{ value }}\n    {% endfor %}\n{% endfor %}\n

    Note that this example also illustrates the ability to use Jinja features such as iterator and for loop control structures when templating notifications.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/blocks/","title":"Blocks","text":"

    Blocks are a primitive within Prefect that enable the storage of configuration and provide an interface for interacting with external systems.

    With blocks, you can securely store credentials for authenticating with services like AWS, GitHub, Slack, and any other system you'd like to orchestrate with Prefect.

    Blocks expose methods that provide pre-built functionality for performing actions against an external system. They can be used to download data from or upload data to an S3 bucket, query data from or write data to a database, or send a message to a Slack channel.

    You may configure blocks through code or via the Prefect Cloud and the Prefect server UI.

    You can access blocks for both configuring flow deployments and directly from within your flow code.

    Prefect provides some built-in block types that you can use right out of the box. Additional blocks are available through Prefect Integrations. To use these blocks you can pip install the package, then register the blocks you want to use with Prefect Cloud or a Prefect server.

    Prefect Cloud and the Prefect server UI display a library of block types available for you to configure blocks that may be used by your flows.

    Blocks and parameters

    Blocks are useful for configuration that needs to be shared across flow runs and between flows.

    For configuration that will change between flow runs, we recommend using parameters.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#prefect-built-in-blocks","title":"Prefect built-in blocks","text":"

    Prefect provides a broad range of commonly used, built-in block types. These block types are available in Prefect Cloud and the Prefect server UI.

    Block Slug Description Azure azure Store data as a file on Azure Datalake and Azure Blob Storage. Date Time date-time A block that represents a datetime. Docker Container docker-container Runs a command in a container. Docker Registry docker-registry Connects to a Docker registry. Requires a Docker Engine to be connectable. GCS gcs Store data as a file on Google Cloud Storage. GitHub github Interact with files stored on public GitHub repositories. JSON json A block that represents JSON. Kubernetes Cluster Config kubernetes-cluster-config Stores configuration for interaction with Kubernetes clusters. Kubernetes Job kubernetes-job Runs a command as a Kubernetes Job. Local File System local-file-system Store data as a file on a local file system. Microsoft Teams Webhook ms-teams-webhook Enables sending notifications via a provided Microsoft Teams webhook. Opsgenie Webhook opsgenie-webhook Enables sending notifications via a provided Opsgenie webhook. Pager Duty Webhook pager-duty-webhook Enables sending notifications via a provided PagerDuty webhook. Process process Run a command in a new process. Remote File System remote-file-system Store data as a file on a remote file system. Supports any remote file system supported by fsspec. S3 s3 Store data as a file on AWS S3. Secret secret A block that represents a secret value. The value stored in this block will be obfuscated when this block is logged or shown in the UI. Slack Webhook slack-webhook Enables sending notifications via a provided Slack webhook. SMB smb Store data as a file on a SMB share. String string A block that represents a string. Twilio SMS twilio-sms Enables sending notifications via Twilio SMS. Webhook webhook Block that enables calling webhooks.","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#blocks-in-prefect-integrations","title":"Blocks in Prefect Integrations","text":"

    Blocks can also be created by anyone and shared with the community. You'll find blocks that are available for consumption in many of the published Prefect Integrations. The following table provides an overview of the blocks available from our most popular Prefect Integrations.

    Integration Block Slug prefect-airbyte Airbyte Connection airbyte-connection prefect-airbyte Airbyte Server airbyte-server prefect-aws AWS Credentials aws-credentials prefect-aws ECS Task ecs-task prefect-aws MinIO Credentials minio-credentials prefect-aws S3 Bucket s3-bucket prefect-azure Azure Blob Storage Credentials azure-blob-storage-credentials prefect-azure Azure Container Instance Credentials azure-container-instance-credentials prefect-azure Azure Container Instance Job azure-container-instance-job prefect-azure Azure Cosmos DB Credentials azure-cosmos-db-credentials prefect-azure AzureML Credentials azureml-credentials prefect-bitbucket BitBucket Credentials bitbucket-credentials prefect-bitbucket BitBucket Repository bitbucket-repository prefect-census Census Credentials census-credentials prefect-census Census Sync census-sync prefect-databricks Databricks Credentials databricks-credentials prefect-dbt dbt CLI BigQuery Target Configs dbt-cli-bigquery-target-configs prefect-dbt dbt CLI Profile dbt-cli-profile prefect-dbt dbt Cloud Credentials dbt-cloud-credentials prefect-dbt dbt CLI Global Configs dbt-cli-global-configs prefect-dbt dbt CLI Postgres Target Configs dbt-cli-postgres-target-configs prefect-dbt dbt CLI Snowflake Target Configs dbt-cli-snowflake-target-configs prefect-dbt dbt CLI Target Configs dbt-cli-target-configs prefect-docker Docker Host docker-host prefect-docker Docker Registry Credentials docker-registry-credentials prefect-email Email Server Credentials email-server-credentials prefect-firebolt Firebolt Credentials firebolt-credentials prefect-firebolt Firebolt Database firebolt-database prefect-gcp BigQuery Warehouse bigquery-warehouse prefect-gcp GCP Cloud Run Job cloud-run-job prefect-gcp GCP Credentials gcp-credentials prefect-gcp GcpSecret gcpsecret prefect-gcp GCS Bucket gcs-bucket prefect-gcp Vertex AI Custom Training Job vertex-ai-custom-training-job prefect-github GitHub Credentials github-credentials prefect-github GitHub Repository github-repository prefect-gitlab GitLab Credentials gitlab-credentials prefect-gitlab GitLab Repository gitlab-repository prefect-hex Hex Credentials hex-credentials prefect-hightouch Hightouch Credentials hightouch-credentials prefect-kubernetes Kubernetes Credentials kubernetes-credentials prefect-monday Monday Credentials monday-credentials prefect-monte-carlo Monte Carlo Credentials monte-carlo-credentials prefect-openai OpenAI Completion Model openai-completion-model prefect-openai OpenAI Image Model openai-image-model prefect-openai OpenAI Credentials openai-credentials prefect-slack Slack Credentials slack-credentials prefect-slack Slack Incoming Webhook slack-incoming-webhook prefect-snowflake Snowflake Connector snowflake-connector prefect-snowflake Snowflake Credentials snowflake-credentials prefect-sqlalchemy Database Credentials database-credentials prefect-sqlalchemy SQLAlchemy Connector sqlalchemy-connector prefect-twitter Twitter Credentials twitter-credentials","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#using-existing-block-types","title":"Using existing block types","text":"

    Blocks are classes that subclass the Block base class. They can be instantiated and used like normal classes.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#instantiating-blocks","title":"Instantiating blocks","text":"

    For example, to instantiate a block that stores a JSON value, use the JSON block:

    from prefect.blocks.system import JSON\n\njson_block = JSON(value={\"the_answer\": 42})\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#saving-blocks","title":"Saving blocks","text":"

    If this JSON value needs to be retrieved later to be used within a flow or task, we can use the .save() method on the block to store the value in a block document on the Prefect database for retrieval later:

    json_block.save(name=\"life-the-universe-everything\")\n

    Utilizing the UI

    Blocks documents can also be created and updated via the Prefect UI.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#loading-blocks","title":"Loading blocks","text":"

    The name given when saving the value stored in the JSON block can be used when retrieving the value during a flow or task run:

    from prefect import flow\nfrom prefect.blocks.system import JSON\n\n@flow\ndef what_is_the_answer():\n    json_block = JSON.load(\"life-the-universe-everything\")\n    print(json_block.value[\"the_answer\"])\n\nwhat_is_the_answer() # 42\n

    Blocks can also be loaded with a unique slug that is a combination of a block type slug and a block document name.

    To load our JSON block document from before, we can run the following:

    from prefect.blocks.core import Block\n\njson_block = Block.load(\"json/life-the-universe-everything\")\nprint(json_block.value[\"the-answer\"]) #42\n

    Sharing Blocks

    Blocks can also be loaded by fellow Workspace Collaborators, available on Prefect Cloud.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#deleting-blocks","title":"Deleting blocks","text":"

    You can delete a block by using the .delete() method on the block:

    from prefect.blocks.core import Block\nBlock.delete(\"json/life-the-universe-everything\")\n

    You can also use the CLI to delete specific blocks with a given slug or id:

    prefect block delete json/life-the-universe-everything\n
    prefect block delete --id <my-id>\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#creating-new-block-types","title":"Creating new block types","text":"

    To create a custom block type, define a class that subclasses Block. The Block base class builds off of Pydantic's BaseModel, so custom blocks can be declared in same manner as a Pydantic model.

    Here's a block that represents a cube and holds information about the length of each edge in inches:

    from prefect.blocks.core import Block\n\nclass Cube(Block):\n    edge_length_inches: float\n

    You can also include methods on a block include useful functionality. Here's the same cube block with methods to calculate the volume and surface area of the cube:

    from prefect.blocks.core import Block\n\nclass Cube(Block):\n    edge_length_inches: float\n\n    def get_volume(self):\n        return self.edge_length_inches**3\n\n    def get_surface_area(self):\n        return 6 * self.edge_length_inches**2\n

    Now the Cube block can be used to store different cube configuration that can later be used in a flow:

    from prefect import flow\n\nrubiks_cube = Cube(edge_length_inches=2.25)\nrubiks_cube.save(\"rubiks-cube\")\n\n@flow\ndef calculate_cube_surface_area(cube_name):\n    cube = Cube.load(cube_name)\n    print(cube.get_surface_area())\n\ncalculate_cube_surface_area(\"rubiks-cube\") # 30.375\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#secret-fields","title":"Secret fields","text":"

    All block values are encrypted before being stored, but if you have values that you would not like visible in the UI or in logs, then you can use the SecretStr field type provided by Pydantic to automatically obfuscate those values. This can be useful for fields that are used to store credentials like passwords and API tokens.

    Here's an example of an AWSCredentials block that uses SecretStr:

    from typing import Optional\n\nfrom prefect.blocks.core import Block\nfrom pydantic import SecretStr\n\nclass AWSCredentials(Block):\n    aws_access_key_id: Optional[str] = None\n    aws_secret_access_key: Optional[SecretStr] = None\n    aws_session_token: Optional[str] = None\n    profile_name: Optional[str] = None\n    region_name: Optional[str] = None\n

    Because aws_secret_access_key has the SecretStr type hint assigned to it, the value of that field will not be exposed if the object is logged:

    aws_credentials_block = AWSCredentials(\n    aws_access_key_id=\"AKIAJKLJKLJKLJKLJKLJK\",\n    aws_secret_access_key=\"secret_access_key\"\n)\n\nprint(aws_credentials_block)\n# aws_access_key_id='AKIAJKLJKLJKLJKLJKLJK' aws_secret_access_key=SecretStr('**********') aws_session_token=None profile_name=None region_name=None\n

    There's also use the SecretDict field type provided by Prefect. This type will allow you to add a dictionary field to your block that will have values at all levels automatically obfuscated in the UI or in logs. This is useful for blocks where typing or structure of secret fields is not known until configuration time.

    Here's an example of a block that uses SecretDict:

    from typing import Dict\n\nfrom prefect.blocks.core import Block\nfrom prefect.blocks.fields import SecretDict\n\n\nclass SystemConfiguration(Block):\n    system_secrets: SecretDict\n    system_variables: Dict\n\n\nsystem_configuration_block = SystemConfiguration(\n    system_secrets={\n        \"password\": \"p@ssw0rd\",\n        \"api_token\": \"token_123456789\",\n        \"private_key\": \"<private key here>\",\n    },\n    system_variables={\n        \"self_destruct_countdown_seconds\": 60,\n        \"self_destruct_countdown_stop_time\": 7,\n    },\n)\n
    system_secrets will be obfuscated when system_configuration_block is displayed, but system_variables will be shown in plain-text:

    print(system_configuration_block)\n# SystemConfiguration(\n#   system_secrets=SecretDict('{'password': '**********', 'api_token': '**********', 'private_key': '**********'}'), \n#   system_variables={'self_destruct_countdown_seconds': 60, 'self_destruct_countdown_stop_time': 7}\n# )\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#blocks-metadata","title":"Blocks metadata","text":"

    The way that a block is displayed can be controlled by metadata fields that can be set on a block subclass.

    Available metadata fields include:

    Property Description _block_type_name Display name of the block in the UI. Defaults to the class name. _block_type_slug Unique slug used to reference the block type in the API. Defaults to a lowercase, dash-delimited version of the block type name. _logo_url URL pointing to an image that should be displayed for the block type in the UI. Default to None. _description Short description of block type. Defaults to docstring, if provided. _code_example Short code snippet shown in UI for how to load/use block type. Default to first example provided in the docstring of the class, if provided.","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#nested-blocks","title":"Nested blocks","text":"

    Block are composable. This means that you can create a block that uses functionality from another block by declaring it as an attribute on the block that you're creating. It also means that configuration can be changed for each block independently, which allows configuration that may change on different time frames to be easily managed and configuration can be shared across multiple use cases.

    To illustrate, here's a an expanded AWSCredentials block that includes the ability to get an authenticated session via the boto3 library:

    from typing import Optional\n\nimport boto3\nfrom prefect.blocks.core import Block\nfrom pydantic import SecretStr\n\nclass AWSCredentials(Block):\n    aws_access_key_id: Optional[str] = None\n    aws_secret_access_key: Optional[SecretStr] = None\n    aws_session_token: Optional[str] = None\n    profile_name: Optional[str] = None\n    region_name: Optional[str] = None\n\n    def get_boto3_session(self):\n        return boto3.Session(\n            aws_access_key_id = self.aws_access_key_id\n            aws_secret_access_key = self.aws_secret_access_key\n            aws_session_token = self.aws_session_token\n            profile_name = self.profile_name\n            region_name = self.region\n        )\n

    The AWSCredentials block can be used within an S3Bucket block to provide authentication when interacting with an S3 bucket:

    import io\n\nclass S3Bucket(Block):\n    bucket_name: str\n    credentials: AWSCredentials\n\n    def read(self, key: str) -> bytes:\n        s3_client = self.credentials.get_boto3_session().client(\"s3\")\n\n        stream = io.BytesIO()\n        s3_client.download_fileobj(Bucket=self.bucket_name, key=key, Fileobj=stream)\n\n        stream.seek(0)\n        output = stream.read()\n\n        return output\n\n    def write(self, key: str, data: bytes) -> None:\n        s3_client = self.credentials.get_boto3_session().client(\"s3\")\n        stream = io.BytesIO(data)\n        s3_client.upload_fileobj(stream, Bucket=self.bucket_name, Key=key)\n

    You can use this S3Bucket block with previously saved AWSCredentials block values in order to interact with the configured S3 bucket:

    my_s3_bucket = S3Bucket(\n    bucket_name=\"my_s3_bucket\",\n    credentials=AWSCredentials.load(\"my_aws_credentials\")\n)\n\nmy_s3_bucket.save(\"my_s3_bucket\")\n

    Saving block values like this links the values of the two blocks so that any changes to the values stored for the AWSCredentials block with the name my_aws_credentials will be seen the next time that block values for the S3Bucket block named my_s3_bucket is loaded.

    Values for nested blocks can also be hard coded by not first saving child blocks:

    my_s3_bucket = S3Bucket(\n    bucket_name=\"my_s3_bucket\",\n    credentials=AWSCredentials(\n        aws_access_key_id=\"AKIAJKLJKLJKLJKLJKLJK\",\n        aws_secret_access_key=\"secret_access_key\"\n    )\n)\n\nmy_s3_bucket.save(\"my_s3_bucket\")\n

    In the above example, the values for AWSCredentials are saved with my_s3_bucket and will not be usable with any other blocks.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#handling-updates-to-custom-block-types","title":"Handling updates to custom Block types","text":"

    Let's say that you now want to add a bucket_folder field to your custom S3Bucket block that represents the default path to read and write objects from (this field exists on our implementation).

    We can add the new field to the class definition:

    class S3Bucket(Block):\n    bucket_name: str\n    credentials: AWSCredentials\n    bucket_folder: str = None\n    ...\n

    Then register the updated block type with either Prefect Cloud or your self-hosted Prefect server.

    If you have any existing blocks of this type that were created before the update and you'd prefer to not re-create them, you can migrate them to the new version of your block type by adding the missing values:

    # Bypass Pydantic validation to allow your local Block class to load the old block version\nmy_s3_bucket_block = S3Bucket.load(\"my-s3-bucket\", validate=False)\n\n# Set the new field to an appropriate value\nmy_s3_bucket_block.bucket_path = \"my-default-bucket-path\"\n\n# Overwrite the old block values and update the expected fields on the block\nmy_s3_bucket_block.save(\"my-s3-bucket\", overwrite=True)\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#registering-blocks-for-use-in-the-prefect-ui","title":"Registering blocks for use in the Prefect UI","text":"

    Blocks can be registered from a Python module available in the current virtual environment with a CLI command like this:

    $ prefect block register --module prefect_aws.credentials\n

    This command is useful for registering all blocks found in the credentials module within Prefect Integrations.

    Or, if a block has been created in a .py file, the block can also be registered with the CLI command:

    $ prefect block register --file my_block.py\n

    The registered block will then be available in the Prefect UI for configuration.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/deployments-block-based/","title":"Block Based Deployments","text":"

    Workers are recommended

    This page is about the block-based deployment model. The Work Pools and Workers based deployment model simplifies the specification of a flow's infrastructure and runtime environment. If you have existing agents, you can upgrade from agents to workers to significantly enhance the experience of deploying flows.

    We encourage you to check out the new deployment experience with guided command line prompts and convenient CI/CD with prefect.yaml files.

    With remote storage blocks, you can package not only your flow code script but also any supporting files, including your custom modules, SQL scripts and any configuration files needed in your project.

    To define how your flow execution environment should be configured, you may either reference pre-configured infrastructure blocks or let Prefect create those automatically for you as anonymous blocks (this happens when you specify the infrastructure type using --infra flag during the build process).

    Work queue affinity improved starting from Prefect 2.0.5

    Until Prefect 2.0.4, tags were used to associate flow runs with work queues. Starting in Prefect 2.0.5, tag-based work queues are deprecated. Instead, work queue names are used to explicitly direct flow runs from deployments into queues.

    Note that backward compatibility is maintained and work queues that use tag-based matching can still be created and will continue to work. However, those work queues are now considered legacy and we encourage you to use the new behavior by specifying work queues explicitly on agents and deployments.

    See Agents & Work Pools for details.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deployments-and-flows","title":"Deployments and flows","text":"

    Each deployment is associated with a single flow, but any given flow can be referenced by multiple deployments.

    Deployments are uniquely identified by the combination of: flow_name/deployment_name.

    graph LR\n    F(\"my_flow\"):::yellow -.-> A(\"Deployment 'daily'\"):::tan --> W(\"my_flow/daily\"):::fgreen\n    F -.-> B(\"Deployment 'weekly'\"):::gold  --> X(\"my_flow/weekly\"):::green\n    F -.-> C(\"Deployment 'ad-hoc'\"):::dgold --> Y(\"my_flow/ad-hoc\"):::dgreen\n    F -.-> D(\"Deployment 'trigger-based'\"):::dgold --> Z(\"my_flow/trigger-based\"):::dgreen\n\n    classDef gold fill:goldenrod,stroke:goldenrod,stroke-width:4px,color:white\n    classDef yellow fill:gold,stroke:gold,stroke-width:4px\n    classDef dgold fill:darkgoldenrod,stroke:darkgoldenrod,stroke-width:4px,color:white\n    classDef tan fill:tan,stroke:tan,stroke-width:4px,color:white\n    classDef fgreen fill:forestgreen,stroke:forestgreen,stroke-width:4px,color:white\n    classDef green fill:green,stroke:green,stroke-width:4px,color:white\n    classDef dgreen fill:darkgreen,stroke:darkgreen,stroke-width:4px,color:white

    This enables you to run a single flow with different parameters, based on multiple schedules and triggers, and in different environments. This also enables you to run different versions of the same flow for testing and production purposes.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deployment-definition","title":"Deployment definition","text":"

    A deployment definition captures the settings for creating a deployment object on the Prefect API. You can create the deployment definition by:

    • Run the prefect deployment build CLI command with deployment options to create a deployment.yaml deployment definition file, then run prefect deployment apply to create a deployment on the API using the settings in deployment.yaml.
    • Define a Deployment Python object, specifying the deployment options as properties of the object, then building and applying the object using methods of Deployment.

    The minimum required information to create a deployment includes:

    • The path and filename of the file containing the flow script.
    • The name of the entrypoint flow function \u2014 this is the flow function that starts the flow and calls and additional tasks or subflows.
    • The name of the deployment.

    You may provide additional settings for the deployment. Any settings you do not explicitly specify are inferred from defaults.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-deployment-on-the-cli","title":"Create a deployment on the CLI","text":"

    To create a deployment on the CLI, there are two steps:

    1. Build the deployment definition file deployment.yaml. This step includes uploading your flow to its configured remote storage location, if one is specified.
    2. Create the deployment on the API.
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#build-the-deployment","title":"Build the deployment","text":"

    To build the deployment definition file deployment.yaml, run the prefect deployment build Prefect CLI command from the folder containing your flow script and any dependencies of the script.

    $ prefect deployment build [OPTIONS] PATH\n

    Path to the flow is specified in the format path-to-script:flow-function-name \u2014 The path and filename of the flow script file, a colon, then the name of the entrypoint flow function.

    For example:

    $ prefect deployment build -n marvin -p default-agent-pool -q test flows/marvin.py:say_hi\n

    When you run this command, Prefect:

    • Creates a marvin_flow-deployment.yaml file for your deployment based on your flow code and options.
    • Uploads your flow files to the configured storage location (local by default).
    • Submit your deployment to the work queue test. The work queue test will be created if it doesn't exist.

    Uploading files may require storage filesystem libraries

    Note that the appropriate filesystem library supporting the storage location must be installed prior to building a deployment with a storage block. For example, the AWS S3 Storage block requires the s3fs library.

    Ignore files or directories from a deployment

    By default, Prefect uploads all files in the current folder to the configured storage location (local by default) when you build a deployment.

    If you want to omit certain files or directories from your deployments, add a .prefectignore file to the root directory. .prefectignore enables users to omit certain files or directories from their deployments.

    Similar to other .ignore files, the syntax supports pattern matching, so an entry of *.pyc will ensure all .pyc files are ignored by the deployment call when uploading to remote storage.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deployment-build-options","title":"Deployment build options","text":"

    You may specify additional options to further customize your deployment.

    Options Description PATH Path, filename, and flow name of the flow definition. (Required) --apply, -a When provided, automatically registers the resulting deployment with the API. --cron TEXT A cron string that will be used to set a CronSchedule on the deployment. For example, --cron \"*/1 * * * *\" to create flow runs from that deployment every minute. --help Display help for available commands and options. --infra-block TEXT, -ib The infrastructure block to use, in block-type/block-name format. --infra, -i The infrastructure type to use. (Default is Process) --interval INTEGER An integer specifying an interval (in seconds) that will be used to set an IntervalSchedule on the deployment. For example, --interval 60 to create flow runs from that deployment every minute. --name TEXT, -n The name of the deployment. --output TEXT, -o Optional location for the YAML manifest generated as a result of the build step. You can version-control that file, but it's not required since the CLI can generate everything you need to define a deployment. --override TEXT One or more optional infrastructure overrides provided as a dot delimited path. For example, specify an environment variable: env.env_key=env_value. For Kubernetes, specify customizations: customizations='[{\"op\": \"add\",\"path\": \"/spec/template/spec/containers/0/resources/limits\", \"value\": {\"memory\": \"8Gi\",\"cpu\": \"4000m\"}}]' (note the string format). --param An optional parameter override, values are parsed as JSON strings. For example, --param question=ultimate --param answer=42. --params An optional parameter override in a JSON string format. For example, --params=\\'{\"question\": \"ultimate\", \"answer\": 42}\\'. --path An optional path to specify a subdirectory of remote storage to upload to, or to point to a subdirectory of a locally stored flow. --pool TEXT, -p The work pool that will handle this deployment's runs. \u2502 --rrule TEXT An RRule that will be used to set an RRuleSchedule on the deployment. For example, --rrule 'FREQ=HOURLY;BYDAY=MO,TU,WE,TH,FR;BYHOUR=9,10,11,12,13,14,15,16,17' to create flow runs from that deployment every hour but only during business hours. --skip-upload When provided, skips uploading this deployment's files to remote storage. --storage-block TEXT, -sb The storage block to use, in block-type/block-name or block-type/block-name/path format. Note that the appropriate library supporting the storage filesystem must be installed. --tag TEXT, -t One or more optional tags to apply to the deployment. --version TEXT, -v An optional version for the deployment. This could be a git commit hash if you use this command from a CI/CD pipeline. --work-queue TEXT, -q The work queue that will handle this deployment's runs. It will be created if it doesn't already exist. Defaults to None. Note that if a work queue is not set, work will not be scheduled.","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#block-identifiers","title":"Block identifiers","text":"

    When specifying a storage block with the -sb or --storage-block flag, you may specify the block by passing its slug. The storage block slug is formatted as block-type/block-name.

    For example, s3/example-block is the slug for an S3 block named example-block.

    In addition, when passing the storage block slug, you may pass just the block slug or the block slug and a path.

    • block-type/block-name indicates just the block, including any path included in the block configuration.
    • block-type/block-name/path indicates a storage path in addition to any path included in the block configuration.

    When specifying an infrastructure block with the -ib or --infra-block flag, you specify the block by passing its slug. The infrastructure block slug is formatted as block-type/block-name.

    Block name Block class name Block type for a slug Azure Azure azure Docker Container DockerContainer docker-container GitHub GitHub github GCS GCS gcs Kubernetes Job KubernetesJob kubernetes-job Process Process process Remote File System RemoteFileSystem remote-file-system S3 S3 s3 SMB SMB smb GitLab Repository GitLabRepository gitlab-repository

    Note that the appropriate library supporting the storage filesystem must be installed prior to building a deployment with a storage block. For example, the AWS S3 Storage block requires the s3fs library. See Storage for more information.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deploymentyaml","title":"deployment.yaml","text":"

    A deployment's YAML file configures additional settings needed to create a deployment on the server.

    A single flow may have multiple deployments created for it, with different schedules, tags, and so on. A single flow definition may have multiple deployment YAML files referencing it, each specifying different settings. The only requirement is that each deployment must have a unique name.

    The default {flow-name}-deployment.yaml filename may be edited as needed with the --output flag to prefect deployment build.

    ###\n### A complete description of a Prefect Deployment for flow 'Cat Facts'\n###\nname: catfact\ndescription: null\nversion: c0fc95308d8137c50d2da51af138aa23\n# The work queue that will handle this deployment's runs\nwork_queue_name: test\nwork_pool_name: null\ntags: []\nparameters: {}\nschedule: null\ninfra_overrides: {}\ninfrastructure:\n  type: process\n  env: {}\n  labels: {}\n  name: null\n  command:\n  - python\n  - -m\n  - prefect.engine\n  stream_output: true\n###\n### DO NOT EDIT BELOW THIS LINE\n###\nflow_name: Cat Facts\nmanifest_path: null\nstorage: null\npath: /Users/terry/test/testflows/catfact\nentrypoint: catfact.py:catfacts_flow\nparameter_openapi_schema:\n  title: Parameters\n  type: object\n  properties:\n    url:\n      title: url\n  required:\n  - url\n  definitions: null\n

    Editing deployment.yaml

    Note the big DO NOT EDIT comment in your deployment's YAML: In practice, anything above this block can be freely edited before running prefect deployment apply to create the deployment on the API.

    We recommend editing most of these fields from the CLI or Prefect UI for convenience.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#parameters-in-deployments","title":"Parameters in deployments","text":"

    You may provide default parameter values in the deployment.yaml configuration, and these parameter values will be used for flow runs based on the deployment.

    To configure default parameter values, add them to the parameters: {} line of deployment.yaml as JSON key-value pairs. The parameter list configured in deployment.yaml must match the parameters expected by the entrypoint flow function.

    parameters: {\"name\": \"Marvin\", \"num\": 42, \"url\": \"https://catfact.ninja/fact\"}\n

    Passing **kwargs as flow parameters

    You may pass **kwargs as a deployment parameter as a \"kwargs\":{} JSON object containing the key-value pairs of any passed keyword arguments.

    parameters: {\"name\": \"Marvin\", \"kwargs\":{\"cattype\":\"tabby\",\"num\": 42}\n

    You can edit default parameters for deployments in the Prefect UI, and you can override default parameter values when creating ad-hoc flow runs via the Prefect UI.

    To edit parameters in the Prefect UI, go the the details page for a deployment, then select Edit from the commands menu. If you change parameter values, the new values are used for all future flow runs based on the deployment.

    To create an ad-hoc flow run with different parameter values, go the the details page for a deployment, select Run, then select Custom. You will be able to provide custom values for any editable deployment fields. Under Parameters, select Custom. Provide the new values, then select Save. Select Run to begin the flow run with custom values.

    If you want the Prefect API to verify the parameter values passed to a flow run against the schema defined by parameter_openapi_schema, set enforce_parameter_schema to true.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-deployment","title":"Create a deployment","text":"

    When you've configured deployment.yaml for a deployment, you can create the deployment on the API by running the prefect deployment apply Prefect CLI command.

    $ prefect deployment apply catfacts_flow-deployment.yaml\n

    For example:

    $ prefect deployment apply ./catfacts_flow-deployment.yaml\nSuccessfully loaded 'catfact'\nDeployment '76a9f1ac-4d8c-4a92-8869-615bec502685' successfully created.\n

    prefect deployment apply accepts an optional --upload flag that, when provided, uploads this deployment's files to remote storage.

    Once the deployment has been created, you'll see it in the Prefect UI and can inspect it using the CLI.

    $ prefect deployment ls\n                               Deployments\n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                           \u2503 ID                                   \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 Cat Facts/catfact              \u2502 76a9f1ac-4d8c-4a92-8869-615bec502685 \u2502\n\u2502 leonardo_dicapriflow/hello_leo \u2502 fb4681d7-aa5a-4617-bf6f-f67e6f964984 \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n

    When you run a deployed flow with Prefect, the following happens:

    • The user runs the deployment, which creates a flow run. (The API creates flow runs automatically for deployments with schedules.)
    • An agent picks up the flow run from a work queue and uses an infrastructure block to create infrastructure for the run.
    • The flow run executes within the infrastructure.

    Agents and work pools enable the Prefect orchestration engine and API to run deployments in your local execution environments. To execute deployed flow runs you need to configure at least one agent.

    Scheduled flow runs

    Scheduled flow runs will not be created unless the scheduler is running with either Prefect Cloud or a local Prefect server started with prefect server start.

    Scheduled flow runs will not run unless an appropriate agent and work pool are configured.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-deployment-from-a-python-object","title":"Create a deployment from a Python object","text":"

    You can also create deployments from Python scripts by using the prefect.deployments.Deployment class.

    Create a new deployment using configuration defaults for an imported flow:

    from my_project.flows import my_flow\nfrom prefect.deployments import Deployment\n\ndeployment = Deployment.build_from_flow(\n    flow=my_flow,\n    name=\"example-deployment\", \n    version=1, \n    work_queue_name=\"demo\",\n    work_pool_name=\"default-agent-pool\",\n)\ndeployment.apply()\n

    Create a new deployment with a pre-defined storage block and an infrastructure override:

    from my_project.flows import my_flow\nfrom prefect.deployments import Deployment\nfrom prefect.filesystems import S3\n\nstorage = S3.load(\"dev-bucket\") # load a pre-defined block\n\ndeployment = Deployment.build_from_flow(\n    flow=my_flow,\n    name=\"s3-example\",\n    version=2,\n    work_queue_name=\"aws\",\n    work_pool_name=\"default-agent-pool\",\n    storage=storage,\n    infra_overrides={\n        \"env\": {\n            \"ENV_VAR\": \"value\"\n        }\n    },\n)\n\ndeployment.apply()\n

    If you have settings that you want to share from an existing deployment you can load those settings:

    deployment = Deployment(\n    name=\"a-name-you-used\", \n    flow_name=\"name-of-flow\"\n)\ndeployment.load() # loads server-side settings\n

    Once the existing deployment settings are loaded, you may update them as needed by changing deployment properties.

    View all of the parameters for the Deployment object in the Python API documentation.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deployment-api-representation","title":"Deployment API representation","text":"

    When you create a deployment, it is constructed from deployment definition data you provide and additional properties set by client-side utilities.

    Deployment properties include:

    Property Description id An auto-generated UUID ID value identifying the deployment. created A datetime timestamp indicating when the deployment was created. updated A datetime timestamp indicating when the deployment was last changed. name The name of the deployment. version The version of the deployment description A description of the deployment. flow_id The id of the flow associated with the deployment. schedule An optional schedule for the deployment. is_schedule_active Boolean indicating whether the deployment schedule is active. Default is True. infra_overrides One or more optional infrastructure overrides parameters An optional dictionary of parameters for flow runs scheduled by the deployment. tags An optional list of tags for the deployment. work_queue_name The optional work queue that will handle the deployment's run parameter_openapi_schema JSON schema for flow parameters. enforce_parameter_schema Whether the API should validate the parameters passed to a flow run against the schema defined by parameter_openapi_schema path The path to the deployment.yaml file entrypoint The path to a flow entry point storage_document_id Storage block configured for the deployment. infrastructure_document_id Infrastructure block configured for the deployment.

    You can inspect a deployment using the CLI with the prefect deployment inspect command, referencing the deployment with <flow_name>/<deployment_name>.

    $ prefect deployment inspect 'Cat Facts/catfact'\n{\n    'id': '76a9f1ac-4d8c-4a92-8869-615bec502685',\n    'created': '2022-07-26T03:48:14.723328+00:00',\n    'updated': '2022-07-26T03:50:02.043238+00:00',\n    'name': 'catfact',\n    'version': '899b136ebc356d58562f48d8ddce7c19',\n    'description': None,\n    'flow_id': '2c7b36d1-0bdb-462e-bb97-f6eb9fef6fd5',\n    'schedule': None,\n    'is_schedule_active': True,\n    'infra_overrides': {},\n    'parameters': {},\n    'tags': [],\n    'work_queue_name': 'test',\n    'parameter_openapi_schema': {\n        'title': 'Parameters',\n        'type': 'object',\n        'properties': {'url': {'title': 'url'}},\n        'required': ['url']\n    },\n    'path': '/Users/terry/test/testflows/catfact',\n    'entrypoint': 'catfact.py:catfacts_flow',\n    'manifest_path': None,\n    'storage_document_id': None,\n    'infrastructure_document_id': 'f958db1c-b143-4709-846c-321125247e07',\n    'infrastructure': {\n        'type': 'process',\n        'env': {},\n        'labels': {},\n        'name': None,\n        'command': ['python', '-m', 'prefect.engine'],\n        'stream_output': True\n    }\n}\n
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-from-a-deployment","title":"Create a flow run from a deployment","text":"","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-with-a-schedule","title":"Create a flow run with a schedule","text":"

    If you specify a schedule for a deployment, the deployment will execute its flow automatically on that schedule as long as a Prefect server and agent are running. Prefect Cloud creates schedules flow runs automatically, and they will run on schedule if an agent is configured to pick up flow runs for the deployment.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-with-an-event-trigger","title":"Create a flow run with an event trigger","text":"

    deployment triggers are only available in Prefect Cloud

    Deployments can optionally take a trigger specification, which will configure an automation to run the deployment based on the presence or absence of events, and optionally pass event data into the deployment run as parameters via jinja templating.

    triggers:\n  - enabled: true\n    match:\n      prefect.resource.id: prefect.flow-run.*\n    expect:\n      - prefect.flow-run.Completed\n    match_related:\n      prefect.resource.name: prefect.flow.etl-flow\n      prefect.resource.role: flow\n    parameters:\n      param_1: \"{{ event }}\"\n

    When applied, this deployment will start a flow run upon the completion of the upstream flow specified in the match_related key, with the flow run passed in as a parameter. Triggers can be configured to respond to the presence or absence of arbitrary internal or external events. The trigger system and API are detailed in Automations.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-with-prefect-ui","title":"Create a flow run with Prefect UI","text":"

    In the Prefect UI, you can click the Run button next to any deployment to execute an ad hoc flow run for that deployment.

    The prefect deployment CLI command provides commands for managing and running deployments locally.

    Command Description apply Create or update a deployment from a YAML file. build Generate a deployment YAML from /path/to/file.py:flow_function. delete Delete a deployment. inspect View details about a deployment. ls View all deployments or deployments for specific flows. pause-schedule Pause schedule of a given deployment. resume-schedule Resume schedule of a given deployment. run Create a flow run for the given flow and deployment. set-schedule Set schedule for a given deployment.","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-in-a-python-script","title":"Create a flow run in a Python script","text":"

    You can create a flow run from a deployment in a Python script with the run_deployment function.

    from prefect.deployments import run_deployment\n\n\ndef main():\n    response = run_deployment(name=\"flow-name/deployment-name\")\n    print(response)\n\n\nif __name__ == \"__main__\":\n   main()\n

    PREFECT_API_URL setting for agents

    You'll need to configure agents and work pools that can create flow runs for deployments in remote environments. PREFECT_API_URL must be set for the environment in which your agent is running.

    If you want the agent to communicate with Prefect Cloud from a remote execution environment such as a VM or Docker container, you must configure PREFECT_API_URL in that environment.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#examples","title":"Examples","text":"
    • How to deploy Prefect flows to AWS
    • How to deploy Prefect flows to GCP
    • How to deploy Prefect flows to Azure
    • How to deploy Prefect flows using files stored locally
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments/","title":"Deployments","text":"

    Deployments are server-side representations of flows. They store the crucial metadata needed for remote orchestration including when, where, and how a workflow should run. Deployments elevate workflows from functions that you must call manually to API-managed entities that can be triggered remotely.

    Here we will focus largely on the metadata that defines a deployment and how it is used. Different ways of creating a deployment populate these fields differently.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#overview","title":"Overview","text":"

    Every Prefect deployment references one and only one \"entrypoint\" flow (though that flow may itself call any number of subflows). Different deployments may reference the same underlying flow, a useful pattern when developing or promoting workflow changes through staged environments.

    The complete schema that defines a deployment is as follows:

    class Deployment:\n    \"\"\"\n    Structure of the schema defining a deployment\n    \"\"\"\n\n    # required defining data\n    name: str \n    flow_id: UUID\n    entrypoint: str\n    path: str = None\n\n    # workflow scheduling and parametrization\n    parameters: dict = None\n    parameter_openapi_schema: dict = None\n    schedule: Schedule = None\n    is_schedule_active: bool = True\n    trigger: Trigger = None\n\n    # metadata for bookkeeping\n    version: str = None\n    description: str = None\n    tags: list = None\n\n    # worker-specific fields\n    work_pool_name: str = None\n    work_queue_name: str = None\n    infra_overrides: dict = None\n    pull_steps: dict = None\n

    All methods for creating Prefect deployments are interfaces for populating this schema. Let's look at each section in turn.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#required-data","title":"Required data","text":"

    Deployments universally require both a name and a reference to an underlying Flow. In almost all instances of deployment creation, users do not need to concern themselves with the flow_id as most interfaces will only need the flow's name. Note that the deployment name is not required to be unique across all deployments but is required to be unique for a given flow ID. As a consequence, you will often see references to the deployment's unique identifying name {FLOW_NAME}/{DEPLOYMENT_NAME}. For example, triggering a run of a deployment from the Prefect CLI can be done via:

    prefect deployment run my-first-flow/my-first-deployment\n

    The other two fields are less obvious:

    • path: the path can generally be interpreted as the runtime working directory for the flow. For example, if a deployment references a workflow defined within a Docker image, the path will be the absolute path to the parent directory where that workflow will run anytime the deployment is triggered. This interpretation is more subtle in the case of flows defined in remote filesystems.
    • entrypoint: the entrypoint of a deployment is a relative reference to a function decorated as a flow that exists on some filesystem. It is always specified relative to the path. Entrypoints use Python's standard path-to-object syntax (e.g., path/to/file.py:function_name or simply path:object).

    The entrypoint must reference the same flow as the flow ID.

    Note that Prefect requires that deployments reference flows defined within Python files. Flows defined within interactive REPLs or notebooks cannot currently be deployed as such. They are still valid flows that will be monitored by the API and observable in the UI whenever they are run, but Prefect cannot trigger them.

    Deployments do not contain code definitions

    Deployment metadata references code that exists in potentially diverse locations within your environment. This separation of concerns means that your flow code stays within your storage and execution infrastructure and never lives on the Prefect server or database.

    This is the heart of the Prefect hybrid model: there's a boundary between your proprietary assets, such as your flow code, and the Prefect backend (including Prefect Cloud).

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#scheduling-and-parametrization","title":"Scheduling and parametrization","text":"

    One of the primary motivations for creating deployments of flows is to remotely schedule and trigger them. Just as flows can be called as functions with different input values, so can deployments be triggered or scheduled with different values through the use of parameters.

    The six fields here capture the necessary metadata to perform such actions:

    • schedule: a schedule object. Most of the convenient interfaces for creating deployments allow users to avoid creating this object themselves. For example, when updating a deployment schedule in the UI basic information such as a cron string or interval is all that's required.
    • trigger (Cloud-only): triggers allow you to define event-based rules for running a deployment. For more information see Automations.
    • parameter_openapi_schema: an OpenAPI compatible schema that defines the types and defaults for the flow's parameters. This is used by both the UI and the backend to expose options for creating manual runs as well as type validation.
    • parameters: default values of flow parameters that this deployment will pass on each run. These can be overwritten through a trigger or when manually creating a custom run.
    • enforce_parameter_schema: a boolean flag that determines whether the API should validate the parameters passed to a flow run against the schema defined by parameter_openapi_schema.

    Scheduling is asynchronous and decoupled

    Because deployments are nothing more than metadata, runs can be created at anytime. Note that pausing a schedule, updating your deployment, and other actions reset your auto-scheduled runs.

    Scheduling deployments from within Python flow code

    Note that Prefect provides a run_deployment function that can be used to schedule the run of an existing deployment when your Python code executes.

    from prefect.deployments import run_deployment\n\ndef main():\n    run_deployment(name=\"my_flow_name/my_deployment_name\")\n

    Pass timeout=0 to return immediately and not block.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#versioning-and-bookkeeping","title":"Versioning and bookkeeping","text":"

    Versions, descriptions and tags are omnipresent fields throughout Prefect that can be easy to overlook. However, putting some extra thought into how you use these fields can pay dividends down the road.

    • version: versions are always set by the client and can be any arbitrary string. We recommend tightly coupling this field on your deployments to your software development lifecycle. For example if you leverage git to manage code changes, use either a tag or commit hash in this field. If you don't set a value for the version, Prefect will compute a hash
    • description: the description field of a deployment is a place to provide rich reference material for downstream stakeholders such as intended use and parameter documentation. Markdown formatting will be rendered in the Prefect UI, allowing for section headers, links, tables, and other formatting. If not provided explicitly, Prefect will use the docstring of your flow function as a default value.
    • tags: tags are a mechanism for grouping related work together across a diverse set of objects. Tags set on a deployment will be inherited by that deployment's flow runs. These tags can then be used to filter what runs are displayed on the primary UI dashboard, allowing you to customize different views into your work. In addition, in Prefect Cloud you can easily find objects through searching by tag.

    All of these bits of metadata can be leveraged to great effect by injecting them into the processes that Prefect is orchestrating. For example you can use both run ID and versions to organize files that you produce from your workflows, or by associating your flow run's tags with the metadata of a job it orchestrates. This metadata is available during execution through Prefect runtime.

    Everything has a version

    Deployments aren't the only entity in Prefect with a version attached; both flows and tasks also have versions that can be set through their respective decorators. These versions will be sent to the API anytime the flow or task is run and thereby allow you to audit your changes across all levels.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#workers-and-work-pools","title":"Workers and Work Pools","text":"

    Workers and work pools are an advanced deployment pattern that allow you to dynamically provision infrastructure for each flow run. In addition, the work pool job template interface allows users to create and govern opinionated interfaces to their workflow infrastructure. To do this, a deployment using workers needs to evaluate the following fields:

    • work_pool_name: the name of the work pool this deployment will be associated with. Work pool types mirror infrastructure types and therefore the decision here affects the options available for the other fields.
    • work_queue_name: if you are using work queues to either manage priority or concurrency, you can associate a deployment with a specific queue within a work pool using this field.
    • infra_overrides: often called job_variables within various interfaces, this field allows deployment authors to customize whatever infrastructure options have been exposed on this work pool. This field is often used for things such as Docker image names, Kubernetes annotations and limits, and environment variables.
    • pull_steps: a JSON description of steps that should be performed to retrieve flow code or configuration and prepare the runtime environment for workflow execution.

    Pull steps allow users to highly decouple their workflow architecture. For example, a common use of pull steps is to dynamically pull code from remote filesystems such as GitHub with each run of their deployment.

    For more information see the guide to deploying with a worker.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#two-approaches-to-deployments","title":"Two approaches to deployments","text":"

    There are two primary ways to deploy flows with Prefect, differentiated by how much control Prefect has over the infrastructure in which the flows run.

    In one setup, deploying Prefect flows is analogous to deploying a webserver - users author their workflows and then start a long-running process (often within a Docker container) that is responsible for managing all of the runs for the associated deployment(s).

    In the other setup, you do a little extra up-front work to set up a work pool and a base job template that defines how individual flow runs will be submitted to infrastructure.

    Prefect provides several types of work pools corresponding to different types of infrastructure. Prefect Cloud provides a Prefect Managed work pool option that is the simplest way to run workflows remotely. A cloud-provider account, such as AWS, is not required with a Prefect Managed work pool.

    Some work pool types require a client-side worker to submit job definitions to the appropriate infrastructure with each run.

    Each of these setups can support production workloads. The choice ultimately boils down to your use case and preferences. Read further to decide which setup is best for your situation.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#serving-flows-on-long-lived-infrastructure","title":"Serving flows on long-lived infrastructure","text":"

    When you have several flows running regularly, the serve method of the Flow object or the serve utility is a great option for managing multiple flows simultaneously.

    Once you have authored your flow and decided on its deployment settings as described above, all that's left is to run this long-running process in a location of your choosing. The process will stay in communication with the Prefect API, monitoring for work and submitting each run within an individual subprocess. Note that because runs are submitted to subprocesses, any external infrastructure configuration will need to be setup beforehand and kept associated with this process.

    This approach has many benefits:

    • Users are in complete control of their infrastructure, and anywhere the \"serve\" Python process can run is a suitable deployment environment.
    • It is simple to reason about.
    • Creating deployments requires a minimal set of decisions.
    • Iteration speed is fast.

    However, there are a few reasons you might consider running flows on dynamically provisioned infrastructure with work pools instead:

    • Flows that have expensive infrastructure needs may be more costly in this setup due to the long-running process.
    • Flows with heterogeneous infrastructure needs across runs will be more difficult to configure and schedule.
    • Large volumes of deployments can be harder to track.
    • If your internal team structure requires that deployment authors be members of a different team than the team managing infrastructure, the work pool interface may be preferred.
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#dynamically-provisioning-infrastructure-with-work-pools","title":"Dynamically provisioning infrastructure with work pools","text":"

    Work pools allow Prefect to exercise greater control of the infrastructure on which flows run. Options for serverless work pools allow you to scale to zero when workflows aren't running. Prefect even provides you with the ability to provision cloud infrastructure via a single CLI command, if you use a Prefect Cloud push work pool option.

    With work pools:

    • You can configure and monitor infrastructure configuration within the Prefect UI.
    • Infrastructure is ephemeral and dynamically provisioned.
    • Prefect is more infrastructure-aware and therefore collects more event data from your infrastructure by default.
    • Highly decoupled setups are possible.

    You don't have to commit to one approach

    You are not required to use only one of these approaches for your deployments. You can mix and match approaches based on the needs of each flow. Further, you can change the deployment approach for a particular flow as its needs evolve. For example, you might use workers for your expensive machine learning pipelines, but use the serve mechanics for smaller, more frequent file-processing pipelines.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/events/","title":"Events","text":"

    An event is a notification of a change. Together, events form a feed of activity recording what's happening across your stack.

    Events power several features in Prefect Cloud, including flow run logs, audit logs, and automations.

    Events can represent API calls, state transitions, or changes in your execution environment or infrastructure.

    Events enable observability into your data stack via the event feed, and the configuration of Prefect's reactivity via automations.

    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#event-specification","title":"Event Specification","text":"

    Events adhere to a structured specification.

    Name Type Required? Description occurred String yes When the event happened event String yes The name of the event that happened resource Object yes The primary Resource this event concerns related Array no A list of additional Resources involved in this event payload Object no An open-ended set of data describing what happened id String yes The client-provided identifier of this event follows String no The ID of an event that is known to have occurred prior to this one.","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#event-grammar","title":"Event Grammar","text":"

    Generally, events have a consistent and informative grammar - an event describes a resource and an action that the resource took or that was taken on that resource. For example, events emitted by Prefect objects take the form of:

    prefect.block.write-method.called\nprefect-cloud.automation.action.executed\nprefect-cloud.user.logged-in\n
    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#event-sources","title":"Event Sources","text":"

    Events are automatically emitted by all Prefect objects, including flows, tasks, deployments, work queues, and logs. Prefect-emitted events will contain the prefect or prefect-cloud resource prefix. Events can also be sent to the Prefect events API via authenticated http request.

    The Prefect SDK provides a method that emits events, for use in arbitrary python code that may not be a task or flow. Running the following code will emit events to Prefect Cloud, which will validate and ingest the event data.

    from prefect.events import emit_event\n\ndef some_function(name: str=\"kiki\") -> None:\n    print(f\"hi {name}!\")\n    emit_event(event=f\"{name}.sent.event!\", resource={\"prefect.resource.id\": f\"coder.{name}\"})\n\nsome_function()\n

    Prefect Cloud offers programmable webhooks to receive HTTP requests from other systems and translate them into events within your workspace. Webhooks can emit pre-defined static events, dynamic events that use portions of the incoming HTTP request, or events derived from CloudEvents.

    Events emitted from any source will appear in the event feed, where you can visualize activity in context and configure automations to react to the presence or absence of it in the future.

    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#resources","title":"Resources","text":"

    Every event has a primary resource, which describes the object that emitted an event. Resources are used as quasi-stable identifiers for sources of events, and are constructed as dot-delimited strings, for example:

    prefect-cloud.automation.5b9c5c3d-6ca0-48d0-8331-79f4b65385b3.action.0\nacme.user.kiki.elt_script_1\nprefect.flow-run.e3755d32-cec5-42ca-9bcd-af236e308ba6\n

    Resources can optionally have additional arbitrary labels which can be used in event aggregation queries, such as:

    \"resource\": {\n    \"prefect.resource.id\": \"prefect-cloud.automation.5b9c5c3d-6ca0-48d0-8331-79f4b65385b3\",\n    \"prefect-cloud.action.type\": \"call-webhook\"\n    }\n

    Events can optionally contain related resources, used to associate the event with other resources, such as in the case that the primary resource acted on or with another resource:

    \"resource\": {\n    \"prefect.resource.id\": \"prefect-cloud.automation.5b9c5c3d-6ca0-48d0-8331-79f4b65385b3.action.0\",\n    \"prefect-cloud.action.type\": \"call-webhook\"\n  },\n\"related\": [\n  {\n      \"prefect.resource.id\": \"prefect-cloud.automation.5b9c5c3d-6ca0-48d0-8331-79f4b65385b3\",\n      \"prefect.resource.role\": \"automation\",\n      \"prefect-cloud.name\": \"webhook_body_demo\",\n      \"prefect-cloud.posture\": \"Reactive\"\n  }\n]\n
    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#events-in-the-cloud-ui","title":"Events in the Cloud UI","text":"

    Prefect Cloud provides an interactive dashboard to analyze and take action on events that occurred in your workspace on the event feed page.

    The event feed is the primary place to view, search, and filter events to understand activity across your stack. Each entry displays data on the resource, related resource, and event that took place.

    You can view more information about an event by clicking into it, where you can view the full details of an event's resource, related resources, and its payload.

    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#reacting-to-events","title":"Reacting to events","text":"

    From an event page, you can configure an automation to trigger on the observation of matching events or a lack of matching events by clicking the automate button in the overflow menu:

    The default trigger configuration will fire every time it sees an event with a matching resource identifier. Advanced configuration is possible via custom triggers.

    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/filesystems/","title":"Filesystems","text":"

    A filesystem block is an object that allows you to read and write data from paths. Prefect provides multiple built-in file system types that cover a wide range of use cases.

    • LocalFileSystem
    • RemoteFileSystem
    • Azure
    • GitHub
    • GitLab
    • GCS
    • S3
    • SMB

    Additional file system types are available in Prefect Collections.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#local-filesystem","title":"Local filesystem","text":"

    The LocalFileSystem block enables interaction with the files in your current development environment.

    LocalFileSystem properties include:

    Property Description basepath String path to the location of files on the local filesystem. Access to files outside of the base path will not be allowed.
    from prefect.filesystems import LocalFileSystem\n\nfs = LocalFileSystem(basepath=\"/foo/bar\")\n

    Limited access to local file system

    Be aware that LocalFileSystem access is limited to the exact path provided. This file system may not be ideal for some use cases. The execution environment for your workflows may not have the same file system as the environment you are writing and deploying your code on.

    Use of this file system can limit the availability of results after a flow run has completed or prevent the code for a flow from being retrieved successfully at the start of a run.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#remote-file-system","title":"Remote file system","text":"

    The RemoteFileSystem block enables interaction with arbitrary remote file systems. Under the hood, RemoteFileSystem uses fsspec and supports any file system that fsspec supports.

    RemoteFileSystem properties include:

    Property Description basepath String path to the location of files on the remote filesystem. Access to files outside of the base path will not be allowed. settings Dictionary containing extra parameters required to access the remote file system.

    The file system is specified using a protocol:

    • s3://my-bucket/my-folder/ will use S3
    • gcs://my-bucket/my-folder/ will use GCS
    • az://my-bucket/my-folder/ will use Azure

    For example, to use it with Amazon S3:

    from prefect.filesystems import RemoteFileSystem\n\nblock = RemoteFileSystem(basepath=\"s3://my-bucket/folder/\")\nblock.save(\"dev\")\n

    You may need to install additional libraries to use some remote storage types.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#remotefilesystem-examples","title":"RemoteFileSystem examples","text":"

    How can we use RemoteFileSystem to store our flow code? The following is a use case where we use MinIO as a storage backend:

    from prefect.filesystems import RemoteFileSystem\n\nminio_block = RemoteFileSystem(\n    basepath=\"s3://my-bucket\",\n    settings={\n        \"key\": \"MINIO_ROOT_USER\",\n        \"secret\": \"MINIO_ROOT_PASSWORD\",\n        \"client_kwargs\": {\"endpoint_url\": \"http://localhost:9000\"},\n    },\n)\nminio_block.save(\"minio\")\n
    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#azure","title":"Azure","text":"

    The Azure file system block enables interaction with Azure Datalake and Azure Blob Storage. Under the hood, the Azure block uses adlfs.

    Azure properties include:

    Property Description bucket_path String path to the location of files on the remote filesystem. Access to files outside of the bucket path will not be allowed. azure_storage_connection_string Azure storage connection string. azure_storage_account_name Azure storage account name. azure_storage_account_key Azure storage account key. azure_storage_tenant_id Azure storage tenant ID. azure_storage_client_id Azure storage client ID. azure_storage_client_secret Azure storage client secret. azure_storage_anon Anonymous authentication, disable to use DefaultAzureCredential.

    To create a block:

    from prefect.filesystems import Azure\n\nblock = Azure(bucket_path=\"my-bucket/folder/\")\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb az/dev\n

    You need to install adlfs to use it.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#github","title":"GitHub","text":"

    The GitHub filesystem block enables interaction with GitHub repositories. This block is read-only and works with both public and private repositories.

    GitHub properties include:

    Property Description reference An optional reference to pin to, such as a branch name or tag. repository The URL of a GitHub repository to read from, in either HTTPS or SSH format. access_token A GitHub Personal Access Token (PAT) with repo scope.

    To create a block:

    from prefect.filesystems import GitHub\n\nblock = GitHub(\n    repository=\"https://github.com/my-repo/\",\n    access_token=<my_access_token> # only required for private repos\n)\nblock.get_directory(\"folder-in-repo\") # specify a subfolder of repo\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb github/dev -a\n
    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#gitlabrepository","title":"GitLabRepository","text":"

    The GitLabRepository block is read-only and works with private GitLab repositories.

    GitLabRepository properties include:

    Property Description reference An optional reference to pin to, such as a branch name or tag. repository The URL of a GitLab repository to read from, in either HTTPS or SSH format. credentials A GitLabCredentials block with Personal Access Token (PAT) with read_repository scope.

    To create a block:

    from prefect_gitlab.credentials import GitLabCredentials\nfrom prefect_gitlab.repositories import GitLabRepository\n\ngitlab_creds = GitLabCredentials(token=\"YOUR_GITLAB_ACCESS_TOKEN\")\ngitlab_repo = GitLabRepository(\n    repository=\"https://gitlab.com/yourorg/yourrepo.git\",\n    reference=\"main\",\n    credentials=gitlab_creds,\n)\ngitlab_repo.save(\"dev\")\n

    To use it in a deployment (and apply):

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb gitlab-repository/dev -a\n

    Note that to use this block, you need to install the prefect-gitlab collection.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#gcs","title":"GCS","text":"

    The GCS file system block enables interaction with Google Cloud Storage. Under the hood, GCS uses gcsfs.

    GCS properties include:

    Property Description bucket_path A GCS bucket path service_account_info The contents of a service account keyfile as a JSON string. project The project the GCS bucket resides in. If not provided, the project will be inferred from the credentials or environment.

    To create a block:

    from prefect.filesystems import GCS\n\nblock = GCS(bucket_path=\"my-bucket/folder/\")\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb gcs/dev\n

    You need to install gcsfsto use it.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#s3","title":"S3","text":"

    The S3 file system block enables interaction with Amazon S3. Under the hood, S3 uses s3fs.

    S3 properties include:

    Property Description bucket_path An S3 bucket path aws_access_key_id AWS Access Key ID aws_secret_access_key AWS Secret Access Key

    To create a block:

    from prefect.filesystems import S3\n\nblock = S3(bucket_path=\"my-bucket/folder/\")\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb s3/dev\n

    You need to install s3fsto use this block.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#smb","title":"SMB","text":"

    The SMB file system block enables interaction with SMB shared network storage. Under the hood, SMB uses smbprotocol. Used to connect to Windows-based SMB shares from Linux-based Prefect flows. The SMB file system block is able to copy files, but cannot create directories.

    SMB properties include:

    Property Description basepath String path to the location of files on the remote filesystem. Access to files outside of the base path will not be allowed. smb_host Hostname or IP address where SMB network share is located. smb_port Port for SMB network share (defaults to 445). smb_username SMB username with read/write permissions. smb_password SMB password.

    To create a block:

    from prefect.filesystems import SMB\n\nblock = SMB(basepath=\"my-share/folder/\")\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb smb/dev\n

    You need to install smbprotocol to use it.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#handling-credentials-for-cloud-object-storage-services","title":"Handling credentials for cloud object storage services","text":"

    If you leverage S3, GCS, or Azure storage blocks, and you don't explicitly configure credentials on the respective storage block, those credentials will be inferred from the environment. Make sure to set those either explicitly on the block or as environment variables, configuration files, or IAM roles within both the build and runtime environment for your deployments.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#filesystem-package-dependencies","title":"Filesystem package dependencies","text":"

    A Prefect installation and doesn't include filesystem-specific package dependencies such as s3fs, gcsfs or adlfs. This includes Prefect base Docker images.

    You must ensure that filesystem-specific libraries are installed in an execution environment where they will be used by flow runs.

    In Dockerized deployments using the Prefect base image, you can leverage the EXTRA_PIP_PACKAGES environment variable. Those dependencies will be installed at runtime within your Docker container or Kubernetes Job before the flow starts running.

    In Dockerized deployments using a custom image, you must include the filesystem-specific package dependency in your image.

    Here is an example from a deployment YAML file showing how to specify the installation of s3fs from into your image:

    infrastructure:\n  type: docker-container\n  env:\n    EXTRA_PIP_PACKAGES: s3fs  # could be gcsfs, adlfs, etc.\n

    You may specify multiple dependencies by providing a comma-delimted list.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#saving-and-loading-file-systems","title":"Saving and loading file systems","text":"

    Configuration for a file system can be saved to the Prefect API. For example:

    fs = RemoteFileSystem(basepath=\"s3://my-bucket/folder/\")\nfs.write_path(\"foo\", b\"hello\")\nfs.save(\"dev-s3\")\n

    This file system can be retrieved for later use with load.

    fs = RemoteFileSystem.load(\"dev-s3\")\nfs.read_path(\"foo\")  # b'hello'\n
    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#readable-and-writable-file-systems","title":"Readable and writable file systems","text":"

    Prefect provides two abstract file system types, ReadableFileSystem and WriteableFileSystem.

    • All readable file systems must implement read_path, which takes a file path to read content from and returns bytes.
    • All writeable file systems must implement write_path which takes a file path and content and writes the content to the file as bytes.

    A file system may implement both of these types.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/flows/","title":"Flows","text":"

    Flows are the most central Prefect object. A flow is a container for workflow logic as-code and allows users to configure how their workflows behave. Flows are defined as Python functions, and any Python function is eligible to be a flow.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#flows-overview","title":"Flows overview","text":"

    Flows can be thought of as special types of functions. They can take inputs, perform work, and return an output. In fact, you can turn any function into a Prefect flow by adding the @flow decorator. When a function becomes a flow, its behavior changes, giving it the following advantages:

    • Every invocation of this function is tracked and all state transitions are reported to the API, allowing observation of flow execution.
    • Input arguments are automatically type checked and coerced to the appropriate types.
    • Retries can be performed on failure.
    • Timeouts can be enforced to prevent unintentional, long-running workflows.

    Flows also take advantage of automatic Prefect logging to capture details about flow runs such as run time and final state.

    Flows can include calls to tasks as well as to other flows, which Prefect calls \"subflows\" in this context. Flows may be defined within modules and imported for use as subflows in your flow definitions.

    Deployments elevate individual workflows from functions that you call manually to API-managed entities.

    Tasks must be called from flows

    All tasks must be called from within a flow. Tasks may not be called from other tasks.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#flow-runs","title":"Flow runs","text":"

    A flow run represents a single execution of the flow.

    You can create a flow run by calling the flow manually. For example, by running a Python script or importing the flow into an interactive session and calling it.

    You can also create a flow run by:

    • Using external schedulers such as cron to invoke a flow function
    • Creating a deployment on Prefect Cloud or a locally run Prefect server.
    • Creating a flow run for the deployment via a schedule, the Prefect UI, or the Prefect API.

    However you run the flow, the Prefect API monitors the flow run, capturing flow run state for observability.

    When you run a flow that contains tasks or additional flows, Prefect will track the relationship of each child run to the parent flow run.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#writing-flows","title":"Writing flows","text":"

    The @flow decorator is used to designate a flow:

    from prefect import flow\n\n@flow\ndef my_flow():\n    return\n

    There are no rigid rules for what code you include within a flow definition - all valid Python is acceptable.

    Flows are uniquely identified by name. You can provide a name parameter value for the flow. If you don't provide a name, Prefect uses the flow function name.

    @flow(name=\"My Flow\")\ndef my_flow():\n    return\n

    Flows can call tasks to allow Prefect to orchestrate and track more granular units of work:

    from prefect import flow, task\n\n@task\ndef print_hello(name):\n    print(f\"Hello {name}!\")\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    print_hello(name)\n

    Flows and tasks

    There's nothing stopping you from putting all of your code in a single flow function \u2014 Prefect will happily run it!

    However, organizing your workflow code into smaller flow and task units lets you take advantage of Prefect features like retries, more granular visibility into runtime state, the ability to determine final state regardless of individual task state, and more.

    In addition, if you put all of your workflow logic in a single flow function and any line of code fails, the entire flow will fail and must be retried from the beginning. This can be avoided by breaking up the code into multiple tasks.

    You may call any number of other tasks, subflows, and even regular Python functions within your flow. You can pass parameters to your flow function that will be used elsewhere in the workflow, and Prefect will report on the progress and final state of any invocation.

    Prefect encourages \"small tasks\" \u2014 each one should represent a single logical step of your workflow. This allows Prefect to better contain task failures.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#flow-settings","title":"Flow settings","text":"

    Flows allow a great deal of configuration by passing arguments to the decorator. Flows accept the following optional settings.

    Argument Description description An optional string description for the flow. If not provided, the description will be pulled from the docstring for the decorated function. name An optional name for the flow. If not provided, the name will be inferred from the function. retries An optional number of times to retry on flow run failure. retry_delay_seconds An optional number of seconds to wait before retrying the flow after failure. This is only applicable if retries is nonzero. flow_run_name An optional name to distinguish runs of this flow; this name can be provided as a string template with the flow's parameters as variables; this name can also be provided as a function that returns a string. task_runner An optional task runner to use for task execution within the flow when you .submit() tasks. If not provided and you .submit() tasks, the ConcurrentTaskRunner will be used. timeout_seconds An optional number of seconds indicating a maximum runtime for the flow. If the flow exceeds this runtime, it will be marked as failed. Flow execution may continue until the next task is called. validate_parameters Boolean indicating whether parameters passed to flows are validated by Pydantic. Default is True. version An optional version string for the flow. If not provided, we will attempt to create a version string as a hash of the file containing the wrapped function. If the file cannot be located, the version will be null.

    For example, you can provide a name value for the flow. Here we've also used the optional description argument and specified a non-default task runner.

    from prefect import flow\nfrom prefect.task_runners import SequentialTaskRunner\n\n@flow(name=\"My Flow\",\n      description=\"My flow using SequentialTaskRunner\",\n      task_runner=SequentialTaskRunner())\ndef my_flow():\n    return\n

    You can also provide the description as the docstring on the flow function.

    @flow(name=\"My Flow\",\n      task_runner=SequentialTaskRunner())\ndef my_flow():\n    \"\"\"My flow using SequentialTaskRunner\"\"\"\n    return\n

    You can distinguish runs of this flow by providing a flow_run_name. This setting accepts a string that can optionally contain templated references to the parameters of your flow. The name will be formatted using Python's standard string formatting syntax as can be seen here:

    import datetime\nfrom prefect import flow\n\n@flow(flow_run_name=\"{name}-on-{date:%A}\")\ndef my_flow(name: str, date: datetime.datetime):\n    pass\n\n# creates a flow run called 'marvin-on-Thursday'\nmy_flow(name=\"marvin\", date=datetime.datetime.utcnow())\n

    Additionally this setting also accepts a function that returns a string for the flow run name:

    import datetime\nfrom prefect import flow\n\ndef generate_flow_run_name():\n    date = datetime.datetime.utcnow()\n\n    return f\"{date:%A}-is-a-nice-day\"\n\n@flow(flow_run_name=generate_flow_run_name)\ndef my_flow(name: str):\n    pass\n\n# creates a flow run called 'Thursday-is-a-nice-day'\nmy_flow(name=\"marvin\")\n

    If you need access to information about the flow, use the prefect.runtime module. For example:

    from prefect import flow\nfrom prefect.runtime import flow_run\n\ndef generate_flow_run_name():\n    flow_name = flow_run.flow_name\n\n    parameters = flow_run.parameters\n    name = parameters[\"name\"]\n    limit = parameters[\"limit\"]\n\n    return f\"{flow_name}-with-{name}-and-{limit}\"\n\n@flow(flow_run_name=generate_flow_run_name)\ndef my_flow(name: str, limit: int = 100):\n    pass\n\n# creates a flow run called 'my-flow-with-marvin-and-100'\nmy_flow(name=\"marvin\")\n

    Note that validate_parameters will check that input values conform to the annotated types on the function. Where possible, values will be coerced into the correct type. For example, if a parameter is defined as x: int and \"5\" is passed, it will be resolved to 5. If set to False, no validation will be performed on flow parameters.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#separating-logic-into-tasks","title":"Separating logic into tasks","text":"

    The simplest workflow is just a @flow function that does all the work of the workflow.

    from prefect import flow\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    print(f\"Hello {name}!\")\n\nhello_world(\"Marvin\")\n

    When you run this flow, you'll see output like the following:

    $ python hello.py\n15:11:23.594 | INFO    | prefect.engine - Created flow run 'benevolent-donkey' for flow 'hello-world'\n15:11:23.594 | INFO    | Flow run 'benevolent-donkey' - Using task runner 'ConcurrentTaskRunner'\nHello Marvin!\n15:11:24.447 | INFO    | Flow run 'benevolent-donkey' - Finished in state Completed()\n

    A better practice is to create @task functions that do the specific work of your flow, and use your @flow function as the conductor that orchestrates the flow of your application:

    from prefect import flow, task\n\n@task(name=\"Print Hello\")\ndef print_hello(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    message = print_hello(name)\n\nhello_world(\"Marvin\")\n

    When you run this flow, you'll see the following output, which illustrates how the work is encapsulated in a task run.

    $ python hello.py\n15:15:58.673 | INFO    | prefect.engine - Created flow run 'loose-wolverine' for flow 'Hello Flow'\n15:15:58.674 | INFO    | Flow run 'loose-wolverine' - Using task runner 'ConcurrentTaskRunner'\n15:15:58.973 | INFO    | Flow run 'loose-wolverine' - Created task run 'Print Hello-84f0fe0e-0' for task 'Print Hello'\nHello Marvin!\n15:15:59.037 | INFO    | Task run 'Print Hello-84f0fe0e-0' - Finished in state Completed()\n15:15:59.568 | INFO    | Flow run 'loose-wolverine' - Finished in state Completed('All states completed.')\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#visualizing-flow-structure","title":"Visualizing flow structure","text":"

    You can get a quick sense of the structure of your flow using the .visualize() method on your flow. Calling this method will attempt to produce a schematic diagram of your flow and tasks without actually running your flow code.

    Functions and code not inside of flows or tasks will still be run when calling .visualize(). This may have unintended consequences. Place your code into tasks to avoid unintended execution.

    To use the visualize() method, Graphviz must be installed and on your PATH. Please install Graphviz from http://www.graphviz.org/download/. And note: just installing the graphviz python package is not sufficient.

    from prefect import flow, task\n\n@task(name=\"Print Hello\")\ndef print_hello(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@task(name=\"Print Hello Again\")\ndef print_hello_again(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    message = print_hello(name)\n    message2 = print_hello_again(message)\n\nhello_world.visualize()\n

    Prefect cannot automatically produce a schematic for dynamic workflows, such as those with loops or if/else control flow. In this case, you can provide tasks with mock return values for use in the visualize() call.

    from prefect import flow, task\n@task(viz_return_value=[4])\ndef get_list():\n    return [1, 2, 3]\n\n@task\ndef append_one(n):\n    return n.append(6)\n\n@flow\ndef viz_return_value_tracked():\n    l = get_list()\n    for num in range(3):\n        l.append(5)\n        append_one(l)\n\nviz_return_value_tracked.visualize()\n

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#composing-flows","title":"Composing flows","text":"

    A subflow run is created when a flow function is called inside the execution of another flow. The primary flow is the \"parent\" flow. The flow created within the parent is the \"child\" flow or \"subflow.\"

    Subflow runs behave like normal flow runs. There is a full representation of the flow run in the backend as if it had been called separately. When a subflow starts, it will create a new task runner for tasks within the subflow. When the subflow completes, the task runner is shut down.

    Subflows will block execution of the parent flow until completion. However, asynchronous subflows can be run in parallel by using AnyIO task groups or asyncio.gather.

    Subflows differ from normal flows in that they will resolve any passed task futures into data. This allows data to be passed from the parent flow to the child easily.

    The relationship between a child and parent flow is tracked by creating a special task run in the parent flow. This task run will mirror the state of the child flow run.

    A task that represents a subflow will be annotated as such in its state_details via the presence of a child_flow_run_id field. A subflow can be identified via the presence of a parent_task_run_id on state_details.

    You can define multiple flows within the same file. Whether running locally or via a deployment, you must indicate which flow is the entrypoint for a flow run.

    Cancelling subflow runs

    Inline subflow runs, specifically those created without run_deployment, cannot be cancelled without cancelling their parent flow run. If you may need to cancel a subflow run independent of its parent flow run, we recommend deploying it separately and starting it using the run_deployment function.

    from prefect import flow, task\n\n@task(name=\"Print Hello\")\ndef print_hello(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@flow(name=\"Subflow\")\ndef my_subflow(msg):\n    print(f\"Subflow says: {msg}\")\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    message = print_hello(name)\n    my_subflow(message)\n\nhello_world(\"Marvin\")\n

    You can also define flows or tasks in separate modules and import them for usage. For example, here's a simple subflow module:

    from prefect import flow, task\n\n@flow(name=\"Subflow\")\ndef my_subflow(msg):\n    print(f\"Subflow says: {msg}\")\n

    Here's a parent flow that imports and uses my_subflow() as a subflow:

    from prefect import flow, task\nfrom subflow import my_subflow\n\n@task(name=\"Print Hello\")\ndef print_hello(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    message = print_hello(name)\n    my_subflow(message)\n\nhello_world(\"Marvin\")\n

    Running the hello_world() flow (in this example from the file hello.py) creates a flow run like this:

    $ python hello.py\n15:19:21.651 | INFO    | prefect.engine - Created flow run 'daft-cougar' for flow 'Hello Flow'\n15:19:21.651 | INFO    | Flow run 'daft-cougar' - Using task runner 'ConcurrentTaskRunner'\n15:19:21.945 | INFO    | Flow run 'daft-cougar' - Created task run 'Print Hello-84f0fe0e-0' for task 'Print Hello'\nHello Marvin!\n15:19:22.055 | INFO    | Task run 'Print Hello-84f0fe0e-0' - Finished in state Completed()\n15:19:22.107 | INFO    | Flow run 'daft-cougar' - Created subflow run 'ninja-duck' for flow 'Subflow'\nSubflow says: Hello Marvin!\n15:19:22.794 | INFO    | Flow run 'ninja-duck' - Finished in state Completed()\n15:19:23.215 | INFO    | Flow run 'daft-cougar' - Finished in state Completed('All states completed.')\n

    Subflows or tasks?

    In Prefect you can call tasks or subflows to do work within your workflow, including passing results from other tasks to your subflow. So a common question is:

    \"When should I use a subflow instead of a task?\"

    We recommend writing tasks that do a discrete, specific piece of work in your workflow: calling an API, performing a database operation, analyzing or transforming a data point. Prefect tasks are well suited to parallel or distributed execution using distributed computation frameworks such as Dask or Ray. For troubleshooting, the more granular you create your tasks, the easier it is to find and fix issues should a task fail.

    Subflows enable you to group related tasks within your workflow. Here are some scenarios where you might choose to use a subflow rather than calling tasks individually:

    • Observability: Subflows, like any other flow run, have first-class observability within the Prefect UI and Prefect Cloud. You'll see subflow status in the Flow Runs dashboard rather than having to dig down into the tasks within a specific flow run. See Final state determination for some examples of leveraging task state within flows.
    • Conditional flows: If you have a group of tasks that run only under certain conditions, you can group them within a subflow and conditionally run the subflow rather than each task individually.
    • Parameters: Flows have first-class support for parameterization, making it easy to run the same group of tasks in different use cases by simply passing different parameters to the subflow in which they run.
    • Task runners: Subflows enable you to specify the task runner used for tasks within the flow. For example, if you want to optimize parallel execution of certain tasks with Dask, you can group them in a subflow that uses the Dask task runner. You can use a different task runner for each subflow.
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#parameters","title":"Parameters","text":"

    Flows can be called with both positional and keyword arguments. These arguments are resolved at runtime into a dictionary of parameters mapping name to value. These parameters are stored by the Prefect orchestration engine on the flow run object.

    Prefect API requires keyword arguments

    When creating flow runs from the Prefect API, parameter names must be specified when overriding defaults \u2014 they cannot be positional.

    Type hints provide an easy way to enforce typing on your flow parameters via pydantic. This means any pydantic model used as a type hint within a flow will be coerced automatically into the relevant object type:

    from prefect import flow\nfrom pydantic import BaseModel\n\nclass Model(BaseModel):\n    a: int\n    b: float\n    c: str\n\n@flow\ndef model_validator(model: Model):\n    print(model)\n

    Note that parameter values can be provided to a flow via API using a deployment. Flow run parameters sent to the API on flow calls are coerced to a serializable form. Type hints on your flow functions provide you a way of automatically coercing JSON provided values to their appropriate Python representation.

    For example, to automatically convert something to a datetime:

    from prefect import flow\nfrom datetime import datetime\n\n@flow\ndef what_day_is_it(date: datetime = None):\n    if date is None:\n        date = datetime.utcnow()\n    print(f\"It was {date.strftime('%A')} on {date.isoformat()}\")\n\nwhat_day_is_it(\"2021-01-01T02:00:19.180906\")\n# It was Friday on 2021-01-01T02:00:19.180906\n

    Parameters are validated before a flow is run. If a flow call receives invalid parameters, a flow run is created in a Failed state. If a flow run for a deployment receives invalid parameters, it will move from a Pending state to a Failed without entering a Running state.

    Flow run parameters cannot exceed 512kb in size

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#final-state-determination","title":"Final state determination","text":"

    Prerequisite

    Read the documentation about states before proceeding with this section.

    The final state of the flow is determined by its return value. The following rules apply:

    • If an exception is raised directly in the flow function, the flow run is marked as failed.
    • If the flow does not return a value (or returns None), its state is determined by the states of all of the tasks and subflows within it.
    • If any task run or subflow run failed, then the final flow run state is marked as FAILED.
    • If any task run was cancelled, then the final flow run state is marked as CANCELLED.
    • If a flow returns a manually created state, it is used as the state of the final flow run. This allows for manual determination of final state.
    • If the flow run returns any other object, then it is marked as completed.

    The following examples illustrate each of these cases:

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#raise-an-exception","title":"Raise an exception","text":"

    If an exception is raised within the flow function, the flow is immediately marked as failed.

    from prefect import flow\n\n@flow\ndef always_fails_flow():\n    raise ValueError(\"This flow immediately fails\")\n\nalways_fails_flow()\n

    Running this flow produces the following result:

    22:22:36.864 | INFO    | prefect.engine - Created flow run 'acrid-tuatara' for flow 'always-fails-flow'\n22:22:36.864 | INFO    | Flow run 'acrid-tuatara' - Starting 'ConcurrentTaskRunner'; submitted tasks will be run concurrently...\n22:22:37.060 | ERROR   | Flow run 'acrid-tuatara' - Encountered exception during execution:\nTraceback (most recent call last):...\nValueError: This flow immediately fails\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-none","title":"Return none","text":"

    A flow with no return statement is determined by the state of all of its task runs.

    from prefect import flow, task\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I fail successfully\")\n\n@task\ndef always_succeeds_task():\n    print(\"I'm fail safe!\")\n    return \"success\"\n\n@flow\ndef always_fails_flow():\n    always_fails_task.submit().result(raise_on_failure=False)\n    always_succeeds_task()\n\nif __name__ == \"__main__\":\n    always_fails_flow()\n

    Running this flow produces the following result:

    18:32:05.345 | INFO    | prefect.engine - Created flow run 'auburn-lionfish' for flow 'always-fails-flow'\n18:32:05.346 | INFO    | Flow run 'auburn-lionfish' - Starting 'ConcurrentTaskRunner'; submitted tasks will be run concurrently...\n18:32:05.582 | INFO    | Flow run 'auburn-lionfish' - Created task run 'always_fails_task-96e4be14-0' for task 'always_fails_task'\n18:32:05.582 | INFO    | Flow run 'auburn-lionfish' - Submitted task run 'always_fails_task-96e4be14-0' for execution.\n18:32:05.610 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Encountered exception during execution:\nTraceback (most recent call last):\n  ...\nValueError: I fail successfully\n18:32:05.638 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Finished in state Failed('Task run encountered an exception.')\n18:32:05.658 | INFO    | Flow run 'auburn-lionfish' - Created task run 'always_succeeds_task-9c27db32-0' for task 'always_succeeds_task'\n18:32:05.659 | INFO    | Flow run 'auburn-lionfish' - Executing 'always_succeeds_task-9c27db32-0' immediately...\nI'm fail safe!\n18:32:05.703 | INFO    | Task run 'always_succeeds_task-9c27db32-0' - Finished in state Completed()\n18:32:05.730 | ERROR   | Flow run 'auburn-lionfish' - Finished in state Failed('1/2 states failed.')\nTraceback (most recent call last):\n  ...\nValueError: I fail successfully\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-a-future","title":"Return a future","text":"

    If a flow returns one or more futures, the final state is determined based on the underlying states.

    from prefect import flow, task\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I fail successfully\")\n\n@task\ndef always_succeeds_task():\n    print(\"I'm fail safe!\")\n    return \"success\"\n\n@flow\ndef always_succeeds_flow():\n    x = always_fails_task.submit().result(raise_on_failure=False)\n    y = always_succeeds_task.submit(wait_for=[x])\n    return y\n\nif __name__ == \"__main__\":\n    always_succeeds_flow()\n

    Running this flow produces the following result \u2014 it succeeds because it returns the future of the task that succeeds:

    18:35:24.965 | INFO    | prefect.engine - Created flow run 'whispering-guan' for flow 'always-succeeds-flow'\n18:35:24.965 | INFO    | Flow run 'whispering-guan' - Starting 'ConcurrentTaskRunner'; submitted tasks will be run concurrently...\n18:35:25.204 | INFO    | Flow run 'whispering-guan' - Created task run 'always_fails_task-96e4be14-0' for task 'always_fails_task'\n18:35:25.205 | INFO    | Flow run 'whispering-guan' - Submitted task run 'always_fails_task-96e4be14-0' for execution.\n18:35:25.232 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Encountered exception during execution:\nTraceback (most recent call last):\n  ...\nValueError: I fail successfully\n18:35:25.265 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Finished in state Failed('Task run encountered an exception.')\n18:35:25.289 | INFO    | Flow run 'whispering-guan' - Created task run 'always_succeeds_task-9c27db32-0' for task 'always_succeeds_task'\n18:35:25.289 | INFO    | Flow run 'whispering-guan' - Submitted task run 'always_succeeds_task-9c27db32-0' for execution.\nI'm fail safe!\n18:35:25.335 | INFO    | Task run 'always_succeeds_task-9c27db32-0' - Finished in state Completed()\n18:35:25.362 | INFO    | Flow run 'whispering-guan' - Finished in state Completed('All states completed.')\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-multiple-states-or-futures","title":"Return multiple states or futures","text":"

    If a flow returns a mix of futures and states, the final state is determined by resolving all futures to states, then determining if any of the states are not COMPLETED.

    from prefect import task, flow\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I am bad task\")\n\n@task\ndef always_succeeds_task():\n    return \"foo\"\n\n@flow\ndef always_succeeds_flow():\n    return \"bar\"\n\n@flow\ndef always_fails_flow():\n    x = always_fails_task()\n    y = always_succeeds_task()\n    z = always_succeeds_flow()\n    return x, y, z\n

    Running this flow produces the following result. It fails because one of the three returned futures failed. Note that the final state is Failed, but the states of each of the returned futures is included in the flow state:

    20:57:51.547 | INFO    | prefect.engine - Created flow run 'impartial-gorilla' for flow 'always-fails-flow'\n20:57:51.548 | INFO    | Flow run 'impartial-gorilla' - Using task runner 'ConcurrentTaskRunner'\n20:57:51.645 | INFO    | Flow run 'impartial-gorilla' - Created task run 'always_fails_task-58ea43a6-0' for task 'always_fails_task'\n20:57:51.686 | INFO    | Flow run 'impartial-gorilla' - Created task run 'always_succeeds_task-c9014725-0' for task 'always_succeeds_task'\n20:57:51.727 | ERROR   | Task run 'always_fails_task-58ea43a6-0' - Encountered exception during execution:\nTraceback (most recent call last):...\nValueError: I am bad task\n20:57:51.787 | INFO    | Task run 'always_succeeds_task-c9014725-0' - Finished in state Completed()\n20:57:51.808 | INFO    | Flow run 'impartial-gorilla' - Created subflow run 'unbiased-firefly' for flow 'always-succeeds-flow'\n20:57:51.884 | ERROR   | Task run 'always_fails_task-58ea43a6-0' - Finished in state Failed('Task run encountered an exception.')\n20:57:52.438 | INFO    | Flow run 'unbiased-firefly' - Finished in state Completed()\n20:57:52.811 | ERROR   | Flow run 'impartial-gorilla' - Finished in state Failed('1/3 states failed.')\nFailed(message='1/3 states failed.', type=FAILED, result=(Failed(message='Task run encountered an exception.', type=FAILED, result=ValueError('I am bad task'), task_run_id=5fd4c697-7c4c-440d-8ebc-dd9c5bbf2245), Completed(message=None, type=COMPLETED, result='foo', task_run_id=df9b6256-f8ac-457c-ba69-0638ac9b9367), Completed(message=None, type=COMPLETED, result='bar', task_run_id=cfdbf4f1-dccd-4816-8d0f-128750017d0c)), flow_run_id=6d2ec094-001a-4cb0-a24e-d2051db6318d)\n

    Returning multiple states

    When returning multiple states, they must be contained in a set, list, or tuple. If other collection types are used, the result of the contained states will not be checked.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-a-manual-state","title":"Return a manual state","text":"

    If a flow returns a manually created state, the final state is determined based on the return value.

    from prefect import task, flow\nfrom prefect.states import Completed, Failed\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I fail successfully\")\n\n@task\ndef always_succeeds_task():\n    print(\"I'm fail safe!\")\n    return \"success\"\n\n@flow\ndef always_succeeds_flow():\n    x = always_fails_task.submit()\n    y = always_succeeds_task.submit()\n    if y.result() == \"success\":\n        return Completed(message=\"I am happy with this result\")\n    else:\n        return Failed(message=\"How did this happen!?\")\n\nif __name__ == \"__main__\":\n    always_succeeds_flow()\n

    Running this flow produces the following result.

    18:37:42.844 | INFO    | prefect.engine - Created flow run 'lavender-elk' for flow 'always-succeeds-flow'\n18:37:42.845 | INFO    | Flow run 'lavender-elk' - Starting 'ConcurrentTaskRunner'; submitted tasks will be run concurrently...\n18:37:43.125 | INFO    | Flow run 'lavender-elk' - Created task run 'always_fails_task-96e4be14-0' for task 'always_fails_task'\n18:37:43.126 | INFO    | Flow run 'lavender-elk' - Submitted task run 'always_fails_task-96e4be14-0' for execution.\n18:37:43.162 | INFO    | Flow run 'lavender-elk' - Created task run 'always_succeeds_task-9c27db32-0' for task 'always_succeeds_task'\n18:37:43.163 | INFO    | Flow run 'lavender-elk' - Submitted task run 'always_succeeds_task-9c27db32-0' for execution.\n18:37:43.175 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Encountered exception during execution:\nTraceback (most recent call last):\n  ...\nValueError: I fail successfully\nI'm fail safe!\n18:37:43.217 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Finished in state Failed('Task run encountered an exception.')\n18:37:43.236 | INFO    | Task run 'always_succeeds_task-9c27db32-0' - Finished in state Completed()\n18:37:43.264 | INFO    | Flow run 'lavender-elk' - Finished in state Completed('I am happy with this result')\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-an-object","title":"Return an object","text":"

    If the flow run returns any other object, then it is marked as completed.

    from prefect import task, flow\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I fail successfully\")\n\n@flow\ndef always_succeeds_flow():\n    always_fails_task().submit()\n    return \"foo\"\n\nif __name__ == \"__main__\":\n    always_succeeds_flow()\n

    Running this flow produces the following result.

    21:02:45.715 | INFO    | prefect.engine - Created flow run 'sparkling-pony' for flow 'always-succeeds-flow'\n21:02:45.715 | INFO    | Flow run 'sparkling-pony' - Using task runner 'ConcurrentTaskRunner'\n21:02:45.816 | INFO    | Flow run 'sparkling-pony' - Created task run 'always_fails_task-58ea43a6-0' for task 'always_fails_task'\n21:02:45.853 | ERROR   | Task run 'always_fails_task-58ea43a6-0' - Encountered exception during execution:\nTraceback (most recent call last):...\nValueError: I am bad task\n21:02:45.879 | ERROR   | Task run 'always_fails_task-58ea43a6-0' - Finished in state Failed('Task run encountered an exception.')\n21:02:46.593 | INFO    | Flow run 'sparkling-pony' - Finished in state Completed()\nCompleted(message=None, type=COMPLETED, result='foo', flow_run_id=7240e6f5-f0a8-4e00-9440-a7b33fb51153)\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#serving-a-flow","title":"Serving a flow","text":"

    The simplest way to create a deployment for your flow is by calling its serve method. This method creates a deployment for the flow and starts a long-running process that monitors for work from the Prefect server. When work is found, it is executed within its own isolated subprocess.

    hello_world.py
    from prefect import flow\n\n\n@flow(log_prints=True)\ndef hello_world(name: str = \"world\", goodbye: bool = False):\n    print(f\"Hello {name} from Prefect! \ud83e\udd17\")\n\n    if goodbye:\n        print(f\"Goodbye {name}!\")\n\n\nif __name__ == \"__main__\":\n    # creates a deployment and stays running to monitor for work instructions generated on the server\n\n    hello_world.serve(name=\"my-first-deployment\",\n                      tags=[\"onboarding\"],\n                      parameters={\"goodbye\": True},\n                      interval=60)\n

    This interface provides all of the configuration needed for a deployment with no strong infrastructure requirements:

    • schedules
    • event triggers
    • metadata such as tags and description
    • default parameter values

    Schedules are auto-paused on shutdown

    By default, stopping the process running flow.serve will pause the schedule for the deployment (if it has one). When running this in environments where restarts are expected use the pause_on_shutdown=False flag to prevent this behavior:

    if __name__ == \"__main__\":\n    hello_world.serve(name=\"my-first-deployment\",\n                      tags=[\"onboarding\"],\n                      parameters={\"goodbye\": True},\n                      pause_on_shutdown=False,\n                      interval=60)\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#serving-multiple-flows-at-once","title":"Serving multiple flows at once","text":"

    You can take this further and serve multiple flows with the same process using the serve utility along with the to_deployment method of flows:

    import time\nfrom prefect import flow, serve\n\n\n@flow\ndef slow_flow(sleep: int = 60):\n    \"Sleepy flow - sleeps the provided amount of time (in seconds).\"\n    time.sleep(sleep)\n\n\n@flow\ndef fast_flow():\n    \"Fastest flow this side of the Mississippi.\"\n    return\n\n\nif __name__ == \"__main__\":\n    slow_deploy = slow_flow.to_deployment(name=\"sleeper\", interval=45)\n    fast_deploy = fast_flow.to_deployment(name=\"fast\")\n    serve(slow_deploy, fast_deploy)\n

    The behavior and interfaces are identical to the single flow case.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#retrieve-a-flow-from-remote-storage","title":"Retrieve a flow from remote storage","text":"

    Flows can be retrieved from remote storage using the flow.from_source method.

    flow.from_source accepts a git repository URL and an entrypoint pointing to the flow to load from the repository:

    load_from_url.py
    from prefect import flow\n\nmy_flow = flow.from_source(\n    source=\"https://github.com/org/repo.git\",\n    entrypoint=\"flows.py:my_flow\"\n)\n\nif __name__ == \"__main__\":\n    my_flow()\n

    A flow entrypoint is the path to the file the flow is located in and the name of the flow function separated by a colon.

    If you need additional configuration, such as specifying a private repository, you can provide a GitRepository instead of URL:

    load_from_storage.py
    from prefect import flow\nfrom prefect.runner.storage import GitRepository\nfrom prefect.blocks.system import Secret\n\nmy_flow = flow.from_source(\n    source=GitRepository(\n        url=\"https://github.com/org/private-repo.git\",\n        branch=\"dev\",\n        credentials={\n            \"access_token\": Secret.load(\"github-access-token\").get()\n        }\n    ),\n    entrypoint=\"flows.py:my_flow\"\n)\n\nif __name__ == \"__main__\":\n    my_flow()\n

    You can serve loaded flows

    Flows loaded from remote storage can be served using the same serve method as local flows:

    serve_loaded_flow.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=\"https://github.com/org/repo.git\",\n        entrypoint=\"flows.py:my_flow\"\n    ).serve(name=\"my-deployment\")\n

    When you serve a flow loaded from remote storage, the serving process will periodically poll your remote storage for updates to the flow's code. This pattern allows you to update your flow code without restarting the serving process.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#pausing-or-suspending-a-flow-run","title":"Pausing or suspending a flow run","text":"

    Prefect provides you with the ability to halt a flow run with two functions that are similar, but slightly different. When a flow run is paused, code execution is stopped and the process continues to run. When a flow run is suspended, code execution is stopped and so is the process.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#pausing-a-flow-run","title":"Pausing a flow run","text":"

    Prefect enables pausing an in-progress flow run for manual approval. Prefect exposes this functionality via the pause_flow_run and resume_flow_run functions, as well as the Prefect UI.

    Timeouts

    Paused flow runs time out after one hour by default. After the timeout, the flow run will fail with a message saying it paused and never resumed. You can specify a different timeout period in seconds using the timeout parameter.

    Most simply, pause_flow_run can be called inside a flow:

    from prefect import task, flow, pause_flow_run, resume_flow_run\n\n@task\nasync def marvin_setup():\n    return \"a raft of ducks walk into a bar...\"\n\n\n@task\nasync def marvin_punchline():\n    return \"it's a wonder none of them ducked!\"\n\n\n@flow\nasync def inspiring_joke():\n    await marvin_setup()\n    await pause_flow_run(timeout=600)  # pauses for 10 minutes\n    await marvin_punchline()\n

    You can also implement conditional pauses:

    from prefect import task, flow, pause_flow_run\n\n@task\ndef task_one():\n    for i in range(3):\n        sleep(1)\n        print(i)\n\n@flow\ndef my_flow():\n    terminal_state = task_one.submit(return_state=True)\n    if terminal_state.type == StateType.COMPLETED:\n        print(\"Task one succeeded! Pausing flow run..\")\n        pause_flow_run(timeout=2) \n    else:\n        print(\"Task one failed. Skipping pause flow run..\")\n

    Calling this flow will block code execution after the first task and wait for resumption to deliver the punchline.

    await inspiring_joke()\n> \"a raft of ducks walk into a bar...\"\n

    Paused flow runs can be resumed by clicking the Resume button in the Prefect UI or calling the resume_flow_run utility via client code.

    resume_flow_run(FLOW_RUN_ID)\n

    The paused flow run will then finish!

    > \"it's a wonder none of them ducked!\"\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#suspending-a-flow-run","title":"Suspending a flow run","text":"

    Similar to pausing a flow run, Prefect enables suspending an in-progress flow run.

    The difference between pausing and suspending a flow run

    There is an important difference between pausing and suspending a flow run. When you pause a flow run, the flow code is still running but is blocked until someone resumes the flow. This is not the case with suspending a flow run! When you suspend a flow run, the flow exits completely and the infrastructure running it (e.g., a Kubernetes Job) tears down.

    This means that you can suspend flow runs to save costs instead of paying for long-running infrastructure. However, when the flow run resumes, the flow code will execute again from the beginning of the flow, so you should use tasks and task caching to avoid recomputing expensive operations.

    Prefect exposes this functionality via the suspend_flow_run and resume_flow_run functions, as well as the Prefect UI.

    When called inside of a flow suspend_flow_run will immediately suspend execution of the flow run. The flow run will be marked as Suspended and will not be resumed until resume_flow_run is called.

    Timeouts

    Suspended flow runs time out after one hour by default. After the timeout, the flow run will fail with a message saying it suspended and never resumed. You can specify a different timeout period in seconds using the timeout parameter or pass timeout=None for no timeout.

    Here is an example of a flow that does not block flow execution while paused. This flow will exit after one task, and will be rescheduled upon resuming. The stored result of the first task is retrieved instead of being rerun.

    from prefect import flow, pause_flow_run, task\n\n@task(persist_result=True)\ndef foo():\n    return 42\n\n@flow(persist_result=True)\ndef noblock_pausing():\n    x = foo.submit()\n    pause_flow_run(timeout=30, reschedule=True)\n    y = foo.submit()\n    z = foo(wait_for=[x])\n    alpha = foo(wait_for=[y])\n    omega = foo(wait_for=[x, y])\n

    Flow runs can be suspended out-of-process by calling suspend_flow_run(flow_run_id=<ID>) or selecting the Suspend button in the Prefect UI or Prefect Cloud.

    Suspended flow runs can be resumed by clicking the Resume button in the Prefect UI or calling the resume_flow_run utility via client code.

    resume_flow_run(FLOW_RUN_ID)\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#waiting-for-input-when-pausing-or-suspending-a-flow-run","title":"Waiting for input when pausing or suspending a flow run","text":"

    Experimental

    The wait_for_input parameter used in the pause_flow_run or suspend_flow_run functions is an experimental feature. The interface or behavior of this feature may change without warning in future releases.

    If you encounter any issues, please let us know in Slack or with a Github issue.

    When pausing or suspending a flow run you may want to wait for input from a user. Prefect provides a way to do this by leveraging the pause_flow_run and suspend_flow_run functions. These functions accept a wait_for_input argument, the value of which should be a subclass of prefect.input.RunInput, a pydantic model. When resuming the flow run, users are required to provide data for this model. Upon successful validation, the flow run resumes, and the return value of the pause_flow_run or suspend_flow_run is an instance of the model containing the provided data.

    Here is an example of a flow that pauses and waits for input from a user:

    from prefect import flow, get_run_logger, pause_flow_run\nfrom prefect.input import RunInput\n\n\nclass UserNameInput(RunInput):\n    name: str\n\n\n@flow\nasync def greet_user():\n    logger = get_run_logger()\n\n    user_input = await pause_flow_run(\n        wait_for_input=UserNameInput\n    )\n\n    logger.info(f\"Hello, {user_input.name}!\")\n
    Running this flow will create a flow run. The flow run will advance until code execution reaches pause_flow_run, at which point it will move into a Paused state. Execution will block and wait for resumption.

    When resuming the flow run, users will be prompted to provide a value for the name field of the UserNameInput model. Upon successful validation, the flow run will resume, and the return value of the pause_flow_run will be an instance of the UserNameInput model containing the provided data.

    For more in-depth information on receiving input from users when pausing and suspending flow runs, see the Creating human in the loop workflows guide.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#canceling-a-flow-run","title":"Canceling a flow run","text":"

    You may cancel a scheduled or in-progress flow run from the CLI, UI, REST API, or Python client.

    When cancellation is requested, the flow run is moved to a \"Cancelling\" state. If the deployment is a work pool-based deployemnt with a worker, then the worker monitors the state of flow runs and detects that cancellation has been requested. The worker then sends a signal to the flow run infrastructure, requesting termination of the run. If the run does not terminate after a grace period (default of 30 seconds), the infrastructure will be killed, ensuring the flow run exits.

    A deployment is required

    Flow run cancellation requires the flow run to be associated with a deployment. A monitoring process must be running to enforce the cancellation. Inline subflow runs, i.e. those created without run_deployment, cannot be cancelled without cancelling the parent flow run. If you may need to cancel a subflow run independent of its parent flow run, we recommend deploying it separately and starting it using the run_deployment function.

    Cancellation is robust to restarts of Prefect workers. To enable this, we attach metadata about the created infrastructure to the flow run. Internally, this is referred to as the infrastructure_pid or infrastructure identifier. Generally, this is composed of two parts:

    1. Scope: identifying where the infrastructure is running.
    2. ID: a unique identifier for the infrastructure within the scope.

    The scope is used to ensure that Prefect does not kill the wrong infrastructure. For example, workers running on multiple machines may have overlapping process IDs but should not have a matching scope.

    The identifiers for infrastructure types:

    • Processes: The machine hostname and the PID.
    • Docker Containers: The Docker API URL and container ID.
    • Kubernetes Jobs: The Kubernetes cluster name and the job name.

    While the cancellation process is robust, there are a few issues than can occur:

    • If the infrastructure block for the flow run has been removed or altered, cancellation may not work.
    • If the infrastructure block for the flow run does not have support for cancellation, cancellation will not work.
    • If the identifier scope does not match when attempting to cancel a flow run the worker will be unable to cancel the flow run. Another worker may attempt cancellation.
    • If the infrastructure associated with the run cannot be found or has already been killed, the worker will mark the flow run as cancelled.
    • If the infrastructre_pid is missing from the flow run will be marked as cancelled but cancellation cannot be enforced.
    • If the worker runs into an unexpected error during cancellation the flow run may or may not be cancelled depending on where the error occurred. The worker will try again to cancel the flow run. Another worker may attempt cancellation.

    Enhanced cancellation

    We are working on improving cases where cancellation can fail. You can try the improved cancellation experience by enabling the PREFECT_EXPERIMENTAL_ENABLE_ENHANCED_CANCELLATION setting on your worker or agents:

    prefect config set PREFECT_EXPERIMENTAL_ENABLE_ENHANCED_CANCELLATION=True\n

    If you encounter any issues, please let us know in SlackSlack or with a Github issue.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#cancel-via-the-cli","title":"Cancel via the CLI","text":"

    From the command line in your execution environment, you can cancel a flow run by using the prefect flow-run cancel CLI command, passing the ID of the flow run.

    prefect flow-run cancel 'a55a4804-9e3c-4042-8b59-b3b6b7618736'\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#cancel-via-the-ui","title":"Cancel via the UI","text":"

    From the UI you can cancel a flow run by navigating to the flow run's detail page and clicking the Cancel button in the upper right corner.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/infrastructure/","title":"Infrastructure","text":"

    Workers are recommended

    Infrastructure blocks are part of the agent based deployment model. Work Pools and Workers simplify the specification of each flow's infrastructure and runtime environment. If you have existing agents, you can upgrade from agents to workers to significantly enhance the experience of deploying flows.

    Users may specify an infrastructure block when creating a deployment. This block will be used to specify infrastructure for flow runs created by the deployment at runtime.

    Infrastructure can only be used with a deployment. When you run a flow directly by calling the flow yourself, you are responsible for the environment in which the flow executes.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#infrastructure-overview","title":"Infrastructure overview","text":"

    Prefect uses infrastructure to create the environment for a user's flow to execute.

    Infrastructure is attached to a deployment and is propagated to flow runs created for that deployment. Infrastructure is deserialized by the agent and it has two jobs:

    • Create execution environment infrastructure for the flow run.
    • Run a Python command to start the prefect.engine in the infrastructure, which retrieves the flow from storage and executes the flow.

    The engine acquires and calls the flow. Infrastructure doesn't know anything about how the flow is stored, it's just passing a flow run ID to the engine.

    Infrastructure is specific to the environments in which flows will run. Prefect currently provides the following infrastructure types:

    • Process runs flows in a local subprocess.
    • DockerContainer runs flows in a Docker container.
    • KubernetesJob runs flows in a Kubernetes Job.
    • ECSTask runs flows in an Amazon ECS Task.
    • Cloud Run runs flows in a Google Cloud Run Job.
    • Container Instance runs flows in an Azure Container Instance.

    What about tasks?

    Flows and tasks can both use configuration objects to manage the environment in which code runs.

    Flows use infrastructure.

    Tasks use task runners. For more on how task runners work, see Task Runners.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#using-infrastructure","title":"Using infrastructure","text":"

    You may create customized infrastructure blocks through the Prefect UI or Prefect Cloud Blocks page or create them in code and save them to the API using the blocks .save() method.

    Once created, there are two distinct ways to use infrastructure in a deployment:

    • Starting with Prefect defaults \u2014 this is what happens when you pass the -i or --infra flag and provide a type when building deployment files.
    • Pre-configure infrastructure settings as blocks and base your deployment infrastructure on those settings \u2014 by passing -ib or --infra-block and a block slug when building deployment files.

    For example, when creating your deployment files, the supported Prefect infrastrucure types are:

    • process
    • docker-container
    • kubernetes-job
    • ecs-task
    • cloud-run-job
    • container-instance-job
    $ prefect deployment build ./my_flow.py:my_flow -n my-flow-deployment -t test -i docker-container -sb s3/my-bucket --override env.EXTRA_PIP_PACKAGES=s3fs\nFound flow 'my-flow'\nSuccessfully uploaded 2 files to s3://bucket-full-of-sunshine\nDeployment YAML created at '/Users/terry/test/flows/infra/deployment.yaml'.\n

    In this example we specify the DockerContainer infrastructure in addition to a preconfigured AWS S3 bucket storage block.

    The default deployment YAML filename may be edited as needed to add an infrastructure type or infrastructure settings.

    ###\n### A complete description of a Prefect Deployment for flow 'my-flow'\n###\nname: my-flow-deployment\ndescription: null\nversion: e29de5d01b06d61b4e321d40f34a480c\n# The work queue that will handle this deployment's runs\nwork_queue_name: default\nwork_pool_name: default-agent-pool\ntags:\n- test\nparameters: {}\nschedule: null\nis_schedule_active: true\ninfra_overrides:\n  env.EXTRA_PIP_PACKAGES: s3fs\ninfrastructure:\n  type: docker-container\n  env: {}\n  labels: {}\n  name: null\n  command:\n  - python\n  - -m\n  - prefect.engine\n  image: prefecthq/prefect:2-latest\n  image_pull_policy: null\n  networks: []\n  network_mode: null\n  auto_remove: false\n  volumes: []\n  stream_output: true\n  memswap_limit: null\n  mem_limit: null\n  privileged: false\n  block_type_slug: docker-container\n  _block_type_slug: docker-container\n\n###\n### DO NOT EDIT BELOW THIS LINE\n###\nflow_name: my-flow\nmanifest_path: my_flow-manifest.json\nstorage:\n  bucket_path: bucket-full-of-sunshine\n  aws_access_key_id: '**********'\n  aws_secret_access_key: '**********'\n  _is_anonymous: true\n  _block_document_name: anonymous-xxxxxxxx-f1ff-4265-b55c-6353a6d65333\n  _block_document_id: xxxxxxxx-06c2-4c3c-a505-4a8db0147011\n  block_type_slug: s3\n  _block_type_slug: s3\npath: ''\nentrypoint: my_flow.py:my-flow\nparameter_openapi_schema:\n  title: Parameters\n  type: object\n  properties: {}\n  required: null\n  definitions: null\ntimestamp: '2023-02-08T23:00:14.974642+00:00'\n

    Editing deployment YAML

    Note the big DO NOT EDIT comment in the deployment YAML: In practice, anything above this block can be freely edited before running prefect deployment apply to create the deployment on the API.

    Once the deployment exists, any flow runs that this deployment starts will use DockerContainer infrastructure.

    You can also create custom infrastructure blocks \u2014 either in the Prefect UI for in code via the API \u2014 and use the settings in the block to configure your infastructure. For example, here we specify settings for Kubernetes infrastructure in a block named k8sdev.

    from prefect.infrastructure import KubernetesJob, KubernetesImagePullPolicy\n\nk8s_job = KubernetesJob(\n    namespace=\"dev\",\n    image=\"prefecthq/prefect:2.0.0-python3.9\",\n    image_pull_policy=KubernetesImagePullPolicy.IF_NOT_PRESENT,\n)\nk8s_job.save(\"k8sdev\")\n

    Now we can apply the infrastrucure type and settings in the block by specifying the block slug kubernetes-job/k8sdev as the infrastructure type when building a deployment:

    prefect deployment build flows/k8s_example.py:k8s_flow --name k8sdev --tag k8s -sb s3/dev -ib kubernetes-job/k8sdev\n

    See Deployments for more information about deployment build options.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#configuring-infrastructure","title":"Configuring infrastructure","text":"

    Every infrastrcture type has type-specific options.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#process","title":"Process","text":"

    Process infrastructure runs a command in a new process.

    Current environment variables and Prefect settings will be included in the created process. Configured environment variables will override any current environment variables.

    Process supports the following settings:

    Attributes Description command A list of strings specifying the command to start the flow run. In most cases you should not override this. env Environment variables to set for the new process. labels Labels for the process. Labels are for metadata purposes only and cannot be attached to the process itself. name A name for the process. For display purposes only.","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#dockercontainer","title":"DockerContainer","text":"

    DockerContainer infrastructure executes flow runs in a container.

    Requirements for DockerContainer:

    • Docker Engine must be available.
    • You must configure remote Storage. Local storage is not supported for Docker.
    • The API must be available from within the flow run container. To facilitate connections to locally hosted APIs, localhost and 127.0.0.1 will be replaced with host.docker.internal.
    • The ephemeral Prefect API won't work with Docker and Kubernetes. You must have a Prefect server or Prefect Cloud API endpoint set in your agent's configuration.

    DockerContainer supports the following settings:

    Attributes Description auto_remove Bool indicating whether the container will be removed on completion. If False, the container will remain after exit for inspection. command A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this. env Environment variables to set for the container. image An optional string specifying the name of a Docker image to use. Defaults to the Prefect image. If the image is stored anywhere other than a public Docker Hub registry, use a corresponding registry block, e.g. DockerRegistry or ensure otherwise that your execution layer is authenticated to pull the image from the image registry. image_pull_policy Specifies if the image should be pulled. One of 'ALWAYS', 'NEVER', 'IF_NOT_PRESENT'. image_registry A DockerRegistry block containing credentials to use if image is stored in a private image registry. labels An optional dictionary of labels, mapping name to value. name An optional name for the container. networks An optional list of strings specifying Docker networks to connect the container to. network_mode Set the network mode for the created container. Defaults to 'host' if a local API url is detected, otherwise the Docker default of 'bridge' is used. If 'networks' is set, this cannot be set. stream_output Bool indicating whether to stream output from the subprocess to local standard output. volumes An optional list of volume mount strings in the format of \"local_path:container_path\".

    Prefect automatically sets a Docker image matching the Python and Prefect version you're using at deployment time. You can see all available images at Docker Hub.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#kubernetesjob","title":"KubernetesJob","text":"

    KubernetesJob infrastructure executes flow runs in a Kubernetes Job.

    Requirements for KubernetesJob:

    • kubectl must be available.
    • You must configure remote Storage. Local storage is not supported for Kubernetes.
    • The ephemeral Prefect API won't work with Docker and Kubernetes. You must have a Prefect server or Prefect Cloud API endpoint set in your agent's configuration.

    The Prefect CLI command prefect kubernetes manifest server automatically generates a Kubernetes manifest with default settings for Prefect deployments. By default, it simply prints out the YAML configuration for a manifest. You can pipe this output to a file of your choice and edit as necessary.

    KubernetesJob supports the following settings:

    Attributes Description cluster_config An optional Kubernetes cluster config to use for this job. command A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this. customizations A list of JSON 6902 patches to apply to the base Job manifest. Alternatively, a valid JSON string is allowed (handy for deployments CLI). env Environment variables to set for the container. finished_job_ttl The number of seconds to retain jobs after completion. If set, finished jobs will be cleaned up by Kubernetes after the given delay. If None (default), jobs will need to be manually removed. image String specifying the tag of a Docker image to use for the Job. image_pull_policy The Kubernetes image pull policy to use for job containers. job The base manifest for the Kubernetes Job. job_watch_timeout_seconds Number of seconds to watch for job creation before timing out (defaults to None). labels Dictionary of labels to add to the Job. name An optional name for the job. namespace String signifying the Kubernetes namespace to use. pod_watch_timeout_seconds Number of seconds to watch for pod creation before timing out (default 60). service_account_name An optional string specifying which Kubernetes service account to use. stream_output Bool indicating whether to stream output from the subprocess to local standard output.","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#kubernetesjob-overrides-and-customizations","title":"KubernetesJob overrides and customizations","text":"

    When creating deployments using KubernetesJob infrastructure, the infra_overrides parameter expects a dictionary. For a KubernetesJob, the customizations parameter expects a list.

    Containers expect a list of objects, even if there is only one. For any patches applying to the container, the path value should be a list, for example: /spec/templates/spec/containers/0/resources

    A Kubernetes-Job infrastructure block defined in Python:

    customizations = [\n {\n     \"op\": \"add\",\n     \"path\": \"/spec/template/spec/containers/0/resources\",\n     \"value\": {\n         \"requests\": {\n             \"cpu\": \"2000m\",\n             \"memory\": \"4gi\"\n         },\n         \"limits\": {\n             \"cpu\": \"4000m\",\n             \"memory\": \"8Gi\",\n             \"nvidia.com/gpu\": \"1\"\n      }\n  },\n }\n]\n\nk8s_job = KubernetesJob(\n        namespace=namespace,\n        image=image_name,\n        image_pull_policy=KubernetesImagePullPolicy.ALWAYS,\n        finished_job_ttl=300,\n        job_watch_timeout_seconds=600,\n        pod_watch_timeout_seconds=600,\n        service_account_name=\"prefect-server\",\n        customizations=customizations,\n    )\nk8s_job.save(\"devk8s\")\n

    A Deployment with infra-overrides defined in Python:

    infra_overrides={ \n    \"customizations\": [\n            {\n                \"op\": \"add\",\n                \"path\": \"/spec/template/spec/containers/0/resources\",\n                \"value\": {\n                    \"requests\": {\n                        \"cpu\": \"2000m\",\n                        \"memory\": \"4gi\"\n                    },\n                    \"limits\": {\n                        \"cpu\": \"4000m\",\n                        \"memory\": \"8Gi\",\n                        \"nvidia.com/gpu\": \"1\"\n                }\n            },\n        }\n    ]\n}\n\n# Load an already created K8s Block\nk8sjob = k8s_job.load(\"devk8s\")\n\ndeployment = Deployment.build_from_flow(\n    flow=my_flow,\n    name=\"s3-example\",\n    version=2,\n    work_queue_name=\"aws\",\n    infrastructure=k8sjob,\n    storage=storage,\n    infra_overrides=infra_overrides,\n)\n\ndeployment.apply()\n
    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#ecstask","title":"ECSTask","text":"

    ECSTask infrastructure runs your flow in an ECS Task.

    Requirements for ECSTask:

    • The ephemeral Prefect API won't work with ECS directly. You must have a Prefect server or Prefect Cloud API endpoint set in your agent's configuration.
    • The prefect-aws collection must be installed within the agent environment: pip install prefect-aws
    • The ECSTask and AwsCredentials blocks must be registered within the agent environment: prefect block register -m prefect_aws.ecs
    • You must configure remote Storage. Local storage is not supported for ECS tasks. The most commonly used type of storage with ECSTask is S3. If you leverage that type of block, make sure that s3fs is installed within your agent and flow run environment. The easiest way to satisfy all the installation-related points mentioned above is to include the following commands in your Dockerfile:
    FROM prefecthq/prefect:2-python3.9  # example base image \nRUN pip install s3fs prefect-aws\n

    Make sure to allocate enough CPU and memory to your agent, and consider adding retries

    When you start a Prefect agent on AWS ECS Fargate, allocate as much CPU and memory as needed for your workloads. Your agent needs enough resources to appropriately provision infrastructure for your flow runs and to monitor their execution. Otherwise, your flow runs may get stuck in a Pending state. Alternatively, set a work-queue concurrency limit to ensure that the agent will not try to process all runs at the same time.

    Some API calls to provision infrastructure may fail due to unexpected issues on the client side (for example, transient errors such as ConnectionError, HTTPClientError, or RequestTimeout), or due to server-side rate limiting from the AWS service. To mitigate those issues, we recommend adding environment variables such as AWS_MAX_ATTEMPTS (can be set to an integer value such as 10) and AWS_RETRY_MODE (can be set to a string value including standard or adaptive modes). Those environment variables must be added within the agent environment, e.g. on your ECS service running the agent, rather than on the ECSTask infrastructure block.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#docker-images","title":"Docker images","text":"

    Learn about options for Prefect-maintained Docker images in the Docker guide.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/results/","title":"Results","text":"

    Results represent the data returned by a flow or a task.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#retrieving-results","title":"Retrieving results","text":"

    When calling flows or tasks, the result is returned directly:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return 1\n\n@flow\ndef my_flow():\n    task_result = my_task()\n    return task_result + 1\n\nresult = my_flow()\nassert result == 2\n

    When working with flow and task states, the result can be retrieved with the State.result() method:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return 1\n\n@flow\ndef my_flow():\n    state = my_task(return_state=True)\n    return state.result() + 1\n\nstate = my_flow(return_state=True)\nassert state.result() == 2\n

    When submitting tasks to a runner, the result can be retrieved with the Future.result() method:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return 1\n\n@flow\ndef my_flow():\n    future = my_task.submit()\n    return future.result() + 1\n\nresult = my_flow()\nassert result == 2\n
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#handling-failures","title":"Handling failures","text":"

    Sometimes your flows or tasks will encounter an exception. Prefect captures all exceptions in order to report states to the orchestrator, but we do not hide them from you (unless you ask us to) as your program needs to know if an unexpected error has occurred.

    When calling flows or tasks, the exceptions are raised as in normal Python:

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    try:\n        my_task()\n    except ValueError:\n        print(\"Oh no! The task failed.\")\n\n    return True\n\nmy_flow()\n

    If you would prefer to check for a failed task without using try/except, you may ask Prefect to return the state:

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    state = my_task(return_state=True)\n\n    if state.is_failed():\n        print(\"Oh no! The task failed. Falling back to '1'.\")\n        result = 1\n    else:\n        result = state.result()\n\n    return result + 1\n\nresult = my_flow()\nassert result == 2\n

    If you retrieve the result from a failed state, the exception will be raised. For this reason, it's often best to check if the state is failed first.

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    state = my_task(return_state=True)\n\n    try:\n        result = state.result()\n    except ValueError:\n        print(\"Oh no! The state raised the error!\")\n\n    return True\n\nmy_flow()\n

    When retrieving the result from a state, you can ask Prefect not to raise exceptions:

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    state = my_task(return_state=True)\n\n    maybe_result = state.result(raise_on_failure=False)\n    if isinstance(maybe_result, ValueError):\n        print(\"Oh no! The task failed. Falling back to '1'.\")\n        result = 1\n    else:\n        result = maybe_result\n\n    return result + 1\n\nresult = my_flow()\nassert result == 2\n

    When submitting tasks to a runner, Future.result() works the same as State.result():

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    future = my_task.submit()\n\n    try:\n        future.result()\n    except ValueError:\n        print(\"Ah! Futures will raise the failure as well.\")\n\n    # You can ask it not to raise the exception too\n    maybe_result = future.result(raise_on_failure=False)\n    print(f\"Got {type(maybe_result)}\")\n\n    return True\n\nmy_flow()\n
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#working-with-async-results","title":"Working with async results","text":"

    When calling flows or tasks, the result is returned directly:

    import asyncio\nfrom prefect import flow, task\n\n@task\nasync def my_task():\n    return 1\n\n@flow\nasync def my_flow():\n    task_result = await my_task()\n    return task_result + 1\n\nresult = asyncio.run(my_flow())\nassert result == 2\n

    When working with flow and task states, the result can be retrieved with the State.result() method:

    import asyncio\nfrom prefect import flow, task\n\n@task\nasync def my_task():\n    return 1\n\n@flow\nasync def my_flow():\n    state = await my_task(return_state=True)\n    result = await state.result(fetch=True)\n    return result + 1\n\nasync def main():\n    state = await my_flow(return_state=True)\n    assert await state.result(fetch=True) == 2\n\nasyncio.run(main())\n

    Resolving results

    Prefect 2.6.0 added automatic retrieval of persisted results. Prior to this version, State.result() did not require an await. For backwards compatibility, when used from an asynchronous context, State.result() returns a raw result type.

    You may opt-in to the new behavior by passing fetch=True as shown in the example above. If you would like this behavior to be used automatically, you may enable the PREFECT_ASYNC_FETCH_STATE_RESULT setting. If you do not opt-in to this behavior, you will see a warning.

    You may also opt-out by setting fetch=False. This will silence the warning, but you will need to retrieve your result manually from the result type.

    When submitting tasks to a runner, the result can be retrieved with the Future.result() method:

    import asyncio\nfrom prefect import flow, task\n\n@task\nasync def my_task():\n    return 1\n\n@flow\nasync def my_flow():\n    future = await my_task.submit()\n    result = await future.result()\n    return result + 1\n\nresult = asyncio.run(my_flow())\nassert result == 2\n
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#persisting-results","title":"Persisting results","text":"

    The Prefect API does not store your results except in special cases. Instead, the result is persisted to a storage location in your infrastructure and Prefect stores a reference to the result.

    The following Prefect features require results to be persisted:

    • Task cache keys
    • Flow run retries

    If results are not persisted, these features may not be usable.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#configuring-persistence-of-results","title":"Configuring persistence of results","text":"

    Persistence of results requires a serializer and a storage location. Prefect sets defaults for these, and you should not need to adjust them until you want to customize behavior. You can configure results on the flow and task decorators with the following options:

    • persist_result: Whether the result should be persisted to storage.
    • result_storage: Where to store the result when persisted.
    • result_serializer: How to convert the result to a storable form.
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#toggling-persistence","title":"Toggling persistence","text":"

    Persistence of the result of a task or flow can be configured with the persist_result option. The persist_result option defaults to a null value, which will automatically enable persistence if it is needed for a Prefect feature used by the flow or task. Otherwise, persistence is disabled by default.

    For example, the following flow has retries enabled. Flow retries require that all task results are persisted, so the task's result will be persisted:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return \"hello world!\"\n\n@flow(retries=2)\ndef my_flow():\n    # This task does not have persistence toggled off and it is needed for the flow feature,\n    # so Prefect will persist its result at runtime\n    my_task()\n

    Flow retries do not require the flow's result to be persisted, so it will not be.

    In this next example, one task has caching enabled. Task caching requires that the given task's result is persisted:

    from prefect import flow, task\nfrom datetime import timedelta\n\n@task(cache_key_fn=lambda: \"always\", cache_expiration=timedelta(seconds=20))\ndef my_task():\n    # This task uses caching so its result will be persisted by default\n    return \"hello world!\"\n\n\n@task\ndef my_other_task():\n    ...\n\n@flow\ndef my_flow():\n    # This task uses a feature that requires result persistence\n    my_task()\n\n    # This task does not use a feature that requires result persistence and the\n    # flow does not use any features that require task result persistence so its\n    # result will not be persisted by default\n    my_other_task()\n

    Persistence of results can be manually toggled on or off:

    from prefect import flow, task\n\n@flow(persist_result=True)\ndef my_flow():\n    # This flow will persist its result even if not necessary for a feature.\n    ...\n\n@task(persist_result=False)\ndef my_task():\n    # This task will never persist its result.\n    # If persistence needed for a feature, an error will be raised.\n    ...\n

    Toggling persistence manually will always override any behavior that Prefect would infer.

    You may also change Prefect's default persistence behavior with the PREFECT_RESULTS_PERSIST_BY_DEFAULT setting. To persist results by default, even if they are not needed for a feature change the value to a truthy value:

    prefect config set PREFECT_RESULTS_PERSIST_BY_DEFAULT=true\n

    Task and flows with persist_result=False will not persist their results even if PREFECT_RESULTS_PERSIST_BY_DEFAULT is true.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-storage-location","title":"Result storage location","text":"

    The result storage location can be configured with the result_storage option. The result_storage option defaults to a null value, which infers storage from the context. Generally, this means that tasks will use the result storage configured on the flow unless otherwise specified. If there is no context to load the storage from and results must be persisted, results will be stored in the path specified by the PREFECT_LOCAL_STORAGE_PATH setting (defaults to ~/.prefect/storage).

    from prefect import flow, task\nfrom prefect.filesystems import LocalFileSystem, S3\n\n@flow(persist_result=True)\ndef my_flow():\n    my_task()  # This task will use the flow's result storage\n\n@task(persist_result=True)\ndef my_task():\n    ...\n\nmy_flow()  # The flow has no result storage configured and no parent, the local file system will be used.\n\n\n# Reconfigure the flow to use a different storage type\nnew_flow = my_flow.with_options(result_storage=S3(bucket_path=\"my-bucket\"))\n\nnew_flow()  # The flow and task within it will use S3 for result storage.\n

    You can configure this to use a specific storage using one of the following:

    • A storage instance, e.g. LocalFileSystem(basepath=\".my-results\")
    • A storage slug, e.g. 's3/dev-s3-block'
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-storage-key","title":"Result storage key","text":"

    The path of the result file in the result storage can be configured with the result_storage_key. The result_storage_key option defaults to a null value, which generates a unique identifier for each result.

    from prefect import flow, task\nfrom prefect.filesystems import LocalFileSystem, S3\n\n@flow(result_storage=S3(bucket_path=\"my-bucket\"))\ndef my_flow():\n    my_task()\n\n@task(persist_result=True, result_storage_key=\"my_task.json\")\ndef my_task():\n    ...\n\nmy_flow()  # The task's result will be persisted to 's3://my-bucket/my_task.json'\n

    Result storage keys are formatted with access to all of the modules in prefect.runtime and the run's parameters. In the following example, we will run a flow with three runs of the same task. Each task run will write its result to a unique file based on the name parameter.

    from prefect import flow, task\n\n@flow()\ndef my_flow():\n    hello_world()\n    hello_world(name=\"foo\")\n    hello_world(name=\"bar\")\n\n@task(persist_result=True, result_storage_key=\"hello-{parameters[name]}.json\")\ndef hello_world(name: str = \"world\"):\n    return f\"hello {name}\"\n\nmy_flow()\n

    After running the flow, we can see three persisted result files in our storage directory:

    $ ls ~/.prefect/storage | grep \"hello-\"\nhello-bar.json\nhello-foo.json\nhello-world.json\n

    In the next example, we include metadata about the flow run from the prefect.runtime.flow_run module:

    from prefect import flow, task\n\n@flow\ndef my_flow():\n    hello_world()\n\n@task(persist_result=True, result_storage_key=\"{flow_run.flow_name}_{flow_run.name}_hello.json\")\ndef hello_world(name: str = \"world\"):\n    return f\"hello {name}\"\n\nmy_flow()\n

    After running this flow, we can see a result file templated with the name of the flow and the flow run:

    \u276f ls ~/.prefect/storage | grep \"my-flow\"    \nmy-flow_industrious-trout_hello.json\n

    If a result exists at a given storage key in the storage location, it will be overwritten.

    Result storage keys can only be configured on tasks at this time.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-serializer","title":"Result serializer","text":"

    The result serializer can be configured with the result_serializer option. The result_serializer option defaults to a null value, which infers the serializer from the context. Generally, this means that tasks will use the result serializer configured on the flow unless otherwise specified. If there is no context to load the serializer from, the serializer defined by PREFECT_RESULTS_DEFAULT_SERIALIZER will be used. This setting defaults to Prefect's pickle serializer.

    You may configure the result serializer using:

    • A type name, e.g. \"json\" or \"pickle\" \u2014 this corresponds to an instance with default values
    • An instance, e.g. JSONSerializer(jsonlib=\"orjson\")
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#compressing-results","title":"Compressing results","text":"

    Prefect provides a CompressedSerializer which can be used to wrap other serializers to provide compression over the bytes they generate. The compressed serializer uses lzma compression by default. We test other compression schemes provided in the Python standard library such as bz2 and zlib, but you should be able to use any compression library that provides compress and decompress methods.

    You may configure compression of results using:

    • A type name, prefixed with compressed/ e.g. \"compressed/json\" or \"compressed/pickle\"
    • An instance e.g. CompressedSerializer(serializer=\"pickle\", compressionlib=\"lzma\")

    Note that the \"compressed/<serializer-type>\" shortcut will only work for serializers provided by Prefect. If you are using custom serializers, you must pass a full instance.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#storage-of-results-in-prefect","title":"Storage of results in Prefect","text":"

    The Prefect API does not store your results in most cases for the following reasons:

    • Results can be large and slow to send to and from the API.
    • Results often contain private information or data.
    • Results would need to be stored in the database or complex logic implemented to hydrate from another source.

    There are a few cases where Prefect will store your results directly in the database. This is an optimization to reduce the overhead of reading and writing to result storage.

    The following data types will be stored by the API without persistence to storage:

    • booleans (True, False)
    • nulls (None)

    If persist_result is set to False, these values will never be stored.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#tracking-results","title":"Tracking results","text":"

    The Prefect API tracks metadata about your results. The value of your result is only stored in specific cases. Result metadata can be seen in the UI on the \"Results\" page for flows.

    Prefect tracks the following result metadata:

    • Data type
    • Storage location (if persisted)
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#caching-of-results-in-memory","title":"Caching of results in memory","text":"

    When running your workflows, Prefect will keep the results of all tasks and flows in memory so they can be passed downstream. In some cases, it is desirable to override this behavior. For example, if you are returning a large amount of data from a task it can be costly to keep it memory for the entire duration of the flow run.

    Flows and tasks both include an option to drop the result from memory with cache_result_in_memory:

    @flow(cache_result_in_memory=False)\ndef foo():\n    return \"pretend this is large data\"\n\n@task(cache_result_in_memory=False)\ndef bar():\n    return \"pretend this is biiiig data\"\n

    When cache_result_in_memory is disabled, the result of your flow or task will be persisted by default. The result will then be pulled from storage when needed.

    @flow\ndef foo():\n    result = bar()\n    state = bar(return_state=True)\n\n    # The result will be retrieved from storage here\n    state.result()\n\n    future = bar.submit()\n    # The result will be retrieved from storage here\n    future.result()\n\n@task(cache_result_in_memory=False)\ndef bar():\n    # This result will persisted\n    return \"pretend this is biiiig data\"\n

    If both cache_result_in_memory and persistence are disabled, your results will not be available downstream.

    @task(persist_result=False, cache_result_in_memory=False)\ndef bar():\n    return \"pretend this is biiiig data\"\n\n@flow\ndef foo():\n    # Raises an error\n    result = bar()\n\n    # This is oaky\n    state = bar(return_state=True)\n\n    # Raises an error\n    state.result()\n\n    # This is okay\n    future = bar.submit()\n\n    # Raises an error\n    future.result()\n
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-storage-types","title":"Result storage types","text":"

    Result storage is responsible for reading and writing serialized data to an external location. At this time, any file system block can be used for result storage.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-serializer-types","title":"Result serializer types","text":"

    A result serializer is responsible for converting your Python object to and from bytes. This is necessary to store the object outside of Python and retrieve it later.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#pickle-serializer","title":"Pickle serializer","text":"

    Pickle is a standard Python protocol for encoding arbitrary Python objects. We supply a custom pickle serializer at prefect.serializers.PickleSerializer. Prefect's pickle serializer uses the cloudpickle project by default to support more object types. Alternative pickle libraries can be specified:

    from prefect.serializers import PickleSerializer\n\nPickleSerializer(picklelib=\"custompickle\")\n

    Benefits of the pickle serializer:

    • Many object types are supported.
    • Objects can define custom pickle support.

    Drawbacks of the pickle serializer:

    • When nested attributes of an object cannot be pickled, it is hard to determine the cause.
    • When deserializing objects, your Python and pickle library versions must match the one used at serialization time.
    • Serialized objects cannot be easily shared across different programming languages.
    • Serialized objects are not human readable.
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#json-serializer","title":"JSON serializer","text":"

    We supply a custom JSON serializer at prefect.serializers.JSONSerializer. Prefect's JSON serializer uses custom hooks by default to support more object types. Specifically, we add support for all types supported by Pydantic.

    By default, we use the standard Python json library. Alternative JSON libraries can be specified:

    from prefect.serializers import JSONSerializer\n\nJSONSerializer(jsonlib=\"orjson\")\n

    Benefits of the JSON serializer:

    • Serialized objects are human readable.
    • Serialized objects can often be shared across different programming languages.
    • Deserialization of serialized objects is generally version agnostic.

    Drawbacks of the JSON serializer:

    • Supported types are limited.
    • Implementing support for additional types must be done at the serializer level.
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-types","title":"Result types","text":"

    Prefect uses internal result types to capture information about the result attached to a state. The following types are used:

    • UnpersistedResult: Stores result metadata but the value is only available when created.
    • LiteralResult: Stores simple values inline.
    • PersistedResult: Stores a reference to a result persisted to storage.

    All result types include a get() method that can be called to return the value of the result. This is done behind the scenes when the result() method is used on states or futures.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#unpersisted-results","title":"Unpersisted results","text":"

    Unpersisted results are used to represent results that have not been and will not be persisted beyond the current flow run. The value associated with the result is stored in memory, but will not be available later. Result metadata is attached to this object for storage in the API and representation in the UI.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#literal-results","title":"Literal results","text":"

    Literal results are used to represent results stored in the Prefect database. The values contained by these results must always be JSON serializable.

    Example:

    result = LiteralResult(value=None)\nresult.json()\n# {\"type\": \"result\", \"value\": \"null\"}\n

    Literal results reduce the overhead required to persist simple results.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#persisted-results","title":"Persisted results","text":"

    The persisted result type contains all of the information needed to retrieve the result from storage. This includes:

    • Storage: A reference to the result storage that can be used to read the serialized result.
    • Key: Indicates where this specific result is in storage.

    Persisted result types also contain metadata for inspection without retrieving the result:

    • Serializer type: The name of the result serializer type.

    The get() method on result references retrieves the data from storage, deserializes it, and returns the original object. The get() operation will cache the resolved object to reduce the overhead of subsequent calls.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#persisted-result-blob","title":"Persisted result blob","text":"

    When results are persisted to storage, they are always written as a JSON document. The schema for this is described by the PersistedResultBlob type. The document contains:

    • The serialized data of the result.
    • A full description of result serializer that can be used to deserialize the result data.
    • The Prefect version used to create the result.
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/schedules/","title":"Schedules","text":"

    Scheduling is one of the primary reasons for using an orchestrator such as Prefect. Prefect allows you to use schedules to automatically create new flow runs for deployments.

    Prefect Cloud can also schedule flow runs through event-driven automations.

    Schedules tell the Prefect API how to create new flow runs for you automatically on a specified cadence.

    You can add a schedule to any flow deployment. The Prefect Scheduler service periodically reviews every deployment and creates new flow runs according to the schedule configured for the deployment.

    There are several recommended ways to create a schedule for a deployment:

    • Through the Prefect UI
    • Via a the cron, interval, or rrule parameters if building your deployment via the serve method of the Flow object or the serve utility for managing multiple flows simultaneously
    • If using worker-based deployments
    • Through the interactive prefect deploy command
    • With the deployments -> schedule section of the prefect.yaml file )
    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-the-ui","title":"Creating schedules through the UI","text":"

    You can add a schedule on the Deployments tab of the UI.

    Under Schedule select the Add button. If you don't see the Schedule section, expand the three dot menu on the top right of the page.

    Then select Interval or Cron to create a schedule.

    Once a schedule has been created, a number of scheduled flow runs will be visible in the UI. The schedule is viewable in human-friendly text on the Deployments page. You can edit the schedule by selecting the Edit button on the Deployment page.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#schedule-types","title":"Schedule types","text":"

    Prefect supports several types of schedules that cover a wide range of use cases and offer a large degree of customization:

    • Cron is most appropriate for users who are already familiar with cron from previous use.
    • Interval is best suited for deployments that need to run at some consistent cadence that isn't related to absolute time.
    • RRule is best suited for deployments that rely on calendar logic for simple recurring schedules, irregular intervals, exclusions, or day-of-month adjustments.
    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-the-serve-method","title":"Creating schedules through the serve method","text":"

    As seen in the Quickstart, you can create a schedule by passing a cron, interval, or rrule parameters to the Flow.serve method or the serve utility.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-the-interactive-prefect-deploy-command","title":"Creating schedules through the interactive prefect deploy command","text":"

    If you are using worker-based deployments, you can create a schedule through the interactive prefect deploy command. You will be prompted to choose which type of schedule to create.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-a-prefectyaml-files-deployments-schedule-section","title":"Creating schedules through a prefect.yaml file's deployments -> schedule section","text":"

    If you save the prefect.yaml file from the prefect deploy command, you will see it has a schedule section for your deployment. Alternatively, you can create a prefect.yaml file from a recipe or from scratch and add a schedule section to it.

    deployments:\n  ...\n  schedule:\n    cron: 0 0 * * *\n    timezone: America/Chicago\n

    Let's discuss the three schedule types in more detail.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#cron","title":"Cron","text":"

    A schedule may be specified with a cron pattern. Users may also provide a timezone to enforce DST behaviors.

    Cron uses croniter to specify datetime iteration with a cron-like format.

    Cron properties include:

    Property Description cron A valid cron string. (Required) day_or Boolean indicating how croniter handles day and day_of_week entries. Default is True. timezone String name of a time zone. (See the IANA Time Zone Database for valid time zones.)

    The day_or property defaults to True, matching cron, which connects those values using OR. If False, the values are connected using AND. This behaves like fcron and enables you to, for example, define a job that executes each 2nd Friday of a month by setting the days of month and the weekday.

    Supported croniter features

    While Prefect supports most features of croniter for creating cron-like schedules, we do not currently support \"R\" random or \"H\" hashed keyword expressions or the schedule jittering possible with those expressions.

    Daylight saving time considerations

    If the timezone is a DST-observing one, then the schedule will adjust itself appropriately.

    The cron rules for DST are based on schedule times, not intervals. This means that an hourly cron schedule fires on every new schedule hour, not every elapsed hour. For example, when clocks are set back, this results in a two-hour pause as the schedule will fire the first time 1am is reached and the first time 2am is reached, 120 minutes later.

    Longer schedules, such as one that fires at 9am every morning, will adjust for DST automatically.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#interval","title":"Interval","text":"

    An Interval schedule creates new flow runs on a regular interval measured in seconds. Intervals are computed from an optional anchor_date. For example, here's how you can create a schedule for every 10 minutes in the deployment YAML file.

    schedule:\n  interval: 600\n  timezone: America/Chicago \n

    Interval properties include:

    Property Description interval datetime.timedelta indicating the time between flow runs. (Required) anchor_date datetime.datetime indicating the starting or \"anchor\" date to begin the schedule. If no anchor_date is supplied, the current UTC time is used. timezone String name of a time zone, used to enforce localization behaviors like DST boundaries. (See the IANA Time Zone Database for valid time zones.)

    Note that the anchor_date does not indicate a \"start time\" for the schedule, but rather a fixed point in time from which to compute intervals. If the anchor date is in the future, then schedule dates are computed by subtracting the interval from it. Note that in this example, we import the Pendulum Python package for easy datetime manipulation. Pendulum isn\u2019t required, but it\u2019s a useful tool for specifying dates.

    Daylight saving time considerations

    If the schedule's anchor_date or timezone are provided with a DST-observing timezone, then the schedule will adjust itself appropriately. Intervals greater than 24 hours will follow DST conventions, while intervals of less than 24 hours will follow UTC intervals.

    For example, an hourly schedule will fire every UTC hour, even across DST boundaries. When clocks are set back, this will result in two runs that appear to both be scheduled for 1am local time, even though they are an hour apart in UTC time.

    For longer intervals, like a daily schedule, the interval schedule will adjust for DST boundaries so that the clock-hour remains constant. This means that a daily schedule that always fires at 9am will observe DST and continue to fire at 9am in the local time zone.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#rrule","title":"RRule","text":"

    An RRule scheduling supports iCal recurrence rules (RRules), which provide convenient syntax for creating repetitive schedules. Schedules can repeat on a frequency from yearly down to every minute.

    RRule uses the dateutil rrule module to specify iCal recurrence rules.

    RRules are appropriate for any kind of calendar-date manipulation, including simple repetition, irregular intervals, exclusions, week day or day-of-month adjustments, and more. RRules can represent complex logic like:

    • The last weekday of each month
    • The fourth Thursday of November
    • Every other day of the week

    RRule properties include:

    Property Description rrule String representation of an RRule schedule. See the rrulestr examples for syntax. timezone String name of a time zone. See the IANA Time Zone Database for valid time zones.

    You may find it useful to use an RRule string generator such as the iCalendar.org RRule Tool to help create valid RRules.

    For example, the following RRule schedule creates flow runs on Monday, Wednesday, and Friday until July 30, 2024.

    schedule:\n  rrule: 'FREQ=WEEKLY;BYDAY=MO,WE,FR;UNTIL=20240730T040000Z'\n

    Max RRule length

    Note the max supported character length of an rrulestr is 6500 characters

    Daylight saving time considerations

    Note that as a calendar-oriented standard, RRules are sensitive to the initial timezone provided. A 9am daily schedule with a DST-aware start date will maintain a local 9am time through DST boundaries. A 9am daily schedule with a UTC start date will maintain a 9am UTC time.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-a-python-deployment-creation-file","title":"Creating schedules through a Python deployment creation file","text":"

    When you create a deployment with through a Python file with flow.serve(), serve, flow.deploy(), or deploy you can specify the schedule. Just add the keyword argument cron, interval, or rrule.

    interval: An interval on which to execute the new deployment. Accepts either a number\n    or a timedelta object. If a number is given, it will be interpreted as seconds.\ncron: A cron schedule of when to execute runs of this deployment.\nrrule: An rrule schedule of when to execute runs of this deployment.\nschedule: A schedule object defining when to execute runs of this deployment. Used to\n  define additional scheduling options like `timezone`.\n

    Here's an example of creating a cron schedule with serve for a deployment flow that will run every minute of every day:

    my_flow.serve(name=\"flowing\", cron=\"* * * * *\")\n

    Here's an example of creating an interval schedule with serve for a deployment flow that will run every 10 minutes with an anchor date and a timezone:

    from datetime import timedelta, datetime\nfrom prefect.client.schemas.schedules import IntervalSchedule\n\nmy_flow.serve(name=\"flowing\", schedule=IntervalSchedule(interval=timedelta(minutes=10), anchor_date=datetime(2023, 1, 1, 0, 0), timezone=\"America/Chicago\"))\n

    Block and agent-based deployments with Python files are not a recommended way to create deployments. However, if you are using that deployment creation method you can create a schedule by passing a schedule parameter to the Deployment.build_from_flow method.

    Here's how you create the equivalent schedule in a Python deployment file, with a timezone specified.

    from prefect.server.schemas.schedules import CronSchedule\n\ncron_demo = Deployment.build_from_flow(\n    pipeline,\n    \"etl\",\n    schedule=(CronSchedule(cron=\"0 0 * * *\", timezone=\"America/Chicago\"))\n)\n

    IntervalSchedule and RRuleSchedule are the other two Python class schedule options.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#the-scheduler-service","title":"The Scheduler service","text":"

    The Scheduler service is started automatically when prefect server start is run and it is a built-in service of Prefect Cloud.

    By default, the Scheduler service visits deployments on a 60-second loop, though recently-modified deployments will be visited more frequently. The Scheduler evaluates each deployment's schedule and creates new runs appropriately. For typical deployments, it will create the next three runs, though more runs will be scheduled if the next 3 would all start in the next hour.

    More specifically, the Scheduler tries to create the smallest number of runs that satisfy the following constraints, in order:

    • No more than 100 runs will be scheduled.
    • Runs will not be scheduled more than 100 days in the future.
    • At least 3 runs will be scheduled.
    • Runs will be scheduled until at least one hour in the future.

    These behaviors can all be adjusted through the relevant settings that can be viewed with the terminal command prefect config view --show-defaults:

    PREFECT_API_SERVICES_SCHEDULER_DEPLOYMENT_BATCH_SIZE='100'\nPREFECT_API_SERVICES_SCHEDULER_ENABLED='True'\nPREFECT_API_SERVICES_SCHEDULER_INSERT_BATCH_SIZE='500'\nPREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS='60.0'\nPREFECT_API_SERVICES_SCHEDULER_MIN_RUNS='3'\nPREFECT_API_SERVICES_SCHEDULER_MAX_RUNS='100'\nPREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME='1:00:00'\nPREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME='100 days, 0:00:00'\n

    See the Settings docs for more information on altering your settings.

    These settings mean that if a deployment has an hourly schedule, the default settings will create runs for the next 4 days (or 100 hours). If it has a weekly schedule, the default settings will maintain the next 14 runs (up to 100 days in the future).

    The Scheduler does not affect execution

    The Prefect Scheduler service only creates new flow runs and places them in Scheduled states. It is not involved in flow or task execution.

    If you change a schedule, previously scheduled flow runs that have not started are removed, and new scheduled flow runs are created to reflect the new schedule.

    To remove all scheduled runs for a flow deployment, you can remove the schedule via the UI.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/states/","title":"States","text":"","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#overview","title":"Overview","text":"

    States are rich objects that contain information about the status of a particular task run or flow run. While you don't need to know the details of the states to use Prefect, you can give your workflows superpowers by taking advantage of it.

    At any moment, you can learn anything you need to know about a task or flow by examining its current state or the history of its states. For example, a state could tell you that a task:

    • is scheduled to make a third run attempt in an hour

    • succeeded and what data it produced

    • was scheduled to run, but later cancelled

    • used the cached result of a previous run instead of re-running

    • failed because it timed out

    By manipulating a relatively small number of task states, Prefect flows can harness the complexity that emerges in workflows.

    Only runs have states

    Though we often refer to the \"state\" of a flow or a task, what we really mean is the state of a flow run or a task run. Flows and tasks are templates that describe what a system does; only when we run the system does it also take on a state. So while we might refer to a task as \"running\" or being \"successful\", we really mean that a specific instance of the task is in that state.

    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#state-types","title":"State Types","text":"

    States have names and types. State types are canonical, with specific orchestration rules that apply to transitions into and out of each state type. A state's name, is often, but not always, synonymous with its type. For example, a task run that is running for the first time has a state with the name Running and the type RUNNING. However, if the task retries, that same task run will have the name Retrying and the type RUNNING. Each time the task run transitions into the RUNNING state, the same orchestration rules are applied.

    There are terminal state types from which there are no orchestrated transitions to any other state type.

    • COMPLETED
    • CANCELLED
    • FAILED
    • CRASHED

    The full complement of states and state types includes:

    Name Type Terminal? Description Scheduled SCHEDULED No The run will begin at a particular time in the future. Late SCHEDULED No The run's scheduled start time has passed, but it has not transitioned to PENDING (5 seconds by default). AwaitingRetry SCHEDULED No The run did not complete successfully because of a code issue and had remaining retry attempts. Pending PENDING No The run has been submitted to run, but is waiting on necessary preconditions to be satisfied. Running RUNNING No The run code is currently executing. Retrying RUNNING No The run code is currently executing after previously not complete successfully. Paused PAUSED No The run code has stopped executing until it receives manual approval to proceed. Cancelling CANCELLING No The infrastructure on which the code was running is being cleaned up. Cancelled CANCELLED Yes The run did not complete because a user determined that it should not. Completed COMPLETED Yes The run completed successfully. Failed FAILED Yes The run did not complete because of a code issue and had no remaining retry attempts. Crashed CRASHED Yes The run did not complete because of an infrastructure issue.","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#returned-values","title":"Returned values","text":"

    When calling a task or a flow, there are three types of returned values:

    • Data: A Python object (such as int, str, dict, list, and so on).
    • State: A Prefect object indicating the state of a flow or task run.
    • PrefectFuture: A Prefect object that contains both data and State.

    Returning data\u200a is the default behavior any time you call your_task().

    Returning Prefect State occurs anytime you call your task or flow with the argument return_state=True.

    Returning PrefectFuture is achieved by calling your_task.submit().

    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#return-data","title":"Return Data","text":"

    By default, running a task will return data:

    from prefect import flow, task \n\n@task \ndef add_one(x):\n    return x + 1\n\n@flow \ndef my_flow():\n    result = add_one(1) # return int\n

    The same rule applies for a subflow:

    @flow \ndef subflow():\n    return 42 \n\n@flow \ndef my_flow():\n    result = subflow() # return data\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#return-prefect-state","title":"Return Prefect State","text":"

    To return a State instead, add return_state=True as a parameter of your task call.

    @flow \ndef my_flow():\n    state = add_one(1, return_state=True) # return State\n

    To get data from a State, call .result().

    @flow \ndef my_flow():\n    state = add_one(1, return_state=True) # return State\n    result = state.result() # return int\n

    The same rule applies for a subflow:

    @flow \ndef subflow():\n    return 42 \n\n@flow \ndef my_flow():\n    state = subflow(return_state=True) # return State\n    result = state.result() # return int\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#return-a-prefectfuture","title":"Return a PrefectFuture","text":"

    To get a PrefectFuture, add .submit() to your task call.

    @flow \ndef my_flow():\n    future = add_one.submit(1) # return PrefectFuture\n

    To get data from a PrefectFuture, call .result().

    @flow \ndef my_flow():\n    future = add_one.submit(1) # return PrefectFuture\n    result = future.result() # return data\n

    To get a State from a PrefectFuture, call .wait().

    @flow \ndef my_flow():\n    future = add_one.submit(1) # return PrefectFuture\n    state = future.wait() # return State\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#final-state-determination","title":"Final state determination","text":"

    The final state of a flow is determined by its return value. The following rules apply:

    • If an exception is raised directly in the flow function, the flow run is marked as FAILED.
    • If the flow does not return a value (or returns None), its state is determined by the states of all of the tasks and subflows within it.
    • If any task run or subflow run failed and none were cancelled, then the final flow run state is marked as FAILED.
    • If any task run or subflow run was cancelled, then the final flow run state is marked as CANCELLED.
    • If a flow returns a manually created state, it is used as the state of the final flow run. This allows for manual determination of final state.
    • If the flow run returns any other object, then it is marked as successfully completed.

    See the Final state determination section of the Flows documentation for further details and examples.

    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#state-change-hooks","title":"State Change Hooks","text":"

    State change hooks execute code in response to changes in flow or task run states, enabling you to define actions for specific state transitions in a workflow.

    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#a-simple-example","title":"A simple example","text":"
    from prefect import flow\n\ndef my_success_hook(flow, flow_run, state):\n    print(\"Flow run succeeded!\")\n\n@flow(on_completion=[my_success_hook])\ndef my_flow():\n    return 42\n\nmy_flow()\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#create-and-use-hooks","title":"Create and use hooks","text":"","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#available-state-change-hooks","title":"Available state change hooks","text":"Type Flow Task Description on_completion \u2713 \u2713 Executes when a flow or task run enters a Completed state. on_failure \u2713 \u2713 Executes when a flow or task run enters a Failed state. on_cancellation \u2713 - Executes when a flow run enters a Cancelling state. on_crashed \u2713 - Executes when a flow run enters a Crashed state.","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#create-flow-run-state-change-hooks","title":"Create flow run state change hooks","text":"
    def my_flow_hook(flow: Flow, flow_run: FlowRun, state: State):\n    \"\"\"This is the required signature for a flow run state\n    change hook. This hook can only be passed into flows.\n    \"\"\"\n\n# pass hook as a list of callables\n@flow(on_completion=[my_flow_hook])\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#create-task-run-state-change-hooks","title":"Create task run state change hooks","text":"
    def my_task_hook(task: Task, task_run: TaskRun, state: State):\n    \"\"\"This is the required signature for a task run state change\n    hook. This hook can only be passed into tasks.\n    \"\"\"\n\n# pass hook as a list of callables\n@task(on_failure=[my_task_hook])\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#use-multiple-state-change-hooks","title":"Use multiple state change hooks","text":"

    State change hooks are versatile, allowing you to specify multiple state change hooks for the same state transition, or to use the same state change hook for different transitions:

    def my_success_hook(task, task_run, state):\n    print(\"Task run succeeded!\")\n\ndef my_failure_hook(task, task_run, state):\n    print(\"Task run failed!\")\n\ndef my_succeed_or_fail_hook(task, task_run, state):\n    print(\"If the task run succeeds or fails, this hook runs.\")\n\n@task(\n    on_completion=[my_success_hook, my_succeed_or_fail_hook],\n    on_failure=[my_failure_hook, my_succeed_or_fail_hook]\n)\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#pass-kwargs-to-your-hooks","title":"Pass kwargs to your hooks","text":"

    The Prefect engine will call your hooks for you upon the state change, passing in the flow, flow run, and state objects.

    However, you can define your hook to have additional default arguments:

    from prefect import flow\n\ndata = {}\n\ndef my_hook(flow, flow_run, state, my_arg=\"custom_value\"):\n    data.update(my_arg=my_arg, state=state)\n\n@flow(on_completion=[my_hook])\ndef lazy_flow():\n    pass\n\nstate = lazy_flow(return_state=True)\n\nassert data == {\"my_arg\": \"custom_value\", \"state\": state}\n

    ... or define your hook to accept arbitrary keyword arguments:

    from functools import partial\nfrom prefect import flow, task\n\ndata = {}\n\ndef my_hook(task, task_run, state, **kwargs):\n    data.update(state=state, **kwargs)\n\n@task\ndef bad_task():\n    raise ValueError(\"meh\")\n\n@flow\ndef ok_with_failure_flow(x: str = \"foo\", y: int = 42):\n    bad_task_with_a_hook = bad_task.with_options(\n        on_failure=[partial(my_hook, **dict(x=x, y=y))]\n    )\n    # return a tuple of \"bar\" and the task run state\n    # to avoid raising the task's exception\n    return \"bar\", bad_task_with_a_hook(return_state=True)\n\n_, task_run_state = ok_with_failure_flow()\n\nassert data == {\"x\": \"foo\", \"y\": 42, \"state\": task_run_state}\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#more-examples-of-state-change-hooks","title":"More examples of state change hooks","text":"
    • Send a notification when a flow run fails
    • Delete a Cloud Run job when a flow crashes
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/storage/","title":"Storage","text":"

    Storage blocks are not recommended

    Storage blocks are part of the legacy block-based deployment model. Instead, using serve or runner-based Python creation methods or workers and work pools with prefect deploy via the CLI are the recommended options for creating a deployment. Flow code storage can be specified in the Python file with serve or runner-based Python creation methods; alternatively, with the work pools and workers style of flow deployment, you can specify flow code storage during the interactive prefect deploy CLI experience and in its resulting prefect.yaml file.

    Storage lets you configure how flow code for deployments is persisted and retrieved by Prefect workers (or legacy agents). Anytime you build a block-based deployment, a storage block is used to upload the entire directory containing your workflow code (along with supporting files) to its configured location. This helps ensure portability of your relative imports, configuration files, and more. Note that your environment dependencies (for example, external Python packages) still need to be managed separately.

    If no storage is explicitly configured, Prefect will use LocalFileSystem storage by default. Local storage works fine for many local flow run scenarios, especially when testing and getting started. However, due to the inherent lack of portability, many use cases are better served by using remote storage such as S3 or Google Cloud Storage.

    Prefect supports creating multiple storage configurations and switching between storage as needed.

    Storage uses blocks

    Blocks are the Prefect technology underlying storage, and enables you to do so much more.

    In addition to creating storage blocks via the Prefect CLI, you can now create storage blocks and other kinds of block configuration objects via the Prefect UI and Prefect Cloud.

    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/storage/#configuring-storage-for-a-deployment","title":"Configuring storage for a deployment","text":"

    When building a deployment for a workflow, you have two options for configuring workflow storage:

    • Use the default local storage
    • Preconfigure a storage block to use
    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/storage/#using-the-default","title":"Using the default","text":"

    Anytime you call prefect deployment build without providing the --storage-block flag, a default LocalFileSystem block will be used. Note that this block will always use your present working directory as its basepath (which is usually desirable). You can see the block's settings by inspecting the deployment.yaml file that Prefect creates after calling prefect deployment build.

    While you generally can't run a deployment stored on a local file system on other machines, any agent running on the same machine will be able to successfully run your deployment.

    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/storage/#supported-storage-blocks","title":"Supported storage blocks","text":"

    Current options for deployment storage blocks include:

    Storage Description Required Library Local File System Store code in a run's local file system. Remote File System Store code in a any filesystem supported by fsspec. AWS S3 Storage Store code in an AWS S3 bucket. s3fs Azure Storage Store code in Azure Datalake and Azure Blob Storage. adlfs GitHub Storage Store code in a GitHub repository. Google Cloud Storage Store code in a Google Cloud Platform (GCP) Cloud Storage bucket. gcsfs SMB Store code in SMB shared network storage. smbprotocol GitLab Repository Store code in a GitLab repository. prefect-gitlab Bitbucket Repository Store code in a Bitbucket repository. prefect-bitbucket

    Accessing files may require storage filesystem libraries

    Note that the appropriate filesystem library supporting the storage location must be installed prior to building a deployment with a storage block or accessing the storage location from flow scripts.

    For example, the AWS S3 Storage block requires the s3fs library.

    See Filesystem package dependencies for more information about configuring filesystem libraries in your execution environment.

    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/storage/#configuring-a-block","title":"Configuring a block","text":"

    You can create these blocks either via the UI or via Python.

    You can create, edit, and manage storage blocks in the Prefect UI and Prefect Cloud. On a Prefect server, blocks are created in the server's database. On Prefect Cloud, blocks are created on a workspace.

    To create a new block, select the + button. Prefect displays a library of block types you can configure to create blocks to be used by your flows.

    Select Add + to configure a new storage block based on a specific block type. Prefect displays a Create page that enables specifying storage settings.

    You can also create blocks using the Prefect Python API:

    from prefect.filesystems import S3\n\nblock = S3(bucket_path=\"my-bucket/a-sub-directory\", \n           aws_access_key_id=\"foo\", \n           aws_secret_access_key=\"bar\"\n)\nblock.save(\"example-block\")\n

    This block configuration is now available to be used by anyone with appropriate access to your Prefect API. We can use this block to build a deployment by passing its slug to the prefect deployment build command. The storage block slug is formatted as block-type/block-name. In this case, s3/example-block for an AWS S3 Bucket block named example-block. See block identifiers for details.

    prefect deployment build ./flows/my_flow.py:my_flow --name \"Example Deployment\" --storage-block s3/example-block\n

    This command will upload the contents of your flow's directory to the designated storage location, then the full deployment specification will be persisted to a newly created deployment.yaml file. For more information, see Deployments.

    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/task-runners/","title":"Task runners","text":"

    Task runners enable you to engage specific executors for Prefect tasks, such as for concurrent, parallel, or distributed execution of tasks.

    Task runners are not required for task execution. If you call a task function directly, the task executes as a regular Python function, without a task runner, and produces whatever result is returned by the function.

    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#task-runner-overview","title":"Task runner overview","text":"

    Calling a task function from within a flow, using the default task settings, executes the function sequentially. Execution of the task function blocks execution of the flow until the task completes. This means, by default, calling multiple tasks in a flow causes them to run in order.

    However, that's not the only way to run tasks!

    You can use the .submit() method on a task function to submit the task to a task runner. Using a task runner enables you to control whether tasks run sequentially, concurrently, or if you want to take advantage of a parallel or distributed execution library such as Dask or Ray.

    Using the .submit() method to submit a task also causes the task run to return a PrefectFuture, a Prefect object that contains both any data returned by the task function and a State, a Prefect object indicating the state of the task run.

    Prefect currently provides the following built-in task runners:

    • SequentialTaskRunner can run tasks sequentially.
    • ConcurrentTaskRunner can run tasks concurrently, allowing tasks to switch when blocking on IO. Tasks will be submitted to a thread pool maintained by anyio.

    In addition, the following Prefect-developed task runners for parallel or distributed task execution may be installed as Prefect Integrations.

    • DaskTaskRunner can run tasks requiring parallel execution using dask.distributed.
    • RayTaskRunner can run tasks requiring parallel execution using Ray.

    Concurrency versus parallelism

    The words \"concurrency\" and \"parallelism\" may sound the same, but they mean different things in computing.

    Concurrency refers to a system that can do more than one thing simultaneously, but not at the exact same time. It may be more accurate to think of concurrent execution as non-blocking: within the restrictions of resources available in the execution environment and data dependencies between tasks, execution of one task does not block execution of other tasks in a flow.

    Parallelism refers to a system that can do more than one thing at the exact same time. Again, within the restrictions of resources available, parallel execution can run tasks at the same time, such as for operations mapped across a dataset.

    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#using-a-task-runner","title":"Using a task runner","text":"

    You do not need to specify a task runner for a flow unless your tasks require a specific type of execution.

    To configure your flow to use a specific task runner, import a task runner and assign it as an argument for the flow when the flow is defined.

    Remember to call .submit() when using a task runner

    Make sure you use .submit() to run your task with a task runner. Calling the task directly, without .submit(), from within a flow will run the task sequentially instead of using a specified task runner.

    For example, you can use ConcurrentTaskRunner to allow tasks to switch when they would block.

    from prefect import flow, task\nfrom prefect.task_runners import ConcurrentTaskRunner\nimport time\n\n@task\ndef stop_at_floor(floor):\n    print(f\"elevator moving to floor {floor}\")\n    time.sleep(floor)\n    print(f\"elevator stops on floor {floor}\")\n\n@flow(task_runner=ConcurrentTaskRunner())\ndef elevator():\n    for floor in range(10, 0, -1):\n        stop_at_floor.submit(floor)\n

    If you specify an uninitialized task runner class, a task runner instance of that type is created with the default settings. You can also pass additional configuration parameters for task runners that accept parameters, such as DaskTaskRunner and RayTaskRunner.

    Default task runner

    If you don't specify a task runner for a flow and you call a task with .submit() within the flow, Prefect uses the default ConcurrentTaskRunner.

    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#running-tasks-sequentially","title":"Running tasks sequentially","text":"

    Sometimes, it's useful to force tasks to run sequentially to make it easier to reason about the behavior of your program. Switching to the SequentialTaskRunner will force submitted tasks to run sequentially rather than concurrently.

    Synchronous and asynchronous tasks

    The SequentialTaskRunner works with both synchronous and asynchronous task functions. Asynchronous tasks are Python functions defined using async def rather than def.

    The following example demonstrates using the SequentialTaskRunner to ensure that tasks run sequentially. In the example, the flow glass_tower runs the task stop_at_floor for floors one through 38, in that order.

    from prefect import flow, task\nfrom prefect.task_runners import SequentialTaskRunner\nimport random\n\n@task\ndef stop_at_floor(floor):\n    situation = random.choice([\"on fire\",\"clear\"])\n    print(f\"elevator stops on {floor} which is {situation}\")\n\n@flow(task_runner=SequentialTaskRunner(),\n      name=\"towering-infernflow\",\n      )\ndef glass_tower():\n    for floor in range(1, 39):\n        stop_at_floor.submit(floor)\n\nglass_tower()\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#using-multiple-task-runners","title":"Using multiple task runners","text":"

    Each flow can only have a single task runner, but sometimes you may want a subset of your tasks to run using a specific task runner. In this case, you can create subflows for tasks that need to use a different task runner.

    For example, you can have a flow (in the example below called sequential_flow) that runs its tasks locally using the SequentialTaskRunner. If you have some tasks that can run more efficiently in parallel on a Dask cluster, you could create a subflow (such as dask_subflow) to run those tasks using the DaskTaskRunner.

    from prefect import flow, task\nfrom prefect.task_runners import SequentialTaskRunner\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@task\ndef hello_local():\n    print(\"Hello!\")\n\n@task\ndef hello_dask():\n    print(\"Hello from Dask!\")\n\n@flow(task_runner=SequentialTaskRunner())\ndef sequential_flow():\n    hello_local.submit()\n    dask_subflow()\n    hello_local.submit()\n\n@flow(task_runner=DaskTaskRunner())\ndef dask_subflow():\n    hello_dask.submit()\n\nif __name__ == \"__main__\":\n    sequential_flow()\n

    Guarding main

    Note that you should guard the main function by using if __name__ == \"__main__\" to avoid issues with parallel processing.

    This script outputs the following logs demonstrating the use of the Dask task runner:

    120:14:29.785 | INFO    | prefect.engine - Created flow run 'ivory-caiman' for flow 'sequential-flow'\n20:14:29.785 | INFO    | Flow run 'ivory-caiman' - Starting 'SequentialTaskRunner'; submitted tasks will be run sequentially...\n20:14:29.880 | INFO    | Flow run 'ivory-caiman' - Created task run 'hello_local-7633879f-0' for task 'hello_local'\n20:14:29.881 | INFO    | Flow run 'ivory-caiman' - Executing 'hello_local-7633879f-0' immediately...\nHello!\n20:14:29.904 | INFO    | Task run 'hello_local-7633879f-0' - Finished in state Completed()\n20:14:29.952 | INFO    | Flow run 'ivory-caiman' - Created subflow run 'nimble-sparrow' for flow 'dask-subflow'\n20:14:29.953 | INFO    | prefect.task_runner.dask - Creating a new Dask cluster with `distributed.deploy.local.LocalCluster`\n20:14:31.862 | INFO    | prefect.task_runner.dask - The Dask dashboard is available at http://127.0.0.1:8787/status\n20:14:31.901 | INFO    | Flow run 'nimble-sparrow' - Created task run 'hello_dask-2b96d711-0' for task 'hello_dask'\n20:14:32.370 | INFO    | Flow run 'nimble-sparrow' - Submitted task run 'hello_dask-2b96d711-0' for execution.\nHello from Dask!\n20:14:33.358 | INFO    | Flow run 'nimble-sparrow' - Finished in state Completed('All states completed.')\n20:14:33.368 | INFO    | Flow run 'ivory-caiman' - Created task run 'hello_local-7633879f-1' for task 'hello_local'\n20:14:33.368 | INFO    | Flow run 'ivory-caiman' - Executing 'hello_local-7633879f-1' immediately...\nHello!\n20:14:33.386 | INFO    | Task run 'hello_local-7633879f-1' - Finished in state Completed()\n20:14:33.399 | INFO    | Flow run 'ivory-caiman' - Finished in state Completed('All states completed.')\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#using-results-from-submitted-tasks","title":"Using results from submitted tasks","text":"

    When you use .submit() to submit a task to a task runner, the task runner creates a PrefectFuture for access to the state and result of the task.

    A PrefectFuture is an object that provides access to a computation happening in a task runner \u2014 even if that computation is happening on a remote system.

    In the following example, we save the return value of calling .submit() on the task say_hello to the variable future, and then we print the type of the variable:

    from prefect import flow, task\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@flow\ndef hello_world():\n    future = say_hello.submit(\"Marvin\")\n    print(f\"variable 'future' is type {type(future)}\")\n\nhello_world()\n

    When you run this code, you'll see that the variable future is a PrefectFuture:

    variable 'future' is type <class 'prefect.futures.PrefectFuture'>\n

    When you pass a future into a task, Prefect waits for the \"upstream\" task \u2014 the one that the future references \u2014 to reach a final state before starting the downstream task.

    This means that the downstream task won't receive the PrefectFuture you passed as an argument. Instead, the downstream task will receive the value that the upstream task returned.

    Take a look at how this works in the following example

    from prefect import flow, task\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@task\ndef print_result(result):\n    print(type(result))\n    print(result)\n\n@flow(name=\"hello-flow\")\ndef hello_world():\n    future = say_hello.submit(\"Marvin\")\n    print_result.submit(future)\n\nhello_world()\n
    <class 'str'>\nHello Marvin!\n

    Futures have a few useful methods. For example, you can get the return value of the task run with .result():

    from prefect import flow, task\n\n@task\ndef my_task():\n    return 42\n\n@flow\ndef my_flow():\n    future = my_task.submit()\n    result = future.result()\n    print(result)\n\nmy_flow()\n

    The .result() method will wait for the task to complete before returning the result to the caller. If the task run fails, .result() will raise the task run's exception. You may disable this behavior with the raise_on_failure option:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return \"I'm a task!\"\n\n\n@flow\ndef my_flow():\n    future = my_task.submit()\n    result = future.result(raise_on_failure=False)\n    if future.get_state().is_failed():\n        # `result` is an exception! handle accordingly\n        ...\n    else:\n        # `result` is the expected return value of our task\n        ...\n

    You can retrieve the current state of the task run associated with the PrefectFuture using .get_state():

    @flow\ndef my_flow():\n    future = my_task.submit()\n    state = future.get_state()\n

    You can also wait for a task to complete by using the .wait() method:

    @flow\ndef my_flow():\n    future = my_task.submit()\n    final_state = future.wait()\n

    You can include a timeout in the wait call to perform logic if the task has not finished in a given amount of time:

    @flow\ndef my_flow():\n    future = my_task.submit()\n    final_state = future.wait(1)  # Wait one second max\n    if final_state:\n        # Take action if the task is done\n        result = final_state.result()\n    else:\n        ... # Task action if the task is still running\n

    You may also use the wait_for=[] parameter when calling a task, specifying upstream task dependencies. This enables you to control task execution order for tasks that do not share data dependencies.

    @task\ndef task_a():\n    pass\n\n@task\ndef task_b():\n    pass\n\n@task\ndef task_c():\n    pass\n\n@task\ndef task_d():\n    pass\n\n@flow\ndef my_flow():\n    a = task_a.submit()\n    b = task_b.submit()\n    # Wait for task_a and task_b to complete\n    c = task_c.submit(wait_for=[a, b])\n    # task_d will wait for task_c to complete\n    # Note: If waiting for one task it must still be in a list.\n    d = task_d(wait_for=[c])\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#when-to-use-result-in-flows","title":"When to use .result() in flows","text":"

    The simplest pattern for writing a flow is either only using tasks or only using pure Python functions. When you need to mix the two, use .result().

    Using only tasks:

    from prefect import flow, task\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@task\ndef say_nice_to_meet_you(hello_greeting):\n    return f\"{hello_greeting} Nice to meet you :)\"\n\n@flow\ndef hello_world():\n    hello = say_hello.submit(\"Marvin\")\n    nice_to_meet_you = say_nice_to_meet_you.submit(hello)\n\nhello_world()\n

    Using only Python functions:

    from prefect import flow, task\n\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\ndef say_nice_to_meet_you(hello_greeting):\n    return f\"{hello_greeting} Nice to meet you :)\"\n\n@flow\ndef hello_world():\n    # because this is just a Python function, calls will not be tracked\n    hello = say_hello(\"Marvin\") \n    nice_to_meet_you = say_nice_to_meet_you(hello)\n\nhello_world()\n

    Mixing tasks and Python functions:

    from prefect import flow, task\n\ndef say_hello_extra_nicely_to_marvin(hello): # not a task or flow!\n    if hello == \"Hello Marvin!\":\n        return \"HI MARVIN!\"\n    return hello\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@task\ndef say_nice_to_meet_you(hello_greeting):\n    return f\"{hello_greeting} Nice to meet you :)\"\n\n@flow\ndef hello_world():\n    # run a task and get the result\n    hello = say_hello.submit(\"Marvin\").result()\n\n    # not calling a task or flow\n    special_greeting = say_hello_extra_nicely_to_marvin(hello)\n\n    # pass our modified greeting back into a task\n    nice_to_meet_you = say_nice_to_meet_you.submit(special_greeting)\n\n    print(nice_to_meet_you.result())\n\nhello_world()\n

    Note that .result() also limits Prefect's ability to track task dependencies. In the \"mixed\" example above, Prefect will not be aware that say_hello is upstream of nice_to_meet_you.

    Calling .result() is blocking

    When calling .result(), be mindful your flow function will have to wait until the task run is completed before continuing.

    from prefect import flow, task\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@task\ndef do_important_stuff():\n    print(\"Doing lots of important stuff!\")\n\n@flow\ndef hello_world():\n    # blocks until `say_hello` has finished\n    result = say_hello.submit(\"Marvin\").result() \n    do_important_stuff.submit()\n\nhello_world()\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#running-tasks-on-dask","title":"Running tasks on Dask","text":"

    The DaskTaskRunner is a parallel task runner that submits tasks to the dask.distributed scheduler. By default, a temporary Dask cluster is created for the duration of the flow run. If you already have a Dask cluster running, either local or cloud hosted, you can provide the connection URL via the address kwarg.

    1. Make sure the prefect-dask collection is installed: pip install prefect-dask.
    2. In your flow code, import DaskTaskRunner from prefect_dask.task_runners.
    3. Assign it as the task runner when the flow is defined using the task_runner=DaskTaskRunner argument.

    For example, this flow uses the DaskTaskRunner configured to access an existing Dask cluster at http://my-dask-cluster.

    from prefect import flow\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@flow(task_runner=DaskTaskRunner(address=\"http://my-dask-cluster\"))\ndef my_flow():\n    ...\n

    DaskTaskRunner accepts the following optional parameters:

    Parameter Description address Address of a currently running Dask scheduler. cluster_class The cluster class to use when creating a temporary Dask cluster. It can be either the full class name (for example, \"distributed.LocalCluster\"), or the class itself. cluster_kwargs Additional kwargs to pass to the cluster_class when creating a temporary Dask cluster. adapt_kwargs Additional kwargs to pass to cluster.adapt when creating a temporary Dask cluster. Note that adaptive scaling is only enabled if adapt_kwargs are provided. client_kwargs Additional kwargs to use when creating a dask.distributed.Client.

    Multiprocessing safety

    Note that, because the DaskTaskRunner uses multiprocessing, calls to flows in scripts must be guarded with if __name__ == \"__main__\": or you will encounter warnings and errors.

    If you don't provide the address of a Dask scheduler, Prefect creates a temporary local cluster automatically. The number of workers used is based on the number of cores on your machine. The default provides a mix of processes and threads that should work well for most workloads. If you want to specify this explicitly, you can pass values for n_workers or threads_per_worker to cluster_kwargs.

    # Use 4 worker processes, each with 2 threads\nDaskTaskRunner(\n    cluster_kwargs={\"n_workers\": 4, \"threads_per_worker\": 2}\n)\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#using-a-temporary-cluster","title":"Using a temporary cluster","text":"

    The DaskTaskRunner is capable of creating a temporary cluster using any of Dask's cluster-manager options. This can be useful when you want each flow run to have its own Dask cluster, allowing for per-flow adaptive scaling.

    To configure, you need to provide a cluster_class. This can be:

    • A string specifying the import path to the cluster class (for example, \"dask_cloudprovider.aws.FargateCluster\")
    • The cluster class itself
    • A function for creating a custom cluster.

    You can also configure cluster_kwargs, which takes a dictionary of keyword arguments to pass to cluster_class when starting the flow run.

    For example, to configure a flow to use a temporary dask_cloudprovider.aws.FargateCluster with 4 workers running with an image named my-prefect-image:

    DaskTaskRunner(\n    cluster_class=\"dask_cloudprovider.aws.FargateCluster\",\n    cluster_kwargs={\"n_workers\": 4, \"image\": \"my-prefect-image\"},\n)\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#connecting-to-an-existing-cluster","title":"Connecting to an existing cluster","text":"

    Multiple Prefect flow runs can all use the same existing Dask cluster. You might manage a single long-running Dask cluster (maybe using the Dask Helm Chart) and configure flows to connect to it during execution. This has a few downsides when compared to using a temporary cluster (as described above):

    • All workers in the cluster must have dependencies installed for all flows you intend to run.
    • Multiple flow runs may compete for resources. Dask tries to do a good job sharing resources between tasks, but you may still run into issues.

    That said, you may prefer managing a single long-running cluster.

    To configure a DaskTaskRunner to connect to an existing cluster, pass in the address of the scheduler to the address argument:

    # Connect to an existing cluster running at a specified address\nDaskTaskRunner(address=\"tcp://...\")\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#adaptive-scaling","title":"Adaptive scaling","text":"

    One nice feature of using a DaskTaskRunner is the ability to scale adaptively to the workload. Instead of specifying n_workers as a fixed number, this lets you specify a minimum and maximum number of workers to use, and the dask cluster will scale up and down as needed.

    To do this, you can pass adapt_kwargs to DaskTaskRunner. This takes the following fields:

    • maximum (int or None, optional): the maximum number of workers to scale to. Set to None for no maximum.
    • minimum (int or None, optional): the minimum number of workers to scale to. Set to None for no minimum.

    For example, here we configure a flow to run on a FargateCluster scaling up to at most 10 workers.

    DaskTaskRunner(\n    cluster_class=\"dask_cloudprovider.aws.FargateCluster\",\n    adapt_kwargs={\"maximum\": 10}\n)\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#dask-annotations","title":"Dask annotations","text":"

    Dask annotations can be used to further control the behavior of tasks.

    For example, we can set the priority of tasks in the Dask scheduler:

    import dask\nfrom prefect import flow, task\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@task\ndef show(x):\n    print(x)\n\n\n@flow(task_runner=DaskTaskRunner())\ndef my_flow():\n    with dask.annotate(priority=-10):\n        future = show.submit(1)  # low priority task\n\n    with dask.annotate(priority=10):\n        future = show.submit(2)  # high priority task\n

    Another common use case is resource annotations:

    import dask\nfrom prefect import flow, task\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@task\ndef show(x):\n    print(x)\n\n# Create a `LocalCluster` with some resource annotations\n# Annotations are abstract in dask and not inferred from your system.\n# Here, we claim that our system has 1 GPU and 1 process available per worker\n@flow(\n    task_runner=DaskTaskRunner(\n        cluster_kwargs={\"n_workers\": 1, \"resources\": {\"GPU\": 1, \"process\": 1}}\n    )\n)\n\ndef my_flow():\n    with dask.annotate(resources={'GPU': 1}):\n        future = show(0)  # this task requires 1 GPU resource on a worker\n\n    with dask.annotate(resources={'process': 1}):\n        # These tasks each require 1 process on a worker; because we've \n        # specified that our cluster has 1 process per worker and 1 worker,\n        # these tasks will run sequentially\n        future = show(1)\n        future = show(2)\n        future = show(3)\n\n\nif __name__ == \"__main__\":\n    my_flow()\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#running-tasks-on-ray","title":"Running tasks on Ray","text":"

    The RayTaskRunner \u2014 installed separately as a Prefect Collection \u2014 is a parallel task runner that submits tasks to Ray. By default, a temporary Ray instance is created for the duration of the flow run. If you already have a Ray instance running, you can provide the connection URL via an address argument.

    Remote storage and Ray tasks

    We recommend configuring remote storage for task execution with the RayTaskRunner. This ensures tasks executing in Ray have access to task result storage, particularly when accessing a Ray instance outside of your execution environment.

    To configure your flow to use the RayTaskRunner:

    1. Make sure the prefect-ray collection is installed: pip install prefect-ray.
    2. In your flow code, import RayTaskRunner from prefect_ray.task_runners.
    3. Assign it as the task runner when the flow is defined using the task_runner=RayTaskRunner argument.

    For example, this flow uses the RayTaskRunner configured to access an existing Ray instance at ray://192.0.2.255:8786.

    from prefect import flow\nfrom prefect_ray.task_runners import RayTaskRunner\n\n@flow(task_runner=RayTaskRunner(address=\"ray://192.0.2.255:8786\"))\ndef my_flow():\n    ... \n

    RayTaskRunner accepts the following optional parameters:

    Parameter Description address Address of a currently running Ray instance, starting with the ray:// URI. init_kwargs Additional kwargs to use when calling ray.init.

    Note that Ray Client uses the ray:// URI to indicate the address of a Ray instance. If you don't provide the address of a Ray instance, Prefect creates a temporary instance automatically.

    Ray environment limitations

    While we're excited about adding support for parallel task execution via Ray to Prefect, there are some inherent limitations with Ray you should be aware of:

    Ray's support for Python 3.11 is experimental.

    Ray support for non-x86/64 architectures such as ARM/M1 processors with installation from pip alone and will be skipped during installation of Prefect. It is possible to manually install the blocking component with conda. See the Ray documentation for instructions.

    See the Ray installation documentation for further compatibility information.

    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/tasks/","title":"Tasks","text":"

    A task is a function that represents a discrete unit of work in a Prefect workflow. Tasks are not required \u2014 you may define Prefect workflows that consist only of flows, using regular Python statements and functions. Tasks enable you to encapsulate elements of your workflow logic in observable units that can be reused across flows and subflows.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#tasks-overview","title":"Tasks overview","text":"

    Tasks are functions: they can take inputs, perform work, and return an output. A Prefect task can do almost anything a Python function can do.

    Tasks are special because they receive metadata about upstream dependencies and the state of those dependencies before they run, even if they don't receive any explicit data inputs from them. This gives you the opportunity to, for example, have a task wait on the completion of another task before executing.

    Tasks also take advantage of automatic Prefect logging to capture details about task runs such as runtime, tags, and final state.

    You can define your tasks within the same file as your flow definition, or you can define tasks within modules and import them for use in your flow definitions. All tasks must be called from within a flow. Tasks may not be called from other tasks.

    Calling a task from a flow

    Use the @task decorator to designate a function as a task. Calling the task from within a flow function creates a new task run:

    from prefect import flow, task\n\n@task\ndef my_task():\n    print(\"Hello, I'm a task\")\n\n@flow\ndef my_flow():\n    my_task()\n

    Tasks are uniquely identified by a task key, which is a hash composed of the task name, the fully-qualified name of the function, and any tags. If the task does not have a name specified, the name is derived from the task function.

    How big should a task be?

    Prefect encourages \"small tasks\" \u2014 each one should represent a single logical step of your workflow. This allows Prefect to better contain task failures.

    To be clear, there's nothing stopping you from putting all of your code in a single task \u2014 Prefect will happily run it! However, if any line of code fails, the entire task will fail and must be retried from the beginning. This can be avoided by splitting the code into multiple dependent tasks.

    Calling a task's function from another task

    Prefect does not allow triggering task runs from other tasks. If you want to call your task's function directly, you can use task.fn().

    from prefect import flow, task\n\n@task\ndef my_first_task(msg):\n    print(f\"Hello, {msg}\")\n\n@task\ndef my_second_task(msg):\n    my_first_task.fn(msg)\n\n@flow\ndef my_flow():\n    my_second_task(\"Trillian\")\n

    Note that in the example above you are only calling the task's function without actually generating a task run. Prefect won't track task execution in your Prefect backend if you call the task function this way. You also won't be able to use features such as retries with this function call.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#task-arguments","title":"Task arguments","text":"

    Tasks allow for customization through optional arguments:

    Argument Description name An optional name for the task. If not provided, the name will be inferred from the function name. description An optional string description for the task. If not provided, the description will be pulled from the docstring for the decorated function. tags An optional set of tags to be associated with runs of this task. These tags are combined with any tags defined by a prefect.tags context at task runtime. cache_key_fn An optional callable that, given the task run context and call parameters, generates a string key. If the key matches a previous completed state, that state result will be restored instead of running the task again. cache_expiration An optional amount of time indicating how long cached states for this task should be restorable; if not provided, cached states will never expire. retries An optional number of times to retry on task run failure. retry_delay_seconds An optional number of seconds to wait before retrying the task after failure. This is only applicable if retries is nonzero. log_prints An optional boolean indicating whether to log print statements. persist_result An optional boolean indicating whether to persist the result of the task run to storage.

    See all possible parameters in the Python SDK API docs.

    For example, you can provide a name value for the task. Here we've used the optional description argument as well.

    @task(name=\"hello-task\", \n      description=\"This task says hello.\")\ndef my_task():\n    print(\"Hello, I'm a task\")\n

    You can distinguish runs of this task by providing a task_run_name; this setting accepts a string that can optionally contain templated references to the keyword arguments of your task. The name will be formatted using Python's standard string formatting syntax as can be seen here:

    import datetime\nfrom prefect import flow, task\n\n@task(name=\"My Example Task\", \n      description=\"An example task for a tutorial.\",\n      task_run_name=\"hello-{name}-on-{date:%A}\")\ndef my_task(name, date):\n    pass\n\n@flow\ndef my_flow():\n    # creates a run with a name like \"hello-marvin-on-Thursday\"\n    my_task(name=\"marvin\", date=datetime.datetime.utcnow())\n

    Additionally this setting also accepts a function that returns a string to be used for the task run name:

    import datetime\nfrom prefect import flow, task\n\ndef generate_task_name():\n    date = datetime.datetime.utcnow()\n    return f\"{date:%A}-is-a-lovely-day\"\n\n@task(name=\"My Example Task\",\n      description=\"An example task for a tutorial.\",\n      task_run_name=generate_task_name)\ndef my_task(name):\n    pass\n\n@flow\ndef my_flow():\n    # creates a run with a name like \"Thursday-is-a-lovely-day\"\n    my_task(name=\"marvin\")\n

    If you need access to information about the task, use the prefect.runtime module. For example:

    from prefect import flow\nfrom prefect.runtime import flow_run, task_run\n\ndef generate_task_name():\n    flow_name = flow_run.flow_name\n    task_name = task_run.task_name\n\n    parameters = task_run.parameters\n    name = parameters[\"name\"]\n    limit = parameters[\"limit\"]\n\n    return f\"{flow_name}-{task_name}-with-{name}-and-{limit}\"\n\n@task(name=\"my-example-task\",\n      description=\"An example task for a tutorial.\",\n      task_run_name=generate_task_name)\ndef my_task(name: str, limit: int = 100):\n    pass\n\n@flow\ndef my_flow(name: str):\n    # creates a run with a name like \"my-flow-my-example-task-with-marvin-and-100\"\n    my_task(name=\"marvin\")\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#tags","title":"Tags","text":"

    Tags are optional string labels that enable you to identify and group tasks other than by name or flow. Tags are useful for:

    • Filtering task runs by tag in the UI and via the Prefect REST API.
    • Setting concurrency limits on task runs by tag.

    Tags may be specified as a keyword argument on the task decorator.

    @task(name=\"hello-task\", tags=[\"test\"])\ndef my_task():\n    print(\"Hello, I'm a task\")\n

    You can also provide tags as an argument with a tags context manager, specifying tags when the task is called rather than in its definition.

    from prefect import flow, task\nfrom prefect import tags\n\n@task\ndef my_task():\n    print(\"Hello, I'm a task\")\n\n@flow\ndef my_flow():\n    with tags(\"test\"):\n        my_task()\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#retries","title":"Retries","text":"

    Prefect can automatically retry tasks on failure. In Prefect, a task fails if its Python function raises an exception.

    To enable retries, pass retries and retry_delay_seconds parameters to your task. If the task fails, Prefect will retry it up to retries times, waiting retry_delay_seconds seconds between each attempt. If the task fails on the final retry, Prefect marks the task as crashed if the task raised an exception or failed if it returned a string.

    Retries don't create new task runs

    A new task run is not created when a task is retried. A new state is added to the state history of the original task run.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#a-real-world-example-making-an-api-request","title":"A real-world example: making an API request","text":"

    Consider the real-world problem of making an API request. In this example, we'll use the httpx library to make an HTTP request.

    import httpx\n\nfrom prefect import flow, task\n\n\n@task(retries=2, retry_delay_seconds=5)\ndef get_data_task(\n    url: str = \"https://api.brittle-service.com/endpoint\"\n) -> dict:\n    response = httpx.get(url)\n\n    # If the response status code is anything but a 2xx, httpx will raise\n    # an exception. This task doesn't handle the exception, so Prefect will\n    # catch the exception and will consider the task run failed.\n    response.raise_for_status()\n\n    return response.json()\n\n\n@flow\ndef get_data_flow():\n    get_data_task()\n

    In this task, if the HTTP request to the brittle API receives any status code other than a 2xx (200, 201, etc.), Prefect will retry the task a maximum of two times, waiting five seconds in between retries.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#custom-retry-behavior","title":"Custom retry behavior","text":"

    The retry_delay_seconds option accepts a list of delays for more custom retry behavior. The following task will wait for successively increasing intervals of 1, 10, and 100 seconds, respectively, before the next attempt starts:

    from prefect import task\n\n@task(retries=3, retry_delay_seconds=[1, 10, 100])\ndef some_task_with_manual_backoff_retries():\n   ...\n

    The retry_condition_fn option accepts a callable that returns a boolean. If the callable returns True, the task will be retried. If the callable returns False, the task will not be retried. The callable accepts three arguments \u2014 the task, the task run, and the state of the task run. The following task will retry on HTTP status codes other than 401 or 404:

    import httpx\nfrom prefect import flow, task\n\ndef retry_handler(task, task_run, state) -> bool:\n    \"\"\"This is a custom retry handler to handle when we want to retry a task\"\"\"\n    try:\n        # Attempt to get the result of the task\n        state.result()\n    except httpx.HTTPStatusError as exc:\n        # Retry on any HTTP status code that is not 401 or 404\n        do_not_retry_on_these_codes = [401, 404]\n        return exc.response.status_code not in do_not_retry_on_these_codes\n    except httpx.ConnectError:\n        # Do not retry\n        return False\n\n    # For any other exception, retry\n    return True\n\n@task(retries=1, retry_condition_fn=retry_handler)\ndef my_api_call_task(url):\n    response = httpx.get(url)\n    response.raise_for_status()\n    return response.json()\n\n@flow\ndef get_data_flow(url):\n    my_api_call_task(url=url)\n\nif __name__ == \"__main__\":\n    get_data_flow(url=\"https://httpbin.org/status/503\")\n

    Additionally, you can pass a callable that accepts the number of retries as an argument and returns a list. Prefect includes an exponential_backoff utility that will automatically generate a list of retry delays that correspond to an exponential backoff retry strategy. The following flow will wait for 10, 20, then 40 seconds before each retry.

    from prefect import task\nfrom prefect.tasks import exponential_backoff\n\n@task(retries=3, retry_delay_seconds=exponential_backoff(backoff_factor=10))\ndef some_task_with_exponential_backoff_retries():\n   ...\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#advanced-topic-adding-jitter","title":"Advanced topic: adding \"jitter\"","text":"

    While using exponential backoff, you may also want to add jitter to the delay times. Jitter is a random amount of time added to retry periods that helps prevent \"thundering herd\" scenarios, which is when many tasks all retry at the exact same time, potentially overwhelming systems.

    The retry_jitter_factor option can be used to add variance to the base delay. For example, a retry delay of 10 seconds with a retry_jitter_factor of 0.5 will be allowed to delay up to 15 seconds. Large values of retry_jitter_factor provide more protection against \"thundering herds,\" while keeping the average retry delay time constant. For example, the following task adds jitter to its exponential backoff so the retry delays will vary up to a maximum delay time of 20, 40, and 80 seconds respectively.

    from prefect import task\nfrom prefect.tasks import exponential_backoff\n\n@task(\n    retries=3,\n    retry_delay_seconds=exponential_backoff(backoff_factor=10),\n    retry_jitter_factor=1,\n)\ndef some_task_with_exponential_backoff_retries():\n   ...\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#configuring-retry-behavior-globally-with-settings","title":"Configuring retry behavior globally with settings","text":"

    You can also set retries and retry delays by using the following global settings. These settings will not override the retries or retry_delay_seconds that are set in the flow or task decorator.

    prefect config set PREFECT_FLOW_DEFAULT_RETRIES=2\nprefect config set PREFECT_TASK_DEFAULT_RETRIES=2\nprefect config set PREFECT_FLOW_DEFAULT_RETRY_DELAY_SECONDS = [1, 10, 100]\nprefect config set PREFECT_TASK_DEFAULT_RETRY_DELAY_SECONDS = [1, 10, 100]\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#caching","title":"Caching","text":"

    Caching refers to the ability of a task run to reflect a finished state without actually running the code that defines the task. This allows you to efficiently reuse results of tasks that may be expensive to run with every flow run, or reuse cached results if the inputs to a task have not changed.

    To determine whether a task run should retrieve a cached state, we use \"cache keys\". A cache key is a string value that indicates if one run should be considered identical to another. When a task run with a cache key finishes, we attach that cache key to the state. When each task run starts, Prefect checks for states with a matching cache key. If a state with an identical key is found, Prefect will use the cached state instead of running the task again.

    To enable caching, specify a cache_key_fn \u2014 a function that returns a cache key \u2014 on your task. You may optionally provide a cache_expiration timedelta indicating when the cache expires. If you do not specify a cache_expiration, the cache key does not expire.

    You can define a task that is cached based on its inputs by using the Prefect task_input_hash. This is a task cache key implementation that hashes all inputs to the task using a JSON or cloudpickle serializer. If the task inputs do not change, the cached results are used rather than running the task until the cache expires.

    Note that, if any arguments are not JSON serializable, the pickle serializer is used as a fallback. If cloudpickle fails, task_input_hash returns a null key indicating that a cache key could not be generated for the given inputs.

    In this example, until the cache_expiration time ends, as long as the input to hello_task() remains the same when it is called, the cached return value is returned. In this situation the task is not rerun. However, if the input argument value changes, hello_task() runs using the new input.

    from datetime import timedelta\nfrom prefect import flow, task\nfrom prefect.tasks import task_input_hash\n\n@task(cache_key_fn=task_input_hash, cache_expiration=timedelta(days=1))\ndef hello_task(name_input):\n    # Doing some work\n    print(\"Saying hello\")\n    return \"hello \" + name_input\n\n@flow\ndef hello_flow(name_input):\n    hello_task(name_input)\n

    Alternatively, you can provide your own function or other callable that returns a string cache key. A generic cache_key_fn is a function that accepts two positional arguments:

    • The first argument corresponds to the TaskRunContext, which stores task run metadata in the attributes task_run_id, flow_run_id, and task.
    • The second argument corresponds to a dictionary of input values to the task. For example, if your task is defined with signature fn(x, y, z) then the dictionary will have keys \"x\", \"y\", and \"z\" with corresponding values that can be used to compute your cache key.

    Note that the cache_key_fn is not defined as a @task.

    Task cache keys

    By default, a task cache key is limited to 2000 characters, specified by the PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH setting.

    from prefect import task, flow\n\ndef static_cache_key(context, parameters):\n    # return a constant\n    return \"static cache key\"\n\n@task(cache_key_fn=static_cache_key)\ndef cached_task():\n    print('running an expensive operation')\n    return 42\n\n@flow\ndef test_caching():\n    cached_task()\n    cached_task()\n    cached_task()\n

    In this case, there's no expiration for the cache key, and no logic to change the cache key, so cached_task() only runs once.

    >>> test_caching()\nrunning an expensive operation\n>>> test_caching()\n>>> test_caching()\n

    When each task run requested to enter a Running state, it provided its cache key computed from the cache_key_fn. The Prefect backend identified that there was a COMPLETED state associated with this key and instructed the run to immediately enter the same COMPLETED state, including the same return values.

    A real-world example might include the flow run ID from the context in the cache key so only repeated calls in the same flow run are cached.

    def cache_within_flow_run(context, parameters):\n    return f\"{context.task_run.flow_run_id}-{task_input_hash(context, parameters)}\"\n\n@task(cache_key_fn=cache_within_flow_run)\ndef cached_task():\n    print('running an expensive operation')\n    return 42\n

    Task results, retries, and caching

    Task results are cached in memory during a flow run and persisted to the location specified by the PREFECT_LOCAL_STORAGE_PATH setting. As a result, task caching between flow runs is currently limited to flow runs with access to that local storage path.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#refreshing-the-cache","title":"Refreshing the cache","text":"

    Sometimes, you want a task to update the data associated with its cache key instead of using the cache. This is a cache \"refresh\".

    The refresh_cache option can be used to enable this behavior for a specific task:

    import random\n\n\ndef static_cache_key(context, parameters):\n    # return a constant\n    return \"static cache key\"\n\n\n@task(cache_key_fn=static_cache_key, refresh_cache=True)\ndef caching_task():\n    return random.random()\n

    When this task runs, it will always update the cache key instead of using the cached value. This is particularly useful when you have a flow that is responsible for updating the cache.

    If you want to refresh the cache for all tasks, you can use the PREFECT_TASKS_REFRESH_CACHE setting. Setting PREFECT_TASKS_REFRESH_CACHE=true will change the default behavior of all tasks to refresh. This is particularly useful if you want to rerun a flow without cached results.

    If you have tasks that should not refresh when this setting is enabled, you may explicitly set refresh_cache to False. These tasks will never refresh the cache \u2014 if a cache key exists it will be read, not updated. Note that, if a cache key does not exist yet, these tasks can still write to the cache.

    @task(cache_key_fn=static_cache_key, refresh_cache=False)\ndef caching_task():\n    return random.random()\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#timeouts","title":"Timeouts","text":"

    Task timeouts are used to prevent unintentional long-running tasks. When the duration of execution for a task exceeds the duration specified in the timeout, a timeout exception will be raised and the task will be marked as failed. In the UI, the task will be visibly designated as TimedOut. From the perspective of the flow, the timed-out task will be treated like any other failed task.

    Timeout durations are specified using the timeout_seconds keyword argument.

    from prefect import task, get_run_logger\nimport time\n\n@task(timeout_seconds=1)\ndef show_timeouts():\n    logger = get_run_logger()\n    logger.info(\"I will execute\")\n    time.sleep(5)\n    logger.info(\"I will not execute\")\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#task-results","title":"Task results","text":"

    Depending on how you call tasks, they can return different types of results and optionally engage the use of a task runner.

    Any task can return:

    • Data\u200a, such as int, str, dict, list, and so on \u2014 \u200athis is the default behavior any time you call your_task().
    • PrefectFuture \u2014 \u200athis is achieved by calling your_task.submit(). A PrefectFuture contains both data and State
    • Prefect State \u200a\u2014 anytime you call your task or flow with the argument return_state=True, it will directly return a state you can use to build custom behavior based on a state change you care about, such as task or flow failing or retrying.

    To run your task with a task runner, you must call the task with .submit().

    See state returned values for examples.

    Task runners are optional

    If you just need the result from a task, you can simply call the task from your flow. For most workflows, the default behavior of calling a task directly and receiving a result is all you'll need.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#wait-for","title":"Wait for","text":"

    To create a dependency between two tasks that do not exchange data, but one needs to wait for the other to finish, use the special wait_for keyword argument:

    @task\ndef task_1():\n    pass\n\n@task\ndef task_2():\n    pass\n\n@flow\ndef my_flow():\n    x = task_1()\n\n    # task 2 will wait for task_1 to complete\n    y = task_2(wait_for=[x])\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#map","title":"Map","text":"

    Prefect provides a .map() implementation that automatically creates a task run for each element of its input data. Mapped tasks represent the computations of many individual children tasks.

    The simplest Prefect map takes a tasks and applies it to each element of its inputs.

    from prefect import flow, task\n\n@task\ndef print_nums(nums):\n    for n in nums:\n        print(n)\n\n@task\ndef square_num(num):\n    return num**2\n\n@flow\ndef map_flow(nums):\n    print_nums(nums)\n    squared_nums = square_num.map(nums) \n    print_nums(squared_nums)\n\nmap_flow([1,2,3,5,8,13])\n

    Prefect also supports unmapped arguments, allowing you to pass static values that don't get mapped over.

    from prefect import flow, task\n\n@task\ndef add_together(x, y):\n    return x + y\n\n@flow\ndef sum_it(numbers, static_value):\n    futures = add_together.map(numbers, static_value)\n    return futures\n\nsum_it([1, 2, 3], 5)\n

    If your static argument is an iterable, you'll need to wrap it with unmapped to tell Prefect that it should be treated as a static value.

    from prefect import flow, task, unmapped\n\n@task\ndef sum_plus(x, static_iterable):\n    return x + sum(static_iterable)\n\n@flow\ndef sum_it(numbers, static_iterable):\n    futures = sum_plus.map(numbers, static_iterable)\n    return futures\n\nsum_it([4, 5, 6], unmapped([1, 2, 3]))\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#async-tasks","title":"Async tasks","text":"

    Prefect also supports asynchronous task and flow definitions by default. All of the standard rules of async apply:

    import asyncio\n\nfrom prefect import task, flow\n\n@task\nasync def print_values(values):\n    for value in values:\n        await asyncio.sleep(1) # yield\n        print(value, end=\" \")\n\n@flow\nasync def async_flow():\n    await print_values([1, 2])  # runs immediately\n    coros = [print_values(\"abcd\"), print_values(\"6789\")]\n\n    # asynchronously gather the tasks\n    await asyncio.gather(*coros)\n\nasyncio.run(async_flow())\n

    Note, if you are not using asyncio.gather, calling .submit() is required for asynchronous execution on the ConcurrentTaskRunner.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#task-run-concurrency-limits","title":"Task run concurrency limits","text":"

    There are situations in which you want to actively prevent too many tasks from running simultaneously. For example, if many tasks across multiple flows are designed to interact with a database that only allows 10 connections, you want to make sure that no more than 10 tasks that connect to this database are running at any given time.

    Prefect has built-in functionality for achieving this: task concurrency limits.

    Task concurrency limits use task tags. You can specify an optional concurrency limit as the maximum number of concurrent task runs in a Running state for tasks with a given tag. The specified concurrency limit applies to any task to which the tag is applied.

    If a task has multiple tags, it will run only if all tags have available concurrency.

    Tags without explicit limits are considered to have unlimited concurrency.

    0 concurrency limit aborts task runs

    Currently, if the concurrency limit is set to 0 for a tag, any attempt to run a task with that tag will be aborted instead of delayed.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#execution-behavior","title":"Execution behavior","text":"

    Task tag limits are checked whenever a task run attempts to enter a Running state.

    If there are no concurrency slots available for any one of your task's tags, the transition to a Running state will be delayed and the client is instructed to try entering a Running state again in 30 seconds (or the value specified by the PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS setting).

    Concurrency limits in subflows

    Using concurrency limits on task runs in subflows can cause deadlocks. As a best practice, configure your tags and concurrency limits to avoid setting limits on task runs in subflows.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#configuring-concurrency-limits","title":"Configuring concurrency limits","text":"

    Flow run concurrency limits are set at a work pool and/or work queue level

    While task run concurrency limits are configured via tags (as shown below), flow run concurrency limits are configured via work pools and/or work queues.

    You can set concurrency limits on as few or as many tags as you wish. You can set limits through:

    • Prefect CLI
    • Prefect API by using PrefectClient Python client
    • Prefect server UI or Prefect Cloud
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#cli","title":"CLI","text":"

    You can create, list, and remove concurrency limits by using Prefect CLI concurrency-limit commands.

    $ prefect concurrency-limit [command] [arguments]\n
    Command Description create Create a concurrency limit by specifying a tag and limit. delete Delete the concurrency limit set on the specified tag. inspect View details about a concurrency limit set on the specified tag. ls View all defined concurrency limits.

    For example, to set a concurrency limit of 10 on the 'small_instance' tag:

    $ prefect concurrency-limit create small_instance 10\n

    To delete the concurrency limit on the 'small_instance' tag:

    $ prefect concurrency-limit delete small_instance\n

    To view details about the concurrency limit on the 'small_instance' tag:

    $ prefect concurrency-limit inspect small_instance\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#python-client","title":"Python client","text":"

    To update your tag concurrency limits programmatically, use PrefectClient.orchestration.create_concurrency_limit.

    create_concurrency_limit takes two arguments:

    • tag specifies the task tag on which you're setting a limit.
    • concurrency_limit specifies the maximum number of concurrent task runs for that tag.

    For example, to set a concurrency limit of 10 on the 'small_instance' tag:

    from prefect import get_client\n\nasync with get_client() as client:\n    # set a concurrency limit of 10 on the 'small_instance' tag\n    limit_id = await client.create_concurrency_limit(\n        tag=\"small_instance\", \n        concurrency_limit=10\n        )\n

    To remove all concurrency limits on a tag, use PrefectClient.delete_concurrency_limit_by_tag, passing the tag:

    async with get_client() as client:\n    # remove a concurrency limit on the 'small_instance' tag\n    await client.delete_concurrency_limit_by_tag(tag=\"small_instance\")\n

    If you wish to query for the currently set limit on a tag, use PrefectClient.read_concurrency_limit_by_tag, passing the tag:

    To see all of your limits across all of your tags, use PrefectClient.read_concurrency_limits.

    async with get_client() as client:\n    # query the concurrency limit on the 'small_instance' tag\n    limit = await client.read_concurrency_limit_by_tag(tag=\"small_instance\")\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/work-pools/","title":"Work Pools & Workers","text":"

    Work pools and workers bridge the Prefect orchestration environment with your execution environment. When a deployment creates a flow run, it is submitted to a specific work pool for scheduling. A worker running in the execution environment can poll its respective work pool for new runs to execute, or the work pool can submit flow runs to serverless infrastructure directly, depending on your configuration.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-pool-overview","title":"Work pool overview","text":"

    Work pools organize work for execution. Work pools have types corresponding to the infrastructure that will execute the flow code, as well as the delivery method of work to that environment. Pull work pools require workers (or less ideally, agents) to poll the work pool for flow runs to execute. Push work pools can submit runs directly to your serverless infrastructure providers such as Google Cloud Run, Azure Container Instances, and AWS ECS without the need for an agent or worker. Managed work pools are administered by Prefect and handle the submission and execution of code on your behalf.

    Work pools are like pub/sub topics

    It's helpful to think of work pools as a way to coordinate (potentially many) deployments with (potentially many) workers through a known channel: the pool itself. This is similar to how \"topics\" are used to connect producers and consumers in a pub/sub or message-based system. By switching a deployment's work pool, users can quickly change the worker that will execute their runs, making it easy to promote runs through environments or even debug locally.

    In addition, users can control aspects of work pool behavior, such as how many runs the pool allows to be run concurrently or pausing delivery entirely. These options can be modified at any time, and any workers requesting work for a specific pool will only see matching flow runs.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-pool-configuration","title":"Work pool configuration","text":"

    You can configure work pools by using:

    • Prefect CLI commands
    • Prefect Python API
    • Prefect UI

    To manage work pools in the UI, click the Work Pools icon. This displays a list of currently configured work pools.

    You can pause a work pool from this page by using the toggle.

    Select the + button to create a new work pool. You'll be able to specify the details for work served by this work pool.

    To create a work pool via the Prefect CLI, use the prefect work-pool create command:

    prefect work-pool create [OPTIONS] NAME\n

    NAME is a required, unique name for the work pool.

    Optional configuration parameters you can specify to filter work on the pool include:

    Option Description --paused If provided, the work pool will be created in a paused state. --type The type of infrastructure that can execute runs from this work pool. --set-as-default Whether to use the created work pool as the local default for deployment. --base-job-template The path to a JSON file containing the base job template to use. If unspecified, Prefect will use the default base job template for the given worker type.

    For example, to create a work pool called test-pool, you would run this command:

    prefect work-pool create test-pool\n
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-pool-types","title":"Work pool types","text":"

    If you don't use the --type flag to specify an infrastructure type, you are prompted to select from the following options:

    Prefect CloudPrefect server instance Infrastructure Type Description Prefect Agent Execute flow runs on heterogeneous infrastructure using infrastructure blocks. Local Subprocess Execute flow runs as subprocesses on a worker. Works well for local execution when first getting started. AWS Elastic Container Service Execute flow runs within containers on AWS ECS. Works with EC2 and Fargate clusters. Requires an AWS account. Azure Container Instances Execute flow runs within containers on Azure's Container Instances service. Requires an Azure account. Docker Execute flow runs within Docker containers. Works well for managing flow execution environments via Docker images. Requires access to a running Docker daemon. Google Cloud Run Execute flow runs within containers on Google Cloud Run. Requires a Google Cloud Platform account. Google Cloud Run V2 Execute flow runs within containers on Google Cloud Run (V2 API). Requires a Google Cloud Platform account. Google Vertex AI Execute flow runs within containers on Google Vertex AI. Requires a Google Cloud Platform account. Kubernetes Execute flow runs within jobs scheduled on a Kubernetes cluster. Requires a Kubernetes cluster. Google Cloud Run - Push Execute flow runs within containers on Google Cloud Run. Requires a Google Cloud Platform account. Flow runs are pushed directly to your environment, without the need for a Prefect worker. AWS Elastic Container Service - Push Execute flow runs within containers on AWS ECS. Works with existing ECS clusters and serverless execution via AWS Fargate. Requires an AWS account. Flow runs are pushed directly to your environment, without the need for a Prefect worker. Azure Container Instances - Push Execute flow runs within containers on Azure's Container Instances service. Requires an Azure account. Flow runs are pushed directly to your environment, without the need for a Prefect worker. Prefect Managed Execute flow runs within containers on Prefect managed infrastructure. Infrastructure Type Description Prefect Agent Execute flow runs on heterogeneous infrastructure using infrastructure blocks. Local Subprocess Execute flow runs as subprocesses on a worker. Works well for local execution when first getting started. AWS Elastic Container Service Execute flow runs within containers on AWS ECS. Works with EC2 and Fargate clusters. Requires an AWS account. Azure Container Instances Execute flow runs within containers on Azure's Container Instances service. Requires an Azure account. Docker Execute flow runs within Docker containers. Works well for managing flow execution environments via Docker images. Requires access to a running Docker daemon. Google Cloud Run Execute flow runs within containers on Google Cloud Run. Requires a Google Cloud Platform account. Google Cloud Run V2 Execute flow runs within containers on Google Cloud Run (V2 API). Requires a Google Cloud Platform account. Google Vertex AI Execute flow runs within containers on Google Vertex AI. Requires a Google Cloud Platform account. Kubernetes Execute flow runs within jobs scheduled on a Kubernetes cluster. Requires a Kubernetes cluster.

    On success, the command returns the details of the newly created work pool.

    Created work pool with properties:\n    name - 'test-pool'\n    id - a51adf8c-58bb-4949-abe6-1b87af46eabd\n    concurrency limit - None\n\nStart a worker to pick up flows from the work pool:\n    prefect worker start -p 'test-pool'\n\nInspect the work pool:\n    prefect work-pool inspect 'test-pool'\n

    Set a work pool as the default for new deployments by adding the --set-as-default flag.

    Which would result in output similar to the following:

    Set 'test-pool' as default work pool for profile 'default'\n\nTo change your default work pool, run:\n\n        prefect config set PREFECT_DEFAULT_WORK_POOL_NAME=<work-pool-name>\n

    To update a work pool via the Prefect CLI, use the prefect work-pool update command:

    prefect work-pool update [OPTIONS] NAME\n

    NAME is the name of the work pool to update.

    Optional configuration parameters you can specify to update the work pool include:

    Option Description --base-job-template The path to a JSON file containing the base job template to use. If unspecified, Prefect will use the default base job template for the given worker type. --description A description of the work pool. --concurrency-limit The maximum number of flow runs to run simultaneously in the work pool.

    Managing work pools in CI/CD

    You can version control your base job template by committing it as a JSON file to your repository and control updates to your work pools' base job templates by using the prefect work-pool update command in your CI/CD pipeline. For example, you could use the following command to update a work pool's base job template to the contents of a file named base-job-template.json:

    prefect work-pool update --base-job-template base-job-template.json my-work-pool\n
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#base-job-template","title":"Base job template","text":"

    Each work pool has a base job template that allows the customization of the behavior of the worker executing flow runs from the work pool.

    The base job template acts as a contract defining the configuration passed to the worker for each flow run and the options available to deployment creators to customize worker behavior per deployment.

    A base job template comprises a job_configuration section and a variables section.

    The variables section defines the fields available to be customized per deployment. The variables section follows the OpenAPI specification, which allows work pool creators to place limits on provided values (type, minimum, maximum, etc.).

    The job configuration section defines how values provided for fields in the variables section should be translated into the configuration given to a worker when executing a flow run.

    The values in the job_configuration can use placeholders to reference values provided in the variables section. Placeholders are declared using double curly braces, e.g., {{ variable_name }}. job_configuration values can also be hard-coded if the value should not be customizable.

    Each worker type is configured with a default base job template, making it easy to start with a work pool. The default base template defines fields that can be edited on a per-deployment basis or for the entire work pool via the Prefect API and UI.

    For example, if we create a process work pool named 'above-ground' via the CLI:

    prefect work-pool create --type process above-ground\n

    We see these configuration options available in the Prefect UI:

    For a process work pool with the default base job template, we can set environment variables for spawned processes, set the working directory to execute flows, and control whether the flow run output is streamed to workers' standard output. You can also see an example of JSON formatted base job template with the 'Advanced' tab.

    You can examine the default base job template for a given worker type by running:

    prefect work-pool get-default-base-job-template --type process\n
    {\n  \"job_configuration\": {\n    \"command\": \"{{ command }}\",\n    \"env\": \"{{ env }}\",\n    \"labels\": \"{{ labels }}\",\n    \"name\": \"{{ name }}\",\n    \"stream_output\": \"{{ stream_output }}\",\n    \"working_dir\": \"{{ working_dir }}\"\n  },\n  \"variables\": {\n    \"type\": \"object\",\n    \"properties\": {\n      \"name\": {\n        \"title\": \"Name\",\n        \"description\": \"Name given to infrastructure created by a worker.\",\n        \"type\": \"string\"\n      },\n      \"env\": {\n        \"title\": \"Environment Variables\",\n        \"description\": \"Environment variables to set when starting a flow run.\",\n        \"type\": \"object\",\n        \"additionalProperties\": {\n          \"type\": \"string\"\n        }\n      },\n      \"labels\": {\n        \"title\": \"Labels\",\n        \"description\": \"Labels applied to infrastructure created by a worker.\",\n        \"type\": \"object\",\n        \"additionalProperties\": {\n          \"type\": \"string\"\n        }\n      },\n      \"command\": {\n        \"title\": \"Command\",\n        \"description\": \"The command to use when starting a flow run. In most cases, this should be left blank and the command will be automatically generated by the worker.\",\n        \"type\": \"string\"\n      },\n      \"stream_output\": {\n        \"title\": \"Stream Output\",\n        \"description\": \"If enabled, workers will stream output from flow run processes to local standard output.\",\n        \"default\": true,\n        \"type\": \"boolean\"\n      },\n      \"working_dir\": {\n        \"title\": \"Working Directory\",\n        \"description\": \"If provided, workers will open flow run processes within the specified path as the working directory. Otherwise, a temporary directory will be created.\",\n        \"type\": \"string\",\n        \"format\": \"path\"\n      }\n    }\n  }\n}\n

    You can override each of these attributes on a per-deployment basis. When deploying a flow, you can specify these overrides in the work_pool.job_variables section of a deployment.yaml.

    If we wanted to turn off streaming output for a specific deployment, we could add the following to our deployment.yaml:

    work_pool:\n    name: above-ground  \n    job_variables:\n        stream_output: false\n

    Advanced Customization of the Base Job Template

    For advanced use cases, you can create work pools with fully customizable job templates. This customization is available when creating or editing a work pool on the 'Advanced' tab within the UI or when updating a work pool via the Prefect CLI.

    Advanced customization is useful anytime the underlying infrastructure supports a high degree of customization. In these scenarios a work pool job template allows you to expose a minimal and easy-to-digest set of options to deployment authors. Additionally, these options are the only customizable aspects for deployment infrastructure, which can be useful for restricting functionality in secure environments. For example, the kubernetes worker type allows users to specify a custom job template that can be used to configure the manifest that workers use to create jobs for flow execution.

    For more information and advanced configuration examples, see the Kubernetes Worker documentation.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#viewing-work-pools","title":"Viewing work pools","text":"

    At any time, users can see and edit configured work pools in the Prefect UI.

    To view work pools with the Prefect CLI, you can:

    • List (ls) all available pools
    • Inspect (inspect) the details of a single pool
    • Preview (preview) scheduled work for a single pool

    prefect work-pool ls lists all configured work pools for the server.

    prefect work-pool ls\n

    For example:

                                   Work pools\n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name       \u2503    Type        \u2503                                   ID \u2503 Concurrency Limit \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 barbeque   \u2502 docker         \u2502 72c0a101-b3e2-4448-b5f8-a8c5184abd17 \u2502 None              \u2502\n\u2502 k8s-pool   \u2502 kubernetes     \u2502 7b6e3523-d35b-4882-84a7-7a107325bb3f \u2502 None              \u2502\n\u2502 test-pool  \u2502 prefect-agent  \u2502 a51adf8c-58bb-4949-abe6-1b87af46eabd \u2502 None              |\n| my-pool    \u2502 process        \u2502 cd6ff9e8-bfd8-43be-9be3-69375f7a11cd \u2502 None              \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n                       (**) denotes a paused pool\n

    prefect work-pool inspect provides all configuration metadata for a specific work pool by ID.

    prefect work-pool inspect 'test-pool'\n

    Outputs information similar to the following:

    Workpool(\n    id='a51adf8c-58bb-4949-abe6-1b87af46eabd',\n    created='2 minutes ago',\n    updated='2 minutes ago',\n    name='test-pool',\n    filter=None,\n)\n

    prefect work-pool preview displays scheduled flow runs for a specific work pool by ID for the upcoming hour. The optional --hours flag lets you specify the number of hours to look ahead.

    prefect work-pool preview 'test-pool' --hours 12\n
    \u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Scheduled Star\u2026 \u2503 Run ID                     \u2503 Name         \u2503 Deployment ID               \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 2022-02-26 06:\u2026 \u2502 741483d4-dc90-4913-b88d-0\u2026 \u2502 messy-petrel \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 05:\u2026 \u2502 14e23a19-a51b-4833-9322-5\u2026 \u2502 unselfish-g\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 04:\u2026 \u2502 deb44d4d-5fa2-4f70-a370-e\u2026 \u2502 solid-ostri\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 03:\u2026 \u2502 07374b5c-121f-4c8d-9105-b\u2026 \u2502 sophisticat\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 02:\u2026 \u2502 545bc975-b694-4ece-9def-8\u2026 \u2502 gorgeous-mo\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 01:\u2026 \u2502 704f2d67-9dfa-4fb8-9784-4\u2026 \u2502 sassy-hedge\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 00:\u2026 \u2502 691312f0-d142-4218-b617-a\u2026 \u2502 sincere-moo\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-25 23:\u2026 \u2502 7cb3ff96-606b-4d8c-8a33-4\u2026 \u2502 curious-cat\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-25 22:\u2026 \u2502 3ea559fe-cb34-43b0-8090-1\u2026 \u2502 primitive-f\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-25 21:\u2026 \u2502 96212e80-426d-4bf4-9c49-e\u2026 \u2502 phenomenal-\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n                                   (**) denotes a late run\n
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-pool-status","title":"Work Pool Status","text":"

    Work pools have three statuses: READY, NOT_READY, and PAUSED. A work pool is considered ready if it has at least one online worker sending heartbeats to the work pool. If a work pool has no online workers, it is considered not ready to execute work. A work pool can be placed in a paused status manually by a user or via an automation. When a paused work pool is unpaused, it will be reassigned the appropriate status based on whether any workers are sending heartbeats.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#pausing-and-deleting-work-pools","title":"Pausing and deleting work pools","text":"

    A work pool can be paused at any time to stop the delivery of work to workers. Workers will not receive any work when polling a paused pool.

    To pause a work pool through the Prefect CLI, use the prefect work-pool pause command:

    prefect work-pool pause 'test-pool'\n

    To resume a work pool through the Prefect CLI, use the prefect work-pool resume command with the work pool name.

    To delete a work pool through the Prefect CLI, use the prefect work-pool delete command with the work pool name.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#managing-concurrency","title":"Managing concurrency","text":"

    Each work pool can optionally restrict concurrent runs of matching flows.

    For example, a work pool with a concurrency limit of 5 will only release new work if fewer than 5 matching runs are currently in a Running or Pending state. If 3 runs are Running or Pending, polling the pool for work will only result in 2 new runs, even if there are many more available, to ensure that the concurrency limit is not exceeded.

    When using the prefect work-pool Prefect CLI command to configure a work pool, the following subcommands set concurrency limits:

    • set-concurrency-limit sets a concurrency limit on a work pool.
    • clear-concurrency-limit clears any concurrency limits from a work pool.
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-queues","title":"Work queues","text":"

    Advanced topic

    Work queues do not require manual creation or configuration, because Prefect will automatically create them whenever needed. Managing work queues offers advanced control over how runs are executed.

    Each work pool has a \"default\" queue that all work will be sent to by default. Additional queues can be added to a work pool. Work queues enable greater control over work delivery through fine grained priority and concurrency. Each work queue has a priority indicated by a unique positive integer. Lower numbers take greater priority in the allocation of work. Accordingly, new queues can be added without changing the rank of the higher-priority queues (e.g. no matter how many queues you add, the queue with priority 1 will always be the highest priority).

    Work queues can also have their own concurrency limits. Note that each queue is also subject to the global work pool concurrency limit, which cannot be exceeded.

    Together work queue priority and concurrency enable precise control over work. For example, a pool may have three queues: A \"low\" queue with priority 10 and no concurrency limit, a \"high\" queue with priority 5 and a concurrency limit of 3, and a \"critical\" queue with priority 1 and a concurrency limit of 1. This arrangement would enable a pattern in which there are two levels of priority, \"high\" and \"low\" for regularly scheduled flow runs, with the remaining \"critical\" queue for unplanned, urgent work, such as a backfill.

    Priority is evaluated to determine the order in which flow runs are submitted for execution. If all flow runs are capable of being executed with no limitation due to concurrency or otherwise, priority is still used to determine order of submission, but there is no impact to execution. If not all flow runs can be executed, usually as a result of concurrency limits, priority is used to determine which queues receive precedence to submit runs for execution.

    Priority for flow run submission proceeds from the highest priority to the lowest priority. In the preceding example, all work from the \"critical\" queue (priority 1) will be submitted, before any work is submitted from \"high\" (priority 5). Once all work has been submitted from priority queue \"critical\", work from the \"high\" queue will begin submission.

    If new flow runs are received on the \"critical\" queue while flow runs are still in scheduled on the \"high\" and \"low\" queues, flow run submission goes back to ensuring all scheduled work is first satisfied from the highest priority queue, until it is empty, in waterfall fashion.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#local-debugging","title":"Local debugging","text":"

    As long as your deployment's infrastructure block supports it, you can use work pools to temporarily send runs to a worker running on your local machine for debugging by running prefect worker start -p my-local-machine and updating the deployment's work pool to my-local-machine.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#worker-overview","title":"Worker overview","text":"

    Workers are lightweight polling services that retrieve scheduled runs from a work pool and execute them.

    Workers are similar to agents, but offer greater control over infrastructure configuration and the ability to route work to specific types of execution environments.

    Workers each have a type corresponding to the execution environment to which they will submit flow runs. Workers are only able to poll work pools that match their type. As a result, when deployments are assigned to a work pool, you know in which execution environment scheduled flow runs for that deployment will run.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#worker-types","title":"Worker types","text":"

    Below is a list of available worker types. Note that most worker types will require installation of an additional package.

    Worker Type Description Required Package process Executes flow runs in subprocesses kubernetes Executes flow runs as Kubernetes jobs prefect-kubernetes docker Executes flow runs within Docker containers prefect-docker ecs Executes flow runs as ECS tasks prefect-aws cloud-run Executes flow runs as Google Cloud Run jobs prefect-gcp vertex-ai Executes flow runs as Google Cloud Vertex AI jobs prefect-gcp azure-container-instance Execute flow runs in ACI containers prefect-azure

    If you don\u2019t see a worker type that meets your needs, consider developing a new worker type!

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#worker-options","title":"Worker options","text":"

    Workers poll for work from one or more queues within a work pool. If the worker references a work queue that doesn't exist, it will be created automatically. The worker CLI is able to infer the worker type from the work pool. Alternatively, you can also specify the worker type explicitly. If you supply the worker type to the worker CLI, a work pool will be created automatically if it doesn't exist (using default job settings).

    Configuration parameters you can specify when starting a worker include:

    Option Description --name, -n The name to give to the started worker. If not provided, a unique name will be generated. --pool, -p The work pool the started worker should poll. --work-queue, -q One or more work queue names for the worker to pull from. If not provided, the worker will pull from all work queues in the work pool. --type, -t The type of worker to start. If not provided, the worker type will be inferred from the work pool. --prefetch-seconds The amount of time before a flow run's scheduled start time to begin submission. Default is the value of PREFECT_WORKER_PREFETCH_SECONDS. --run-once Only run worker polling once. By default, the worker runs forever. --limit, -l The maximum number of flow runs to start simultaneously. --with-healthcheck Start a healthcheck server for the worker. --install-policy Install policy to use workers from Prefect integration packages.

    You must start a worker within an environment that can access or create the infrastructure needed to execute flow runs. The worker will deploy flow runs to the infrastructure corresponding to the worker type. For example, if you start a worker with type kubernetes, the worker will deploy flow runs to a Kubernetes cluster.

    Prefect must be installed in execution environments

    Prefect must be installed in any environment (virtual environment, Docker container, etc.) where you intend to run the worker or execute a flow run.

    PREFECT_API_URL and PREFECT_API_KEYsettings for workers

    PREFECT_API_URL must be set for the environment in which your worker is running. You must also have a user or service account with the Worker role, which can be configured by setting the PREFECT_API_KEY.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#worker-status","title":"Worker status","text":"

    Workers have two statuses: ONLINE and OFFLINE. A worker is online if it sends regular heartbeat messages to the Prefect API. If a worker has missed three heartbeats, it is considered offline. By default, a worker is considered offline a maximum of 90 seconds after it stopped sending heartbeats, but the threshold can be configured via the PREFECT_WORKER_HEARTBEAT_SECONDS setting.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#starting-a-worker","title":"Starting a worker","text":"

    Use the prefect worker start CLI command to start a worker. You must pass at least the work pool name. If the work pool does not exist, it will be created if the --type flag is used.

    prefect worker start -p [work pool name]\n

    For example:

    prefect worker start -p \"my-pool\"\n

    Results in output like this:

    Discovered worker type 'process' for work pool 'my-pool'.\nWorker 'ProcessWorker 65716280-96f8-420b-9300-7e94417f2673' started!\n

    In this case, Prefect automatically discovered the worker type from the work pool. To create a work pool and start a worker in one command, use the --type flag:

    prefect worker start -p \"my-pool\" --type \"process\"\n
    Worker 'ProcessWorker d24f3768-62a9-4141-9480-a056b9539a25' started!\n06:57:53.289 | INFO    | prefect.worker.process.processworker d24f3768-62a9-4141-9480-a056b9539a25 - Worker pool 'my-pool' created.\n

    In addition, workers can limit the number of flow runs they will start simultaneously with the --limit flag. For example, to limit a worker to five concurrent flow runs:

    prefect worker start --pool \"my-pool\" --limit 5\n
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#configuring-prefetch","title":"Configuring prefetch","text":"

    By default, the worker begins submitting flow runs a short time (10 seconds) before they are scheduled to run. This behavior allows time for the infrastructure to be created so that the flow run can start on time.

    In some cases, infrastructure will take longer than 10 seconds to start the flow run. The prefetch can be increased using the --prefetch-seconds option or the PREFECT_WORKER_PREFETCH_SECONDS setting.

    If this value is more than the amount of time it takes for the infrastructure to start, the flow run will wait until its scheduled start time.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#polling-for-work","title":"Polling for work","text":"

    Workers poll for work every 15 seconds by default. This interval is configurable in your profile settings with the PREFECT_WORKER_QUERY_SECONDS setting.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#install-policy","title":"Install policy","text":"

    The Prefect CLI can install the required package for Prefect-maintained worker types automatically. You can configure this behavior with the --install-policy option. The following are valid install policies

    Install Policy Description always Always install the required package. Will update the required package to the most recent version if already installed. if-not-present Install the required package if it is not already installed. never Never install the required package. prompt Prompt the user to choose whether to install the required package. This is the default install policy. If prefect worker start is run non-interactively, the prompt install policy will behave the same as never.","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#additional-resources","title":"Additional resources","text":"

    See how to daemonize a Prefect worker in this guide.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"contributing/overview/","title":"Contributing","text":"

    Thanks for considering contributing to Prefect!

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#setting-up-a-development-environment","title":"Setting up a development environment","text":"

    First, you'll need to download the source code and install an editable version of the Python package:

    # Clone the repository\ngit clone https://github.com/PrefectHQ/prefect.git\ncd prefect\n\n# We recommend using a virtual environment\n\npython -m venv .venv\nsource .venv/bin/activate\n\n# Install the package with development dependencies\n\npip install -e \".[dev]\"\n\n# Setup pre-commit hooks for required formatting\n\npre-commit install\n

    If you don't want to install the pre-commit hooks, you can manually install the formatting dependencies with:

    pip install $(./scripts/precommit-versions.py)\n

    You'll need to run black and ruff before a contribution can be accepted.

    After installation, you can run the test suite with pytest:

    # Run all the tests\npytest tests\n\n# Run a subset of tests\n\npytest tests/test_flows.py\n

    Building the Prefect UI

    If you intend to run a local Prefect server during development, you must first build the UI. See UI development for instructions.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#prefect-code-of-conduct","title":"Prefect Code of Conduct","text":"","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#our-pledge","title":"Our Pledge","text":"

    In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, religion, or sexual identity and orientation.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#our-standards","title":"Our Standards","text":"

    Examples of behavior that contributes to creating a positive environment include:

    • Using welcoming and inclusive language
    • Being respectful of differing viewpoints and experiences
    • Gracefully accepting constructive criticism
    • Focusing on what is best for the community
    • Showing empathy towards other community members

    Examples of unacceptable behavior by participants include:

    • The use of sexualized language or imagery and unwelcome sexual attention or advances
    • Trolling, insulting/derogatory comments, and personal or political attacks
    • Public or private harassment
    • Publishing others' private information, such as a physical or electronic address, without explicit permission
    • Other conduct which could reasonably be considered inappropriate in a professional setting
    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#our-responsibilities","title":"Our Responsibilities","text":"

    Project maintainers are responsible for clarifying the standards of acceptable behavior and are expected to take appropriate and fair corrective action in response to any instances of unacceptable behavior.

    Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, or to ban temporarily or permanently any contributor for other behaviors that they deem inappropriate, threatening, offensive, or harmful.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#scope","title":"Scope","text":"

    This Code of Conduct applies within all project spaces, and it also applies when an individual is representing the project or its community in public spaces. Examples of representing a project or community include using an official project e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event. Representation of a project may be further defined and clarified by project maintainers.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#enforcement","title":"Enforcement","text":"

    Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by contacting Chris White at chris@prefect.io. All complaints will be reviewed and investigated and will result in a response that is deemed necessary and appropriate to the circumstances. The project team is obligated to maintain confidentiality with regard to the reporter of an incident. Further details of specific enforcement policies may be posted separately.

    Project maintainers who do not follow or enforce the Code of Conduct in good faith may face temporary or permanent repercussions as determined by other members of the project's leadership.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#attribution","title":"Attribution","text":"

    This Code of Conduct is adapted from the Contributor Covenant, version 1.4, available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html

    For answers to common questions about this code of conduct, see https://www.contributor-covenant.org/faq

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#developer-tooling","title":"Developer tooling","text":"

    The Prefect CLI provides several helpful commands to aid development.

    Start all services with hot-reloading on code changes (requires UI dependencies to be installed):

    prefect dev start\n

    Start a Prefect API that reloads on code changes:

    prefect dev api\n

    Start a Prefect worker that reloads on code changes:

    prefect dev agent\n
    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#ui-development","title":"UI development","text":"

    Developing the Prefect UI requires that npm is installed.

    Start a development UI that reloads on code changes:

    prefect dev ui\n

    Build the static UI (the UI served by prefect server start):

    prefect dev build-ui\n
    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#docs-development","title":"Docs Development","text":"

    Prefect uses mkdocs for the docs website and the mkdocs-material theme. While we use mkdocs-material-insiders for production, builds can still happen without the extra plugins. Deploy previews are available on pull requests, so you'll be able to browse the final look of your changes before merging.

    To build the docs:

    mkdocs build\n

    To serve the docs locally at http://127.0.0.1:8000/:

    mkdocs serve\n

    For additional mkdocs help and options:

    mkdocs --help\n

    We use the mkdocs-material theme. To add additional JavaScript or CSS to the docs, please see the theme documentation here.

    Internal developers can install the production theme by running:

    pip install -e git+https://github.com/PrefectHQ/mkdocs-material-insiders.git#egg=mkdocs-material\nmkdocs build # or mkdocs build --config-file mkdocs.insiders.yml if needed\n
    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#kubernetes-development","title":"Kubernetes development","text":"

    Generate a manifest to deploy a development API to a local kubernetes cluster:

    prefect dev kubernetes-manifest\n

    To access the Prefect UI running in a Kubernetes cluster, use the kubectl port-forward command to forward a port on your local machine to an open port within the cluster. For example:

    kubectl port-forward deployment/prefect-dev 4200:4200\n

    This forwards port 4200 on the default internal loop IP for localhost to the Prefect server deployment.

    To tell the local prefect command how to communicate with the Prefect API running in Kubernetes, set the PREFECT_API_URL environment variable:

    export PREFECT_API_URL=http://localhost:4200/api\n

    Since you previously configured port forwarding for the localhost port to the Kubernetes environment, you\u2019ll be able to interact with the Prefect API running in Kubernetes when using local Prefect CLI commands.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#adding-database-migrations","title":"Adding Database Migrations","text":"

    To make changes to a table, first update the SQLAlchemy model in src/prefect/server/database/orm_models.py. For example, if you wanted to add a new column to the flow_run table, you would add a new column to the FlowRun model:

    # src/prefect/server/database/orm_models.py\n\n@declarative_mixin\nclass ORMFlowRun(ORMRun):\n    \"\"\"SQLAlchemy model of a flow run.\"\"\"\n    ...\n    new_column = Column(String, nullable=True) # <-- add this line\n

    Next, you will need to generate new migration files. You must generate a new migration file for each database type. Migrations will be generated for whatever database type PREFECT_API_DATABASE_CONNECTION_URL is set to. See here for how to set the database connection URL for each database type.

    To generate a new migration file, run the following command:

    prefect server database revision --autogenerate -m \"<migration name>\"\n

    Try to make your migration name brief but descriptive. For example:

    • add_flow_run_new_column
    • add_flow_run_new_column_idx
    • rename_flow_run_old_column_to_new_column

    The --autogenerate flag will automatically generate a migration file based on the changes to the models.

    Always inspect the output of --autogenerate

    --autogenerate will generate a migration file based on the changes to the models. However, it is not perfect. Be sure to check the file to make sure it only includes the changes you want to make. Additionally, you may need to remove extra statements that were included and not related to your change.

    The new migration can be found in the src/prefect/server/database/migrations/versions/ directory. Each database type has its own subdirectory. For example, the SQLite migrations are stored in src/prefect/server/database/migrations/versions/sqlite/.

    After you have inspected the migration file, you can apply the migration to your database by running the following command:

    prefect server database upgrade -y\n

    Once you have successfully created and applied migrations for all database types, make sure to update MIGRATION-NOTES.md to document your additions.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/style/","title":"Code style and practices","text":"

    Generally, we follow the Google Python Style Guide. This document covers sections where we differ or where additional clarification is necessary.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#imports","title":"Imports","text":"

    A brief collection of rules and guidelines for how imports should be handled in this repository.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#imports-in-__init__-files","title":"Imports in __init__ files","text":"

    Leave __init__ files empty unless exposing an interface. If you must expose objects to present a simpler API, please follow these rules.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#exposing-objects-from-submodules","title":"Exposing objects from submodules","text":"

    If importing objects from submodules, the __init__ file should use a relative import. This is required for type checkers to understand the exposed interface.

    # Correct\nfrom .flows import flow\n
    # Wrong\nfrom prefect.flows import flow\n
    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#exposing-submodules","title":"Exposing submodules","text":"

    Generally, submodules should not be imported in the __init__ file. Submodules should only be exposed when the module is designed to be imported and used as a namespaced object.

    For example, we do this for our schema and model modules because it is important to know if you are working with an API schema or database model, both of which may have similar names.

    import prefect.server.schemas as schemas\n\n# The full module is accessible now\nschemas.core.FlowRun\n

    If exposing a submodule, use a relative import as you would when exposing an object.

    # Correct\nfrom . import flows\n
    # Wrong\nimport prefect.flows\n
    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#importing-to-run-side-effects","title":"Importing to run side-effects","text":"

    Another use case for importing submodules is perform global side-effects that occur when they are imported.

    Often, global side-effects on import are a dangerous pattern. Avoid them if feasible.

    We have a couple acceptable use-cases for this currently:

    • To register dispatchable types, e.g. prefect.serializers.
    • To extend a CLI application e.g. prefect.cli.
    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#imports-in-modules","title":"Imports in modules","text":"","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#importing-other-modules","title":"Importing other modules","text":"

    The from syntax should be reserved for importing objects from modules. Modules should not be imported using the from syntax.

    # Correct\nimport prefect.server.schemas  # use with the full name\nimport prefect.server.schemas as schemas  # use the shorter name\n
    # Wrong\nfrom prefect.server import schemas\n

    Unless in an __init__.py file, relative imports should not be used.

    # Correct\nfrom prefect.utilities.foo import bar\n
    # Wrong\nfrom .utilities.foo import bar\n

    Imports dependent on file location should never be used without explicit indication it is relative. This avoids confusion about the source of a module.

    # Correct\nfrom . import test\n
    # Wrong\nimport test\n
    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#resolving-circular-dependencies","title":"Resolving circular dependencies","text":"

    Sometimes, we must defer an import and perform it within a function to avoid a circular dependency.

    ## This function in `settings.py` requires a method from the global `context` but the context\n## uses settings\ndef from_context():\n    from prefect.context import get_profile_context\n\n    ...\n

    Attempt to avoid circular dependencies. This often reveals overentanglement in the design.

    When performing deferred imports, they should all be placed at the top of the function.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#with-type-annotations","title":"With type annotations","text":"

    If you are just using the imported object for a type signature, you should use the TYPE_CHECKING flag.

    # Correct\nfrom typing import TYPE_CHECKING\n\nif TYPE_CHECKING:\n    from prefect.server.schemas.states import State\n\ndef foo(state: \"State\"):\n    pass\n

    Note that usage of the type within the module will need quotes e.g. \"State\" since it is not available at runtime.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#importing-optional-requirements","title":"Importing optional requirements","text":"

    We do not have a best practice for this yet. See the kubernetes, docker, and distributed implementations for now.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#delaying-expensive-imports","title":"Delaying expensive imports","text":"

    Sometimes, imports are slow. We'd like to keep the prefect module import times fast. In these cases, we can lazily import the slow module by deferring import to the relevant function body. For modules that are consumed by many functions, the pattern used for optional requirements may be used instead.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#command-line-interface-cli-output-messages","title":"Command line interface (CLI) output messages","text":"

    Upon executing a command that creates an object, the output message should offer: - A short description of what the command just did. - A bullet point list, rehashing user inputs, if possible. - Next steps, like the next command to run, if applicable. - Other relevant, pre-formatted commands that can be copied and pasted, if applicable. - A new line before the first line and after the last line.

    Output Example:

    $ prefect work-queue create testing\n\nCreated work queue with properties:\n    name - 'abcde'\n    uuid - 940f9828-c820-4148-9526-ea8107082bda\n    tags - None\n    deployment_ids - None\n\nStart an agent to pick up flows from the created work queue:\n    prefect agent start -q 'abcde'\n\nInspect the created work queue:\n    prefect work-queue inspect 'abcde'\n

    Additionally:

    • Wrap generated arguments in apostrophes (') to ensure validity by using suffixing formats with !r.
    • Indent example commands, instead of wrapping in backticks (`).
    • Use placeholders if the example cannot be pre-formatted completely.
    • Capitalize placeholder labels and wrap them in less than (<) and greater than (>) signs.
    • Utilize textwrap.dedent to remove extraneous spacing for strings that are written with triple quotes (\"\"\").

    Placeholder Example:

    Create a work queue with tags:\n    prefect work-queue create '<WORK QUEUE NAME>' -t '<OPTIONAL TAG 1>' -t '<OPTIONAL TAG 2>'\n

    Dedent Example:

    from textwrap import dedent\n...\noutput_msg = dedent(\n    f\"\"\"\n    Created work queue with properties:\n        name - {name!r}\n        uuid - {result}\n        tags - {tags or None}\n        deployment_ids - {deployment_ids or None}\n\n    Start an agent to pick up flows from the created work queue:\n        prefect agent start -q {name!r}\n\n    Inspect the created work queue:\n        prefect work-queue inspect {name!r}\n    \"\"\"\n)\n

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#api-versioning","title":"API Versioning","text":"

    The Prefect client can be run separately from the Prefect orchestration server and communicate entirely via an API. Among other things, the Prefect client includes anything that runs task or flow code, (e.g. agents, and the Python client) or any consumer of Prefect metadata, (e.g. the Prefect UI, and CLI). The Prefect server stores this metadata and serves it via the REST API.

    Sometimes, we make breaking changes to the API (for good reasons). In order to check that a Prefect client is compatible with the API it's making requests to, every API call the client makes includes a three-component API_VERSION header with major, minor, and patch versions.

    For example, a request with the X-PREFECT-API-VERSION=3.2.1 header has a major version of 3, minor version 2, and patch version 1.

    This version header can be changed by modifying the API_VERSION constant in prefect.server.api.server.

    When making a breaking change to the API, we should consider if the change might be backwards compatible for clients, meaning that the previous version of the client would still be able to make calls against the updated version of the server code. This might happen if the changes are purely additive: such as adding a non-critical API route. In these cases, we should make sure to bump the patch version.

    In almost all other cases, we should bump the minor version, which denotes a non-backwards-compatible API change. We have reserved the major version chanes to denote also-backwards compatible change that might be significant in some way, such as a major release milestone.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/versioning/","title":"Versioning","text":"","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#understanding-version-numbers","title":"Understanding version numbers","text":"

    Versions are composed of three parts: MAJOR.MINOR.PATCH. For example, the version 2.5.0 has a major version of 2, a minor version of 5, and patch version of 0.

    Occasionally, we will add a suffix to the version such as rc, a, or b. These indicate pre-release versions that users can opt-into installing to test functionality before it is ready for release.

    Each release will increase one of the version numbers. If we increase a number other than the patch version, the versions to the right of it will be reset to zero.

    ","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#prefects-versioning-scheme","title":"Prefect's versioning scheme","text":"

    Prefect will increase the major version when significant and widespread changes are made to the core product. It is very unlikely that the major version will change without extensive warning.

    Prefect will increase the minor version when:

    • Introducing a new concept that changes how Prefect can be used
    • Changing an existing concept in a way that fundamentally alters how it is used
    • Removing a deprecated feature

    Prefect will increase the patch version when:

    • Making enhancements to existing features
    • Fixing behavior in existing features
    • Adding new functionality to existing concepts
    • Updating dependencies
    ","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#breaking-changes-and-deprecation","title":"Breaking changes and deprecation","text":"

    A breaking change means that your code will need to change to use a new version of Prefect. We strive to avoid breaking changes in all releases.

    At times, Prefect will deprecate a feature. This means that a feature has been marked for removal in the future. When you use it, you may see warnings that it will be removed. A feature is deprecated when it will no longer be maintained. Frequently, a deprecated feature will have a new and improved alternative. Deprecated features will be retained for at least 3 minor version increases or 6 months, whichever is longer. We may retain deprecated features longer than this time period.

    Prefect will sometimes include changes to behavior to fix a bug. These changes are not categorized as breaking changes.

    ","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#client-compatibility-with-prefect","title":"Client compatibility with Prefect","text":"

    When running a Prefect server, you are in charge of ensuring the version is compatible with those of the clients that are using the server. Prefect aims to maintain backwards compatibility with old clients for each server release. In contrast, sometimes new clients cannot be used with an old server. The new client may expect the server to support functionality that it does not yet include. For this reason, we recommend that all clients are the same version as the server or older.

    For example, a client on 2.1.0 can be used with a server on 2.5.0. A client on 2.5.0 cannot be used with a server on 2.1.0.

    ","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#client-compatibility-with-cloud","title":"Client compatibility with Cloud","text":"

    Prefect Cloud targets compatibility with all versions of Prefect clients. If you encounter a compatibility issue, please file a bug report.

    ","tags":["versioning","semver"],"boost":2},{"location":"getting-started/installation/","title":"Installation","text":"

    Prefect requires Python 3.8 or newer.

    We recommend installing Prefect using a Python virtual environment manager such as pipenv, conda, or virtualenv/venv.

    Windows and Linux requirements

    See Windows installation notes and Linux installation notes for details on additional installation requirements and considerations.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#install-prefect","title":"Install Prefect","text":"

    The following sections describe how to install Prefect in your development or execution environment.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#installing-the-latest-version","title":"Installing the latest version","text":"

    Prefect is published as a Python package. To install the latest release or upgrade an existing Prefect install, run the following command in your terminal:

    pip install -U prefect\n

    To install a specific version, specify the version number like this:

    pip install -U \"prefect==2.10.4\"\n

    See available release versions in the Prefect Release Notes.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#installing-the-bleeding-edge","title":"Installing the bleeding edge","text":"

    If you'd like to test with the most up-to-date code, you can install directly off the main branch on GitHub:

    pip install -U git+https://github.com/PrefectHQ/prefect\n

    The main branch may not be stable

    Please be aware that this method installs unreleased code and may not be stable.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#installing-for-development","title":"Installing for development","text":"

    If you'd like to install a version of Prefect for development:

    1. Clone the Prefect repository.
    2. Install an editable version of the Python package with pip install -e.
    3. Install pre-commit hooks.
    $ git clone https://github.com/PrefectHQ/prefect.git\n$ cd prefect\n$ pip install -e \".[dev]\"\n$ pre-commit install\n

    See our Contributing guide for more details about standards and practices for contributing to Prefect.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#checking-your-installation","title":"Checking your installation","text":"

    To confirm that Prefect was installed correctly, run the command prefect version to print the version and environment details to your console.

    $ prefect version\n\nVersion:             2.10.21\nAPI version:         0.8.4\nPython version:      3.10.12\nGit commit:          da816542\nBuilt:               Thu, Jul 13, 2023 2:05 PM\nOS/Arch:             darwin/arm64\nProfile:              local\nServer type:         ephemeral\nServer:\n  Database:          sqlite\n  SQLite version:    3.42.0\n
    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#windows-installation-notes","title":"Windows installation notes","text":"

    You can install and run Prefect via Windows PowerShell, the Windows Command Prompt, or conda. After installation, you may need to manually add the Python local packages Scripts folder to your Path environment variable.

    The Scripts folder path looks something like this (the username and Python version may be different on your system):

    C:\\Users\\MyUserNameHere\\AppData\\Local\\Packages\\PythonSoftwareFoundation.Python.3.11_qbz5n2kfra8p0\\LocalCache\\local-packages\\Python311\\Scripts\n

    Watch the pip install output messages for the Scripts folder path on your system.

    If you're using Windows Subsystem for Linux (WSL), see Linux installation notes.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#linux-installation-notes","title":"Linux installation notes","text":"

    Linux is a popular operating system for running Prefect. You can use Prefect Cloud as your API server, or host your own Prefect server backed by PostgreSQL.

    For development, you can use SQLite 2.24 or newer as your database. Note that certain Linux versions of SQLite can be problematic. Compatible versions include Ubuntu 22.04 LTS and Ubuntu 20.04 LTS.

    Alternatively, you can install SQLite on Red Hat Enterprise Linux (RHEL) or use the conda virtual environment manager and configure a compatible SQLite version.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#using-a-self-signed-ssl-certificate","title":"Using a self-signed SSL certificate","text":"

    If you're using a self-signed SSL certificate, you need to configure your environment to trust the certificate. You can add the certificate to your system bundle and pointing your tools to use that bundle by configuring the SSL_CERT_FILE environment variable.

    If the certificate is not part of your system bundle, you can set the PREFECT_API_TLS_INSECURE_SKIP_VERIFY to True to disable certificate verification altogether.

    Note: Disabling certificate validation is insecure and only suggested as an option for testing!

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#proxies","title":"Proxies","text":"

    Prefect supports communicating via proxies through environment variables. Simply set HTTPS_PROXY and SSL_CERT_FILE in your environment, and the underlying network libraries will route Prefect\u2019s requests appropriately. Read more about using Prefect Cloud with proxies here.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#external-requirements","title":"External requirements","text":"","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#sqlite","title":"SQLite","text":"

    You can use Prefect Cloud as your API server, or host your own Prefect server backed by PostgreSQL.

    By default, a local Prefect server instance uses SQLite as the backing database. SQLite is not packaged with the Prefect installation. Most systems will already have SQLite installed, because it is typically bundled as a part of Python. Prefect requires SQLite version 3.24.0 or later.

    You can check your SQLite version by executing the following command in a terminal:

    $ sqlite3 --version\n

    Or use the Prefect CLI command prefect version, which prints version and environment details to your console, including the server database and version. For example:

    $ prefect version\nVersion:             2.10.21\nAPI version:         0.8.4\nPython version:      3.10.12\nGit commit:          a46cbebb\nBuilt:               Sat, Jul 15, 2023 7:59 AM\nOS/Arch:             darwin/arm64\nProfile:              default\nServer type:         cloud\n
    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#install-sqlite-on-rhel","title":"Install SQLite on RHEL","text":"

    The following steps are needed to install an appropriate version of SQLite on Red Hat Enterprise Linux (RHEL). Note that some RHEL instances have no C compiler, so you may need to check for and install gcc first:

    yum install gcc\n

    Download and extract the tarball for SQLite.

    wget https://www.sqlite.org/2022/sqlite-autoconf-3390200.tar.gz\ntar -xzf sqlite-autoconf-3390200.tar.gz\n

    Move to the extracted SQLite directory, then build and install SQLite.

    cd sqlite-autoconf-3390200/\n./configure\nmake\nmake install\n

    Add LD_LIBRARY_PATH to your profile.

    echo 'export LD_LIBRARY_PATH=\"/usr/local/lib\"' >> /etc/profile\n

    Restart your shell to register these changes.

    Now you can install Prefect using pip.

    pip3 install prefect\n
    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#using-prefect-in-an-environment-with-http-proxies","title":"Using Prefect in an environment with HTTP proxies","text":"

    If you are using Prefect Cloud or hosting your own Prefect server instance, the Prefect library will connect to the API via any proxies you have listed in the HTTP_PROXY, HTTPS_PROXY, or ALL_PROXY environment variables. You may also use the NO_PROXY environment variable to specify which hosts should not be sent through the proxy.

    For more information about these environment variables, see the cURL documentation.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#next-steps","title":"Next steps","text":"

    Now that you have Prefect installed and your environment configured, you may want to check out the Tutorial to get more familiar with Prefect.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/quickstart/","title":"Quickstart","text":"

    Prefect is an orchestration and observability platform that empowers developers to build and scale resilient code quickly, turning their python scripts into resilient, recurring workflows.

    In this quickstart, you'll see how you can schedule your code on remote infrastructure and observe the state of your workflows. With Prefect, you can go from a Python script to a production-ready workflow that runs remotely in minutes.

    Let's get started!

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#setup","title":"Setup","text":"

    Here's a basic script that fetches statistics about the main Prefect GitHub repository.

    import httpx\n\ndef get_repo_info():\n    url = \"https://api.github.com/repos/PrefectHQ/prefect\"\n    response = httpx.get(url)\n    repo = response.json()\n    print(\"PrefectHQ/prefect repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n\nif __name__ == \"__main__\":\n    get_repo_info()\n

    How can we make this script schedulable, observable, resilient, and capable of running anywhere?

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-1-install-prefect","title":"Step 1: Install Prefect","text":"
    pip install -U prefect\n

    See the install guide for more detailed installation instructions, if needed.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-2-connect-to-prefects-api","title":"Step 2: Connect to Prefect's API","text":"

    Much of Prefect's functionality is backed by an API. Sign up for a forever free Prefect Cloud account or accept your organization's invite to join their Prefect Cloud account.

    1. Create a new account or sign in at https://app.prefect.cloud/.
    2. Use the prefect cloud login CLI command to log in to Prefect Cloud from your environment.
    prefect cloud login\n

    Choose Log in with a web browser and click the Authorize button in the browser window that opens.

    Self-hosted Prefect server instance

    If you would like to host a Prefect server instance on your own infrastructure, see the tutorial and select the \"Self-hosted\" tab. Note that you will need to both host your own server and run your flows on your own infrastructure.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-3-turn-your-function-into-a-prefect-flow","title":"Step 3: Turn your function into a Prefect flow","text":"

    The fastest way to get started with Prefect is to add a @flow decorator to your Python function. Flows are the core observable, deployable units in Prefect and are the primary entrypoint to orchestrated work.

    my_gh_workflow.py
    import httpx\nfrom prefect import flow, task\n\n\n@task(retries=2)\ndef get_repo_info(repo_owner: str, repo_name: str):\n    \"\"\"Get info about a repo - will retry twice after failing\"\"\"\n    url = f\"https://api.github.com/repos/{repo_owner}/{repo_name}\"\n    api_response = httpx.get(url)\n    api_response.raise_for_status()\n    repo_info = api_response.json()\n    return repo_info\n\n\n@task\ndef get_contributors(repo_info: dict):\n    \"\"\"Get contributors for a repo\"\"\"\n    contributors_url = repo_info[\"contributors_url\"]\n    response = httpx.get(contributors_url)\n    response.raise_for_status()\n    contributors = response.json()\n    return contributors\n\n\n@flow(log_prints=True)\ndef repo_info(repo_owner: str = \"PrefectHQ\", repo_name: str = \"prefect\"):\n    \"\"\"\n    Given a GitHub repository, logs the number of stargazers\n    and contributors for that repo.\n    \"\"\"\n    repo_info = get_repo_info(repo_owner, repo_name)\n    print(f\"Stars \ud83c\udf20 : {repo_info['stargazers_count']}\")\n\n    contributors = get_contributors(repo_info)\n    print(f\"Number of contributors \ud83d\udc77: {len(contributors)}\")\n\n\nif __name__ == \"__main__\":\n    repo_info()\n

    Note that we added a log_prints=True argument to the @flow decorator so that print statements within the flow-decorated function will be logged. Also note that our flow calls two tasks, which are defined by the task decorator. Tasks are the smallest unit of observed and orchestrated work in Prefect.

    python my_gh_workflow.py\n

    Now when we run this script, Prefect will automatically track the state of the flow run and log the output where we can see it in the UI and CLI.

    14:28:31.099 | INFO    | prefect.engine - Created flow run 'energetic-panther' for flow 'repo-info'\n14:28:31.100 | INFO    | Flow run 'energetic-panther' - View at https://app.prefect.cloud/account/123/workspace/abc/flow-runs/flow-run/xyz\n14:28:32.178 | INFO    | Flow run 'energetic-panther' - Created task run 'get_repo_info-0' for task 'get_repo_info'\n14:28:32.179 | INFO    | Flow run 'energetic-panther' - Executing 'get_repo_info-0' immediately...\n14:28:32.584 | INFO    | Task run 'get_repo_info-0' - Finished in state Completed()\n14:28:32.599 | INFO    | Flow run 'energetic-panther' - Stars \ud83c\udf20 : 13609\n14:28:32.682 | INFO    | Flow run 'energetic-panther' - Created task run 'get_contributors-0' for task 'get_contributors'\n14:28:32.682 | INFO    | Flow run 'energetic-panther' - Executing 'get_contributors-0' immediately...\n14:28:33.118 | INFO    | Task run 'get_contributors-0' - Finished in state Completed()\n14:28:33.134 | INFO    | Flow run 'energetic-panther' - Number of contributors \ud83d\udc77: 30\n14:28:33.255 | INFO    | Flow run 'energetic-panther' - Finished in state Completed('All states completed.')\n

    You should see similar output in your terminal, with your own randomly generated flow run name and your own Prefect Cloud account URL.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-4-choose-a-remote-infrastructure-location","title":"Step 4: Choose a remote infrastructure location","text":"

    Let's get this workflow running on infrastructure other than your local machine! We can tell Prefect where we want to run our workflow by creating a work pool.

    We can have Prefect Cloud run our flow code for us with a Prefect Managed work pool.

    Let's create a Prefect Managed work pool so that Prefect can run our flows for us. We can create a work pool in the UI or from the CLI. Let's use the CLI:

    prefect work-pool create my-managed-pool --type prefect:managed\n

    You should see a message in the CLI that your work pool was created. Feel free to check out your new work pool on the Work Pools page in the UI.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-4-make-your-code-schedulable","title":"Step 4: Make your code schedulable","text":"

    We have a flow function and we have a work pool where we can run our flow remotely. Let's package both of these things, along with the location for where to find our flow code, into a deployment so that we can schedule our workflow to run remotely.

    Deployments elevate flows to remotely configurable entities that have their own API.

    Let's make a script to build a deployment with the name my-first-deployment and set it to run on a schedule.

    create_deployment.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=\"https://github.com/discdiver/demos.git\",\n        entrypoint=\"my_gh_workflow.py:repo_info\",\n    ).deploy(\n        name=\"my-first-deployment\",\n        work_pool_name=\"my-managed-pool\",\n        cron=\"0 1 * * *\",\n    )\n

    Run the script to create the deployment on the Prefect Cloud server. Note that the cron argument will schedule the deployment to run at 1am every day.

    python create_deployment.py\n

    You should see a message that your deployment was created, similar to the one below.

    Successfully created/updated all deployments!\n\n                     Deployments                     \n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                          \u2503 Status  \u2503 Details \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 repo-info/my-first-deployment \u2502 applied \u2502         \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n\nTo schedule a run for this deployment, use the following command:\n\n        $ prefect deployment run 'repo-info/my-first-deployment'\n\n\nYou can also run your flow via the Prefect UI: <https://app.prefect.cloud/account/abc/workspace/123/deployments/deployment/xyz>\n

    Head to the Deployments page of the UI to check it out.

    Code storage options

    You can store your flow code in nearly any location. You just need to tell Prefect where to find it. In this example, we use a GitHub repository, but you could bake your code into a Docker image or store it in cloud provider storage. Read more here.

    Push your code to GitHub

    In the example above, we use an existing GitHub repository. If you make changes to the flow code, you will need to push those changes to your own GitHub account and update the source argument to point to your repository.

    You can trigger a manual run of this deployment by either clicking the Run button in the top right of the deployment page in the UI, or by running the following CLI command in your terminal:

    prefect deployment run 'repo-info/my-first-deployment'\n

    The deployment is configured to run on a Prefect Managed work pool, so Prefect will automatically spin up the infrastructure to run this flow. It may take a minute to set up the Docker image in which the flow will run.

    After a minute or so, you should see the flow run graph and logs on the Flow Run page in the UI.

    Remove the schedule

    Click the Remove button in the top right of the Deployment page so that the workflow is no longer scheduled to run once per day.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#next-steps","title":"Next steps","text":"

    You've seen how to move from a Python script to a scheduled, observable, remotely orchestrated workflow with Prefect.

    To learn how to run flows on your own infrastructure, see how to customize the Docker image where your flow runs, and see how to gain lots of orchestration and observation benefits check out the tutorial.

    Need help?

    Get your questions answered by a Prefect Product Advocate! Book a Meeting

    Happy building!

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"guides/","title":"How-to Guides","text":"

    This section of the documentation contains how-to guides for common workflows and use cases.

    ","tags":["guides","how to"],"boost":2},{"location":"guides/#development","title":"Development","text":"Title Description Hosting Host your own Prefect server instance. Profiles & Settings Configure Prefect and save your settings. Testing Easily test your workflows. Global Concurrency Limits Limit flow runs. Runtime Context Enable a flow to access metadata about itself and its context when it runs. Variables Store and retrieve configuration data. Prefect Client Use PrefectClient to interact with the API server. Human-in-the-Loop Workflows: Create human-in-the-loop workflows by pausing flow runs for input. Webhooks Receive, observe, and react to events from other systems. Terraform Provider Use the Terraform Provider for Prefect Cloud for infrastructure as code. CI/CD Use CI/CD with Prefect. Prefect Recipes Common, extensible examples for setting up Prefect.","tags":["guides","how to"],"boost":2},{"location":"guides/#execution","title":"Execution","text":"Title Description Docker Deploy flows with Docker containers. State Change Hooks Execute code in response to state changes. Dask and Ray Scale your flows with parallel computing frameworks. Read and Write Data Read and write data to and from cloud provider storage. Big Data Handle large data with Prefect. Logging Configure Prefect's logger and aggregate logs from other tools. Troubleshooting Identify and resolve common issues with Prefect. Managed Execution Let prefect run your code.","tags":["guides","how to"],"boost":2},{"location":"guides/#work-pools","title":"Work Pools","text":"Title Description Deploying Flows to Work Pools and Workers Learn how to run you code with dynamic infrastructure. Upgrade from Agents to Workers Why and how to upgrade from agents to workers. Storage Store your code for deployed flows. Kubernetes Deploy flows on Kubernetes. Serverless Push Work Pools Run flows on serverless infrastructure without a worker. Serverless Work Pools with Workers Run flows on serverless infrastructure with a worker. Automatic infrastructure provisioning Automatically provision cloud infrastructure. Daemonize Processes Set up a systemd service to run a Prefect worker or .serve process. Custom Workers Develop your own worker type.

    Need help?

    Get your questions answered by a Prefect Product Advocate! Book a Meeting

    ","tags":["guides","how to"],"boost":2},{"location":"guides/big-data/","title":"Big data with Prefect","text":"

    In this guide you'll learn tips for working with large amounts of data in Prefect.

    Big data doesn't have a widely accepted, precise definition. In this guide, we'll discuss methods to reduce the processing time or memory utilization of Prefect workflows, without editing your Python code.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#optimizing-your-python-code-with-prefect-for-big-data","title":"Optimizing your Python code with Prefect for big data","text":"

    Depending upon your needs, you may want to optimize your Python code for speed, memory, compute, or disk space.

    Prefect provides several options that we'll explore in this guide:

    1. Remove task introspection with quote to save time running your code.
    2. Write task results to cloud storage such as S3 using a block to save memory.
    3. Save data to disk within a flow rather than using results.
    4. Cache task results to save time and compute.
    5. Compress results written to disk to save space.
    6. Use a task runner for parallelizable operations to save time.
    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#remove-task-introspection","title":"Remove task introspection","text":"

    When a task is called from a flow, each argument is introspected by Prefect, by default. To speed up your flow runs, you can disable this behavior for a task by wrapping the argument using quote.

    To demonstrate, let's use a basic example that extracts and transforms some New York taxi data.

    et_quote.py
    from prefect import task, flow\nfrom prefect.utilities.annotations import quote\nimport pandas as pd\n\n\n@task\ndef extract(url: str):\n    \"\"\"Extract data\"\"\"\n    df_raw = pd.read_parquet(url)\n    print(df_raw.info())\n    return df_raw\n\n\n@task\ndef transform(df: pd.DataFrame):\n    \"\"\"Basic transformation\"\"\"\n    df[\"tip_fraction\"] = df[\"tip_amount\"] / df[\"total_amount\"]\n    print(df.info())\n    return df\n\n\n@flow(log_prints=True)\ndef et(url: str):\n    \"\"\"ET pipeline\"\"\"\n    df_raw = extract(url)\n    df = transform(quote(df_raw))\n\n\nif __name__ == \"__main__\":\n    url = \"https://d37ci6vzurychx.cloudfront.net/trip-data/yellow_tripdata_2023-09.parquet\"\n    et(url)\n

    Introspection can take significant time when the object being passed is a large collection, such as dictionary or DataFrame, where each element needs to be visited. Note that using quote reduces execution time at the expense of disabling task dependency tracking for the wrapped object.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#write-task-results-to-cloud-storage","title":"Write task results to cloud storage","text":"

    By default, the results of task runs are stored in memory in your execution environment. This behavior makes flow runs fast for small data, but can be problematic for large data. You can save memory by writing results to disk. In production, you'll generally want to write results to a cloud provider storage such as AWS S3. Prefect lets you to use a storage block from a Prefect cloud integration library such as prefect-aws to save your configuration information. Learn more about blocks here.

    Install the relevant library, register the block with the server, and create your storage block. Then you can reference the block in your flow like this:

    ...\nfrom prefect_aws.s3 import S3Bucket\n\nmy_s3_block = S3Bucket.load(\"MY_BLOCK_NAME\")\n\n...\n@task(result_storage=my_s3_block)\n

    Now the result of the task will be written to S3, rather than stored in memory.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#save-data-to-disk-within-a-flow","title":"Save data to disk within a flow","text":"

    To save memory and time with big data, you don't need to pass results between tasks at all. Instead, you can write and read data to disk directly in your flow code. Prefect has integration libraries for each of the major cloud providers. Each library contains blocks with methods that make it convenient to read and write data to and from cloud object storage. The moving data guide has step-by-step examples for each cloud provider.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#cache-task-results","title":"Cache task results","text":"

    Caching allows you to avoid re-running tasks when doing so is unnecessary. Caching can save you time and compute. Note that caching requires task result persistence. Caching is discussed in detail in the tasks concept page.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#compress-results-written-to-disk","title":"Compress results written to disk","text":"

    If you're using Prefect's task result persistence, you can save disk space by compressing the results. You just need to specify the result type with compressed/ prefixed like this:

    @task(result_serializer=\"compressed/json\")\n

    Read about compressing results with Prefect for more details. The tradeoff of using compression is that it takes time to compress and decompress the data.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#use-a-task-runner-for-parallelizable-operations","title":"Use a task runner for parallelizable operations","text":"

    Prefect's task runners allow you to use the Dask and Ray Python libraries to run tasks in parallel and distributed across multiple machines. This can save you time and compute when operating on large data structures. See the guide to working with Dask and Ray Task Runners for details.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/ci-cd/","title":"CI/CD with Prefect","text":"

    Many organizations deploy Prefect workflows via their CI/CD process. Each organization has their own unique CI/CD setup, but a common pattern is to use CI/CD to manage Prefect deployments. Combining Prefect's deployment features with CI/CD tools enables efficient management of flow code updates, scheduling changes, and container builds. This guide uses GitHub Actions to implement a CI/CD process, but these concepts are generally applicable across many CI/CD tools.

    Note that Prefect's primary ways for creating deployments, a .deploy flow method or a prefect.yaml configuration file, are both designed with building and pushing images to a Docker registry in mind.

    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#getting-started-with-github-actions-and-prefect","title":"Getting started with GitHub Actions and Prefect","text":"

    In this example, you'll write a GitHub Actions workflow that will run each time you push to your repository's main branch. This workflow will build and push a Docker image containing your flow code to Docker Hub, then deploy the flow to Prefect Cloud.

    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#repository-secrets","title":"Repository secrets","text":"

    Your CI/CD process must be able to authenticate with Prefect in order to deploy flows.

    Deploying flows securely and non-interactively in your CI/CD process can be accomplished by saving your PREFECT_API_URL and PREFECT_API_KEY as secrets in your repository's settings so they can be accessed in your CI/CD runner's environment without exposing them in any scripts or configuration files.

    In this scenario, deploying flows involves building and pushing Docker images, so add DOCKER_USERNAME and DOCKER_PASSWORD as secrets to your repository as well.

    You can create secrets for GitHub Actions in your repository under Settings -> Secrets and variables -> Actions -> New repository secret:

    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#writing-a-github-workflow","title":"Writing a GitHub workflow","text":"

    To deploy your flow via GitHub Actions, you'll need a workflow YAML file. GitHub will look for workflow YAML files in the .github/workflows/ directory in the root of your repository. In their simplest form, GitHub workflow files are made up of triggers and jobs.

    The on: trigger is set to run the workflow each time a push occurs on the main branch of the repository.

    The deploy job is comprised of four steps:

    • Checkout clones your repository into the GitHub Actions runner so you can reference files or run scripts from your repository in later steps.
    • Log in to Docker Hub authenticates to DockerHub so your image can be pushed to the Docker registry in your DockerHub account. docker/login-action is an existing GitHub action maintained by Docker. with: passes values into the Action, similar to passing parameters to a function.
    • Setup Python installs your selected version of Python.
    • Prefect Deploy installs the dependencies used in your flow, then deploys your flow. env: makes the PREFECT_API_KEY and PREFECT_API_URL secrets from your repository available as environment variables during this step's execution.

    For reference, the examples below can be found on their respective branches of this repository.

    .deployprefect.yaml
    .\n\u251c\u2500\u2500 .github/\n\u2502   \u2514\u2500\u2500 workflows/\n\u2502       \u2514\u2500\u2500 deploy-prefect-flow.yaml\n\u251c\u2500\u2500 flow.py\n\u2514\u2500\u2500 requirements.txt\n

    flow.py

    from prefect import flow\n\n@flow(log_prints=True)\ndef hello():\n  print(\"Hello!\")\n\nif __name__ == \"__main__\":\n    hello.deploy(\n        name=\"my-deployment\",\n        work_pool_name=\"my-work-pool\",\n        image=\"my_registry/my_image:my_image_tag\",\n    )\n

    .github/workflows/deploy-prefect-flow.yaml

    name: Deploy Prefect flow\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    name: Deploy\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout\n        uses: actions/checkout@v4\n\n      - name: Log in to Docker Hub\n        uses: docker/login-action@v3\n        with:\n          username: ${{ secrets.DOCKER_USERNAME }}\n          password: ${{ secrets.DOCKER_PASSWORD }}\n\n      - name: Setup Python\n        uses: actions/setup-python@v5\n        with:\n          python-version: '3.11'\n\n      - name: Prefect Deploy\n        env:\n          PREFECT_API_KEY: ${{ secrets.PREFECT_API_KEY }}\n          PREFECT_API_URL: ${{ secrets.PREFECT_API_URL }}\n        run: |\n          pip install -r requirements.txt\n          python flow.py\n
    .\n\u251c\u2500\u2500 .github/\n\u2502   \u2514\u2500\u2500 workflows/\n\u2502       \u2514\u2500\u2500 deploy-prefect-flow.yaml\n\u251c\u2500\u2500 flow.py\n\u251c\u2500\u2500 prefect.yaml\n\u2514\u2500\u2500 requirements.txt\n

    flow.py

    from prefect import flow\n\n@flow(log_prints=True)\ndef hello():\n  print(\"Hello!\")\n

    prefect.yaml

    name: cicd-example\nprefect-version: 2.14.11\n\nbuild:\n  - prefect_docker.deployments.steps.build_docker_image:\n      id: build_image\n      requires: prefect-docker>=0.3.1\n      image_name: my_registry/my_image\n      tag: my_image_tag\n      dockerfile: auto\n\npush:\n  - prefect_docker.deployments.steps.push_docker_image:\n      requires: prefect-docker>=0.3.1\n      image_name: \"{{ build_image.image_name }}\"\n      tag: \"{{ build_image.tag }}\"\n\npull: null\n\ndeployments:\n  - name: my-deployment\n    entrypoint: flow.py:hello\n    work_pool:\n      name: my-work-pool\n      work_queue_name: default\n      job_variables:\n        image: \"{{ build-image.image }}\"\n

    .github/workflows/deploy-prefect-flow.yaml

    name: Deploy Prefect flow\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    name: Deploy\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout\n        uses: actions/checkout@v4\n\n      - name: Log in to Docker Hub\n        uses: docker/login-action@v3\n        with:\n          username: ${{ secrets.DOCKER_USERNAME }}\n          password: ${{ secrets.DOCKER_PASSWORD }}\n\n      - name: Setup Python\n        uses: actions/setup-python@v5\n        with:\n          python-version: '3.11'\n\n      - name: Prefect Deploy\n        env:\n          PREFECT_API_KEY: ${{ secrets.PREFECT_API_KEY }}\n          PREFECT_API_URL: ${{ secrets.PREFECT_API_URL }}\n        run: |\n          pip install -r requirements.txt\n          prefect deploy -n my-deployment\n
    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#running-a-github-workflow","title":"Running a GitHub workflow","text":"

    After pushing commits to your repository, GitHub will automatically trigger a run of your workflow. The status of running and completed workflows can be monitored from the Actions tab of your repository.

    You can view the logs from each workflow step as they run. The Prefect Deploy step will include output about your image build and push, and the creation/update of your deployment.

    Successfully built image '***/cicd-example:latest'\n\nSuccessfully pushed image '***/cicd-example:latest'\n\nSuccessfully created/updated all deployments!\n\n                Deployments\n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                \u2503 Status  \u2503 Details \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 hello/my-deployment \u2502 applied \u2502         \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n
    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#prefect-github-actions","title":"Prefect GitHub Actions","text":"

    Prefect provides its own GitHub Actions for authentication and deployment creation. These actions can simplify deploying with CI/CD when using prefect.yaml, especially in cases where a repository contains flows that are used in multiple deployments across multiple Prefect Cloud workspaces.

    Here's an example of integrating these actions into the workflow we created above:

    name: Deploy Prefect flow\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    name: Deploy\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout\n        uses: actions/checkout@v4\n\n      - name: Log in to Docker Hub\n        uses: docker/login-action@v3\n        with:\n          username: ${{ secrets.DOCKER_USERNAME }}\n          password: ${{ secrets.DOCKER_PASSWORD }}\n\n      - name: Setup Python\n        uses: actions/setup-python@v5\n        with:\n          python-version: \"3.11\"\n\n      - name: Prefect Auth\n        uses: PrefectHQ/actions-prefect-auth@v1\n        with:\n          prefect-api-key: ${{ secrets.PREFECT_API_KEY }}\n          prefect-workspace: ${{ secrets.PREFECT_WORKSPACE }}\n\n      - name: Run Prefect Deploy\n        uses: PrefectHQ/actions-prefect-deploy@v3\n        with:\n          deployment-names: my-deployment\n          requirements-file-paths: requirements.txt\n
    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#authenticating-to-other-docker-image-registries","title":"Authenticating to other Docker image registries","text":"

    The docker/login-action GitHub Action supports pushing images to a wide variety of image registries.

    For example, if you are storing Docker images in AWS Elastic Container Registry, you can add your ECR registry URL to the registry key in the with: part of the action and use an AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY as your username and password.

    - name: Login to ECR\n  uses: docker/login-action@v3\n  with:\n    registry: <aws-account-number>.dkr.ecr.<region>.amazonaws.com\n    username: ${{ secrets.AWS_ACCESS_KEY_ID }}\n    password: ${{ secrets.AWS_SECRET_ACCESS_KEY }}\n
    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#other-resources","title":"Other resources","text":"

    Check out the Prefect Cloud Terraform provider if you're using Terraform to manage your infrastructure.

    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/creating-human-in-the-loop-workflows/","title":"Creating Human-in-the-Loop Workflows","text":"

    Experimental

    The wait_for_input parameter used in the pause_flow_run or suspend_flow_run functions is an experimental feature. The interface or behavior of this feature may change without warning in future releases.

    If you encounter any issues, please let us know in Slack or with a Github issue.

    When a flow run is paused or suspended, you can receive input from the user. This is useful when you need to ask the user for additional information or feedback before resuming the flow run.

    ","tags":["flow run","pause","suspend","input"],"boost":2},{"location":"guides/creating-human-in-the-loop-workflows/#waiting-for-input","title":"Waiting for input","text":"

    To receive input you must use the wait_for_input parameter in the pause_flow_run or suspend_flow_run functions. This parameter accepts a subclass of prefect.input.RunInput. RunInput is a subclass of pydantic.BaseModel and can be used to define the input that you want to receive:

    from prefect.input import RunInput\n\nclass UserNameInput(RunInput):\n    name: str\n

    In this case we are defining a UserNameInput class that will receive a name string from the user. You can then use this class in the wait_for_input parameter:

    @flow\nasync def greet_user():\n    logger = get_run_logger()\n\n    user_input = await pause_flow_run(\n        wait_for_input=UserNameInput\n    )\n\n    logger.info(f\"Hello, {user_input.name}!\")\n

    When the flow run is paused, the user will be prompted to enter a name. If the user does not enter a name, the flow run will not resume. If the user enters a name, the flow run will resume and the user_input variable will contain the name that the user entered.

    ","tags":["flow run","pause","suspend","input"],"boost":2},{"location":"guides/creating-human-in-the-loop-workflows/#providing-initial-data","title":"Providing initial data","text":"

    You can set default values for fields in your model by using the with_initial_data method. This is useful when you want to provide default values for the fields in your own RunInput subclasses.

    Expanding on the example above, you could default the name field to something anonymous.

    @flow\nasync def greet_user():\n    logger = get_run_logger()\n\n    user_input = await pause_flow_run(\n        wait_for_input=UserNameInput.with_initial_data(name=\"anonymous\")\n    )\n\n    if user_input.name == \"anonymous\":\n        logger.info(\"Hello, stranger!\")\n    else:\n        logger.info(f\"Hello, {user_input.name}!\")\n
    ","tags":["flow run","pause","suspend","input"],"boost":2},{"location":"guides/creating-human-in-the-loop-workflows/#handling-custom-validation","title":"Handling custom validation","text":"

    Prefect uses the fields and type hints on your RunInput subclass to validate the general structure of input your flow run receives, but you might require more complex validation. If you do, you can use Pydantic validators.

    Custom validation runs after the flow run resumes

    Prefect transforms the type annotations in your RunInput class to a JSON schema and use that schema in the UI to do client-side validation. However, custom validation requires running logic defined in your RunInput class. This happens after the flow resumes, so you'll probably want to handle it explicitly in your flow. Continue reading for an example best practice.

    The following is an example RunInput class that uses a custom field validator:

    import pydantic\nfrom prefect.input import RunInput\n\n\nclass ShirtOrder(RunInput):\n    size: Literal[\"small\", \"medium\", \"large\", \"xlarge\"]\n    color: Literal[\"red\", \"green\", \"black\"]\n\n    @pydantic.validator(\"color\")\n    def validate_age(cls, value, values, **kwargs):\n        if value == \"green\" and values[\"size\"] == \"small\":\n            raise ValueError(\"Green is only in-stock for medium, large, and XL sizes.\")\n\n        return value\n

    In the example, we use Pydantic's validator decorator to define a custom validation method for the color field. We can use it in a flow like this:

    import pydantic\nfrom prefect import flow\nfrom prefect.input import RunInput\n\n\nclass ShirtOrder(RunInput):\n    size: Literal[\"small\", \"medium\", \"large\", \"xlarge\"]\n    color: Literal[\"red\", \"green\", \"black\"]\n\n    @pydantic.validator(\"color\")\n    def validate_age(cls, value, values, **kwargs):\n        if value == \"green\" and values[\"size\"] == \"small\":\n            raise ValueError(\"Green is only in-stock for medium, large, and XL sizes.\")\n\n        return value\n\n\n@flow\ndef get_shirt_order():\n    shirt_order = pause_flow_run(wait_for_input=ShirtOrder)\n

    If a user chooses any size and color combination other than small and green, the flow run will resume successfully. However, if the user chooses size small and color green, the flow run will resume, and pause_flow_run will raise a ValidationError exception. This will cause the flow run to fail and log the error.

    However, what if you don't want the flow run to fail? One way to handle this case is to use a while loop and pause again if the ValidationError exception is raised:

    import pydantic\nfrom prefect import flow, get_run_logger\nfrom prefect.input import RunInput\n\n\nclass ShirtOrder(RunInput):\n    size: Literal[\"small\", \"medium\", \"large\", \"xlarge\"]\n    color: Literal[\"red\", \"green\", \"black\"]\n\n    @pydantic.validator(\"color\")\n    def validate_age(cls, value, values, **kwargs):\n        if value == \"green\" and values[\"size\"] == \"small\":\n            raise ValueError(\"Green is only in-stock for medium, large, and XL sizes.\")\n\n        return value\n\n\n@flow\ndef get_shirt_order():\n    logger = get_run_logger()\n    shirt_order = None\n\n    while shirt_order is None:\n        try:\n            shirt_order = pause_flow_run(wait_for_input=ShirtOrder)\n        except pydantic.ValidationError as exc:\n            logger.error(f\"Invalid size and color combination: {exc}\")\n\n    logger.info(f\"Shirt order: {shirt_order.size}, {shirt_order.color}\")\n

    This code will cause the flow run to continually pause until the user enters a valid age.

    As an additional step, you may want to use an automation or notification to alert the user to the error.

    ","tags":["flow run","pause","suspend","input"],"boost":2},{"location":"guides/dask-ray-task-runners/","title":"Dask and Ray Task Runners","text":"

    Task runners provide an execution environment for tasks. In a flow decorator, you can specify a task runner to run the tasks called in that flow.

    The default task runner is the ConcurrentTaskRunner.

    Use .submit to run your tasks asynchronously

    To run tasks asynchronously use the .submit method when you call them. If you call a task as you would normally in Python code it will run synchronously, even if you are calling the task within a flow that uses the ConcurrentTaskRunner, DaskTaskRunner, or RayTaskRunner.

    Many real-world data workflows benefit from true parallel, distributed task execution. For these use cases, the following Prefect-developed task runners for parallel task execution may be installed as Prefect Integrations.

    • DaskTaskRunner runs tasks requiring parallel execution using dask.distributed.
    • RayTaskRunner runs tasks requiring parallel execution using Ray.

    These task runners can spin up a local Dask cluster or Ray instance on the fly, or let you connect with a Dask or Ray environment you've set up separately. Then you can take advantage of massively parallel computing environments.

    Use Dask or Ray in your flows to choose the execution environment that fits your particular needs.

    To show you how they work, let's start small.

    Remote storage

    We recommend configuring remote file storage for task execution with DaskTaskRunner or RayTaskRunner. This ensures tasks executing in Dask or Ray have access to task result storage, particularly when accessing a Dask or Ray instance outside of your execution environment.

    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/dask-ray-task-runners/#configuring-a-task-runner","title":"Configuring a task runner","text":"

    You may have seen this briefly in a previous tutorial, but let's look a bit more closely at how you can configure a specific task runner for a flow.

    Let's start with the SequentialTaskRunner. This task runner runs all tasks synchronously and may be useful when used as a debugging tool in conjunction with async code.

    Let's start with this simple flow. We import the SequentialTaskRunner, specify a task_runner on the flow, and call the tasks with .submit().

    from prefect import flow, task\nfrom prefect.task_runners import SequentialTaskRunner\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n@flow(task_runner=SequentialTaskRunner())\ndef greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n\ngreetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n

    Save this as sequential_flow.py and run it in a terminal. You'll see output similar to the following:

    $ python sequential_flow.py\n16:51:17.967 | INFO    | prefect.engine - Created flow run 'humongous-mink' for flow 'greetings'\n16:51:17.967 | INFO    | Flow run 'humongous-mink' - Starting 'SequentialTaskRunner'; submitted tasks will be run sequentially...\n16:51:18.038 | INFO    | Flow run 'humongous-mink' - Created task run 'say_hello-811087cd-0' for task 'say_hello'\n16:51:18.038 | INFO    | Flow run 'humongous-mink' - Executing 'say_hello-811087cd-0' immediately...\nhello arthur\n16:51:18.060 | INFO    | Task run 'say_hello-811087cd-0' - Finished in state Completed()\n16:51:18.107 | INFO    | Flow run 'humongous-mink' - Created task run 'say_goodbye-261e56a8-0' for task 'say_goodbye'\n16:51:18.107 | INFO    | Flow run 'humongous-mink' - Executing 'say_goodbye-261e56a8-0' immediately...\ngoodbye arthur\n16:51:18.123 | INFO    | Task run 'say_goodbye-261e56a8-0' - Finished in state Completed()\n16:51:18.134 | INFO    | Flow run 'humongous-mink' - Created task run 'say_hello-811087cd-1' for task 'say_hello'\n16:51:18.134 | INFO    | Flow run 'humongous-mink' - Executing 'say_hello-811087cd-1' immediately...\nhello trillian\n16:51:18.150 | INFO    | Task run 'say_hello-811087cd-1' - Finished in state Completed()\n16:51:18.159 | INFO    | Flow run 'humongous-mink' - Created task run 'say_goodbye-261e56a8-1' for task 'say_goodbye'\n16:51:18.159 | INFO    | Flow run 'humongous-mink' - Executing 'say_goodbye-261e56a8-1' immediately...\ngoodbye trillian\n16:51:18.181 | INFO    | Task run 'say_goodbye-261e56a8-1' - Finished in state Completed()\n16:51:18.190 | INFO    | Flow run 'humongous-mink' - Created task run 'say_hello-811087cd-2' for task 'say_hello'\n16:51:18.190 | INFO    | Flow run 'humongous-mink' - Executing 'say_hello-811087cd-2' immediately...\nhello ford\n16:51:18.210 | INFO    | Task run 'say_hello-811087cd-2' - Finished in state Completed()\n16:51:18.219 | INFO    | Flow run 'humongous-mink' - Created task run 'say_goodbye-261e56a8-2' for task 'say_goodbye'\n16:51:18.219 | INFO    | Flow run 'humongous-mink' - Executing 'say_goodbye-261e56a8-2' immediately...\ngoodbye ford\n16:51:18.237 | INFO    | Task run 'say_goodbye-261e56a8-2' - Finished in state Completed()\n16:51:18.246 | INFO    | Flow run 'humongous-mink' - Created task run 'say_hello-811087cd-3' for task 'say_hello'\n16:51:18.246 | INFO    | Flow run 'humongous-mink' - Executing 'say_hello-811087cd-3' immediately...\nhello marvin\n16:51:18.264 | INFO    | Task run 'say_hello-811087cd-3' - Finished in state Completed()\n16:51:18.273 | INFO    | Flow run 'humongous-mink' - Created task run 'say_goodbye-261e56a8-3' for task 'say_goodbye'\n16:51:18.273 | INFO    | Flow run 'humongous-mink' - Executing 'say_goodbye-261e56a8-3' immediately...\ngoodbye marvin\n16:51:18.290 | INFO    | Task run 'say_goodbye-261e56a8-3' - Finished in state Completed()\n16:51:18.321 | INFO    | Flow run 'humongous-mink' - Finished in state Completed('All states completed.')\n

    If we take out the log messages and just look at the printed output of the tasks, you see they're executed in sequential order:

    $ python sequential_flow.py\nhello arthur\ngoodbye arthur\nhello trillian\ngoodbye trillian\nhello ford\ngoodbye ford\nhello marvin\ngoodbye marvin\n
    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/dask-ray-task-runners/#running-parallel-tasks-with-dask","title":"Running parallel tasks with Dask","text":"

    You could argue that this simple flow gains nothing from parallel execution, but let's roll with it so you can see just how simple it is to take advantage of the DaskTaskRunner.

    To configure your flow to use the DaskTaskRunner:

    1. Make sure the prefect-dask collection is installed by running pip install prefect-dask.
    2. In your flow code, import DaskTaskRunner from prefect_dask.task_runners.
    3. Assign it as the task runner when the flow is defined using the task_runner=DaskTaskRunner argument.
    4. Use the .submit method when calling functions.

    This is the same flow as above, with a few minor changes to use DaskTaskRunner where we previously configured SequentialTaskRunner. Install prefect-dask, made these changes, then save the updated code as dask_flow.py.

    from prefect import flow, task\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n@flow(task_runner=DaskTaskRunner())\ndef greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n\nif __name__ == \"__main__\":\n    greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n

    Note that, because you're using DaskTaskRunner in a script, you must use if __name__ == \"__main__\": or you'll see warnings and errors.

    Now run dask_flow.py. If you get a warning about accepting incoming network connections, that's okay - everything is local in this example.

    $ python dask_flow.py\n19:29:03.798 | INFO    | prefect.engine - Created flow run 'fine-bison' for flow 'greetings'\n\n19:29:03.798 | INFO    | Flow run 'fine-bison' - Using task runner 'DaskTaskRunner'\n\n19:29:04.080 | INFO    | prefect.task_runner.dask - Creating a new Dask cluster with `distributed.deploy.local.LocalCluster`\n16:54:18.465 | INFO    | prefect.engine - Created flow run 'radical-finch' for flow 'greetings'\n16:54:18.465 | INFO    | Flow run 'radical-finch' - Starting 'DaskTaskRunner'; submitted tasks will be run concurrently...\n16:54:18.465 | INFO    | prefect.task_runner.dask - Creating a new Dask cluster with `distributed.deploy.local.LocalCluster`\n16:54:19.811 | INFO    | prefect.task_runner.dask - The Dask dashboard is available at <http://127.0.0.1:8787/status>\n16:54:19.881 | INFO    | Flow run 'radical-finch' - Created task run 'say_hello-811087cd-0' for task 'say_hello'\n16:54:20.364 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_hello-811087cd-0' for execution.\n16:54:20.379 | INFO    | Flow run 'radical-finch' - Created task run 'say_goodbye-261e56a8-0' for task 'say_goodbye'\n16:54:20.386 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_goodbye-261e56a8-0' for execution.\n16:54:20.397 | INFO    | Flow run 'radical-finch' - Created task run 'say_hello-811087cd-1' for task 'say_hello'\n16:54:20.401 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_hello-811087cd-1' for execution.\n16:54:20.417 | INFO    | Flow run 'radical-finch' - Created task run 'say_goodbye-261e56a8-1' for task 'say_goodbye'\n16:54:20.423 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_goodbye-261e56a8-1' for execution.\n16:54:20.443 | INFO    | Flow run 'radical-finch' - Created task run 'say_hello-811087cd-2' for task 'say_hello'\n16:54:20.449 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_hello-811087cd-2' for execution.\n16:54:20.462 | INFO    | Flow run 'radical-finch' - Created task run 'say_goodbye-261e56a8-2' for task 'say_goodbye'\n16:54:20.474 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_goodbye-261e56a8-2' for execution.\n16:54:20.500 | INFO    | Flow run 'radical-finch' - Created task run 'say_hello-811087cd-3' for task 'say_hello'\n16:54:20.511 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_hello-811087cd-3' for execution.\n16:54:20.544 | INFO    | Flow run 'radical-finch' - Created task run 'say_goodbye-261e56a8-3' for task 'say_goodbye'\n16:54:20.555 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_goodbye-261e56a8-3' for execution.\nhello arthur\ngoodbye ford\ngoodbye arthur\nhello ford\ngoodbye marvin\ngoodbye trillian\nhello trillian\nhello marvin\n

    DaskTaskRunner automatically creates a local Dask cluster, then starts executing all of the tasks in parallel. The results do not return in the same order as the sequential code above.

    Notice what happens if you do not use the submit method when calling tasks:

    from prefect import flow, task\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n\n@flow(task_runner=DaskTaskRunner())\ndef greetings(names):\n    for name in names:\n        say_hello(name)\n        say_goodbye(name)\n\n\nif __name__ == \"__main__\":\n    greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n
    $ python dask_flow.py\n\n16:57:34.534 | INFO    | prefect.engine - Created flow run 'papaya-honeybee' for flow 'greetings'\n16:57:34.534 | INFO    | Flow run 'papaya-honeybee' - Starting 'DaskTaskRunner'; submitted tasks will be run concurrently...\n16:57:34.535 | INFO    | prefect.task_runner.dask - Creating a new Dask cluster with `distributed.deploy.local.LocalCluster`\n16:57:35.715 | INFO    | prefect.task_runner.dask - The Dask dashboard is available at <http://127.0.0.1:8787/status>\n16:57:35.787 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_hello-811087cd-0' for task 'say_hello'\n16:57:35.788 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_hello-811087cd-0' immediately...\nhello arthur\n16:57:35.810 | INFO    | Task run 'say_hello-811087cd-0' - Finished in state Completed()\n16:57:35.820 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_goodbye-261e56a8-0' for task 'say_goodbye'\n16:57:35.820 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_goodbye-261e56a8-0' immediately...\ngoodbye arthur\n16:57:35.840 | INFO    | Task run 'say_goodbye-261e56a8-0' - Finished in state Completed()\n16:57:35.849 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_hello-811087cd-1' for task 'say_hello'\n16:57:35.849 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_hello-811087cd-1' immediately...\nhello trillian\n16:57:35.869 | INFO    | Task run 'say_hello-811087cd-1' - Finished in state Completed()\n16:57:35.878 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_goodbye-261e56a8-1' for task 'say_goodbye'\n16:57:35.878 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_goodbye-261e56a8-1' immediately...\ngoodbye trillian\n16:57:35.894 | INFO    | Task run 'say_goodbye-261e56a8-1' - Finished in state Completed()\n16:57:35.907 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_hello-811087cd-2' for task 'say_hello'\n16:57:35.907 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_hello-811087cd-2' immediately...\nhello ford\n16:57:35.924 | INFO    | Task run 'say_hello-811087cd-2' - Finished in state Completed()\n16:57:35.933 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_goodbye-261e56a8-2' for task 'say_goodbye'\n16:57:35.933 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_goodbye-261e56a8-2' immediately...\ngoodbye ford\n16:57:35.951 | INFO    | Task run 'say_goodbye-261e56a8-2' - Finished in state Completed()\n16:57:35.959 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_hello-811087cd-3' for task 'say_hello'\n16:57:35.959 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_hello-811087cd-3' immediately...\nhello marvin\n16:57:35.976 | INFO    | Task run 'say_hello-811087cd-3' - Finished in state Completed()\n16:57:35.985 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_goodbye-261e56a8-3' for task 'say_goodbye'\n16:57:35.985 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_goodbye-261e56a8-3' immediately...\ngoodbye marvin\n16:57:36.004 | INFO    | Task run 'say_goodbye-261e56a8-3' - Finished in state Completed()\n16:57:36.289 | INFO    | Flow run 'papaya-honeybee' - Finished in state Completed('All states completed.')\n

    The tasks are not submitted to the DaskTaskRunner and are run sequentially.

    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/dask-ray-task-runners/#running-parallel-tasks-with-ray","title":"Running parallel tasks with Ray","text":"

    To demonstrate the ability to flexibly apply the task runner appropriate for your workflow, use the same flow as above, with a few minor changes to use the RayTaskRunner where we previously configured DaskTaskRunner.

    To configure your flow to use the RayTaskRunner:

    1. Make sure the prefect-ray collection is installed by running pip install prefect-ray.
    2. In your flow code, import RayTaskRunner from prefect_ray.task_runners.
    3. Assign it as the task runner when the flow is defined using the task_runner=RayTaskRunner argument.

    Ray environment limitations

    While we're excited about parallel task execution via Ray to Prefect, there are some inherent limitations with Ray you should be aware of:

    • Support for Python 3.11 is experimental.
    • Ray support for non-x86/64 architectures such as ARM/M1 processors with installation from pip alone and will be skipped during installation of Prefect. It is possible to manually install the blocking component with conda. See the Ray documentation for instructions.
    • Ray's Windows support is currently in beta.

    See the Ray installation documentation for further compatibility information.

    Save this code in ray_flow.py.

    from prefect import flow, task\nfrom prefect_ray.task_runners import RayTaskRunner\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n@flow(task_runner=RayTaskRunner())\ndef greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n\nif __name__ == \"__main__\":\n    greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n

    Now run ray_flow.py RayTaskRunner automatically creates a local Ray instance, then immediately starts executing all of the tasks in parallel. If you have an existing Ray instance, you can provide the address as a parameter to run tasks in the instance. See Running tasks on Ray for details.

    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/dask-ray-task-runners/#using-multiple-task-runners","title":"Using multiple task runners","text":"

    Many workflows include a variety of tasks, and not all of them benefit from parallel execution. You'll most likely want to use the Dask or Ray task runners and spin up their respective resources only for those tasks that need them.

    Because task runners are specified on flows, you can assign different task runners to tasks by using subflows to organize those tasks.

    This example uses the same tasks as the previous examples, but on the parent flow greetings() we use the default ConcurrentTaskRunner. Then we call a ray_greetings() subflow that uses the RayTaskRunner to execute the same tasks in a Ray instance.

    from prefect import flow, task\nfrom prefect_ray.task_runners import RayTaskRunner\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n@flow(task_runner=RayTaskRunner())\ndef ray_greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n\n@flow()\ndef greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n    ray_greetings(names)\n\nif __name__ == \"__main__\":\n    greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n

    If you save this as ray_subflow.py and run it, you'll see that the flow greetings runs as you'd expect for a concurrent flow, then flow ray-greetings spins up a Ray instance to run the tasks again.

    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/docker/","title":"Running Flows with Docker","text":"

    In the Deployments tutorial, we looked at serving a flow that enables scheduling or creating flow runs via the Prefect API.

    With our Python script in hand, we can build a Docker image for our script, allowing us to serve our flow in various remote environments. We'll use Kubernetes in this guide, but you can use any Docker-compatible infrastructure.

    In this guide we'll:

    • Write a Dockerfile to build an image that stores our Prefect flow code.
    • Build a Docker image for our flow.
    • Deploy and run our Docker image on a Kubernetes cluster.
    • Look at the Prefect-maintained Docker images and discuss options for use

    Note that in this guide we'll create a Dockerfile from scratch. Alternatively, Prefect makes it convenient to build a Docker image as part of deployment creation. You can even include environment variables and specify additional Python packages to install at runtime.

    If creating a deployment with a prefect.yaml file, the build step makes it easy to customize your Docker image and push it to the registry of your choice. See an example here.

    Deployment creation with a Python script that includes flow.deploy similarly allows you to customize your Docker image with keyword arguments as shown below.

    ...\n\nif __name__ == \"__main__\":\n    hello_world.deploy(\n        name=\"my-first-deployment\",\n        work_pool_name=\"above-ground\",\n        image='my_registry/hello_world:demo',\n        job_variables={\"env\": { \"EXTRA_PIP_PACKAGES\": \"boto3\" } }\n    )\n
    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#prerequisites","title":"Prerequisites","text":"

    To complete this guide, you'll need the following:

    • A Python script that defines and serves a flow.
    • We'll use the flow script and deployment from the Deployments tutorial.
    • Access to a running Prefect API server.
    • You can sign up for a forever free Prefect Cloud account or run a Prefect API server locally with prefect server start.
    • Docker Desktop installed on your machine.
    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#writing-a-dockerfile","title":"Writing a Dockerfile","text":"

    First let's make a clean directory to work from, prefect-docker-guide.

    mkdir prefect-docker-guide\ncd prefect-docker-guide\n

    In this directory, we'll create a sub-directory named flows and put our flow script from the Deployments tutorial in it.

    mkdir flows\ncd flows\ntouch prefect-docker-guide-flow.py\n

    Here's the flow code for reference:

    prefect-docker-guide-flow.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.serve(name=\"prefect-docker-guide\")\n

    The next file we'll add to the prefect-docker-guide directory is a requirements.txt. We'll include all dependencies required for our prefect-docker-guide-flow.py script in the Docker image we'll build.

    touch requirements.txt\n

    Here's what we'll put in our requirements.txt file:

    requirements.txt
    prefect>=2.12.0\nhttpx\n

    Next, we'll create a Dockerfile that we'll use to create a Docker image that will also store the flow code.

    touch Dockerfile\n

    We'll add the following content to our Dockerfile:

    Dockerfile
    # We're using the latest version of Prefect with Python 3.10\nFROM prefecthq/prefect:2-python3.10\n\n# Add our requirements.txt file to the image and install dependencies\nCOPY requirements.txt .\nRUN pip install -r requirements.txt --trusted-host pypi.python.org --no-cache-dir\n\n# Add our flow code to the image\nCOPY flows /opt/prefect/flows\n\n# Run our flow script when the container starts\nCMD [\"python\", \"flows/prefect-docker-guide-flow.py\"]\n
    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#building-a-docker-image","title":"Building a Docker image","text":"

    Now that we have a Dockerfile we can build our image by running:

    docker build -t prefect-docker-guide-image .\n

    We can check that our build worked by running a container from our new image.

    CloudSelf-hosted

    Our container will need an API URL and and API key to communicate with Prefect Cloud.

    • You can get an API key from the API Keys section of the user settings in the Prefect UI.

    • You can get your API URL by running prefect config view and copying the PREFECT_API_URL value.

    We'll provide both these values to our container by passing them as environment variables with the -e flag.

    docker run -e PREFECT_API_URL=YOUR_PREFECT_API_URL -e PREFECT_API_KEY=YOUR_API_KEY prefect-docker-guide-image\n

    After running the above command, the container should start up and serve the flow within the container!

    Our container will need an API URL and network access to communicate with the Prefect API.

    For this guide, we'll assume the Prefect API is running on the same machine that we'll run our container on and the Prefect API was started with prefect server start. If you're running a different setup, check out the Hosting a Prefect server guide for information on how to connect to your Prefect API instance.

    To ensure that our flow container can communicate with the Prefect API, we'll set our PREFECT_API_URL to http://host.docker.internal:4200/api. If you're running Linux, you'll need to set your PREFECT_API_URL to http://localhost:4200/api and use the --network=\"host\" option instead.

    docker run --network=\"host\" -e PREFECT_API_URL=http://host.docker.internal:4200/api prefect-docker-guide-image\n

    After running the above command, the container should start up and serve the flow within the container!

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#deploying-to-a-remote-environment","title":"Deploying to a remote environment","text":"

    Now that we have a Docker image with our flow code embedded, we can deploy it to a remote environment!

    For this guide, we'll simulate a remote environment by using Kubernetes locally with Docker Desktop. You can use the instructions provided by Docker to set up Kubernetes locally.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#creating-a-kubernetes-deployment-manifest","title":"Creating a Kubernetes deployment manifest","text":"

    To ensure the process serving our flow is always running, we'll create a Kubernetes deployment. If our flow's container ever crashes, Kubernetes will automatically restart it, ensuring that we won't miss any scheduled runs.

    First, we'll create a deployment-manifest.yaml file in our prefect-docker-guide directory:

    touch deployment-manifest.yaml\n

    And we'll add the following content to our deployment-manifest.yaml file:

    CloudSelf-hosted deployment-manifest.yaml
    apiVersion: apps/v1\nkind: Deployment\nmetadata:\n  name: prefect-docker-guide\nspec:\n  replicas: 1\n  selector:\n    matchLabels:\n      flow: get-repo-info\n  template:\n    metadata:\n      labels:\n        flow: get-repo-info\n    spec:\n      containers:\n      - name: flow-container\n        image: prefect-docker-guide-image:latest\n        env:\n        - name: PREFECT_API_URL\n          value: YOUR_PREFECT_API_URL\n        - name: PREFECT_API_KEY\n          value: YOUR_API_KEY\n        # Never pull the image because we're using a local image\n        imagePullPolicy: Never\n

    Keep your API key secret

    In the above manifest we are passing in the Prefect API URL and API key as environment variables. This approach is simple, but it is not secure. If you are deploying your flow to a remote cluster, you should use a Kubernetes secret to store your API key.

    deployment-manifest.yaml
    apiVersion: apps/v1\nkind: Deployment\nmetadata:\n  name: prefect-docker-guide\nspec:\n  replicas: 1\n  selector:\n    matchLabels:\n      flow: get-repo-info\n  template:\n    metadata:\n      labels:\n        flow: get-repo-info\n    spec:\n      containers:\n      - name: flow-container\n        image: prefect-docker-guide-image:latest\n        env:\n        - name: PREFECT_API_URL\n          value: <http://host.docker.internal:4200/api>\n        # Never pull the image because we're using a local image\n        imagePullPolicy: Never\n

    Linux users

    If you're running Linux, you'll need to set your PREFECT_API_URL to use the IP address of your machine instead of host.docker.internal.

    This manifest defines how our image will run when deployed in our Kubernetes cluster. Note that we will be running a single replica of our flow container. If you want to run multiple replicas of your flow container to keep up with an active schedule, or because our flow is resource-intensive, you can increase the replicas value.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#deploying-our-flow-to-the-cluster","title":"Deploying our flow to the cluster","text":"

    Now that we have a deployment manifest, we can deploy our flow to the cluster by running:

    kubectl apply -f deployment-manifest.yaml\n

    We can monitor the status of our Kubernetes deployment by running:

    kubectl get deployments\n

    Once the deployment has successfully started, we can check the logs of our flow container by running the following:

    kubectl logs -l flow=get-repo-info\n

    Now that we're serving our flow in our cluster, we can trigger a flow run by running:

    prefect deployment run get-repo-info/prefect-docker-guide\n

    If we navigate to the URL provided by the prefect deployment run command, we can follow the flow run via the logs in the Prefect UI!

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#prefect-maintained-docker-images","title":"Prefect-maintained Docker images","text":"

    Every release of Prefect results in several new Docker images. These images are all named prefecthq/prefect and their tags identify their differences.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#image-tags","title":"Image tags","text":"

    When a release is published, images are built for all of Prefect's supported Python versions. These images are tagged to identify the combination of Prefect and Python versions contained. Additionally, we have \"convenience\" tags which are updated with each release to facilitate automatic updates.

    For example, when release 2.11.5 is published:

    1. Images with the release packaged are built for each supported Python version (3.8, 3.9, 3.10, 3.11) with both standard Python and Conda.
    2. These images are tagged with the full description, e.g. prefect:2.1.1-python3.10 and prefect:2.1.1-python3.10-conda.
    3. For users that want more specific pins, these images are also tagged with the SHA of the git commit of the release, e.g. sha-88a7ff17a3435ec33c95c0323b8f05d7b9f3f6d2-python3.10
    4. For users that want to be on the latest 2.1.x release, receiving patch updates, we update a tag without the patch version to this release, e.g. prefect.2.1-python3.10.
    5. For users that want to be on the latest 2.x.y release, receiving minor version updates, we update a tag without the minor or patch version to this release, e.g. prefect.2-python3.10
    6. Finally, for users who want the latest 2.x.y release without specifying a Python version, we update 2-latest to the image for our highest supported Python version, which in this case would be equivalent to prefect:2.1.1-python3.10.

    Choose image versions carefully

    It's a good practice to use Docker images with specific Prefect versions in production.

    Use care when employing images that automatically update to new versions (such as prefecthq/prefect:2-python3.11 or prefecthq/prefect:2-latest).

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#standard-python","title":"Standard Python","text":"

    Standard Python images are based on the official Python slim images, e.g. python:3.10-slim.

    Tag Prefect Version Python Version 2-latest most recent v2 PyPi version 3.10 2-python3.11 most recent v2 PyPi version 3.11 2-python3.10 most recent v2 PyPi version 3.10 2-python3.9 most recent v2 PyPi version 3.9 2-python3.8 most recent v2 PyPi version 3.8 2.X-python3.11 2.X 3.11 2.X-python3.10 2.X 3.10 2.X-python3.9 2.X 3.9 2.X-python3.8 2.X 3.8 sha-<hash>-python3.11 <hash> 3.11 sha-<hash>-python3.10 <hash> 3.10 sha-<hash>-python3.9 <hash> 3.9 sha-<hash>-python3.8 <hash> 3.8","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#conda-flavored-python","title":"Conda-flavored Python","text":"

    Conda flavored images are based on continuumio/miniconda3. Prefect is installed into a conda environment named prefect.

    Tag Prefect Version Python Version 2-latest-conda most recent v2 PyPi version 3.10 2-python3.11-conda most recent v2 PyPi version 3.11 2-python3.10-conda most recent v2 PyPi version 3.10 2-python3.9-conda most recent v2 PyPi version 3.9 2-python3.8-conda most recent v2 PyPi version 3.8 2.X-python3.11-conda 2.X 3.11 2.X-python3.10-conda 2.X 3.10 2.X-python3.9-conda 2.X 3.9 2.X-python3.8-conda 2.X 3.8 sha-<hash>-python3.11-conda <hash> 3.11 sha-<hash>-python3.10-conda <hash> 3.10 sha-<hash>-python3.9-conda <hash> 3.9 sha-<hash>-python3.8-conda <hash> 3.8","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#building-your-own-image","title":"Building your own image","text":"

    If your flow relies on dependencies not found in the default prefecthq/prefect images, you may want to build your own image. You can either base it off of one of the provided prefecthq/prefect images, or build your own image. See the Work pool deployment guide for discussion of how Prefect can help you build custom images with dependencies specifiied in a requirements.txt file.

    By default, Prefect work pools that use containers refer to the 2-latest image. You can specify another image at work pool creation. The work pool image choice can be overridden in individual deployments.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#extending-the-prefecthqprefect-image-manually","title":"Extending the prefecthq/prefect image manually","text":"

    Here we provide an example Dockerfile for building an image based on prefecthq/prefect:2-latest, but with scikit-learn installed.

    FROM prefecthq/prefect:2-latest\n\nRUN pip install scikit-learn\n
    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#choosing-an-image-strategy","title":"Choosing an image strategy","text":"

    The options described above have different complexity (and performance) characteristics. For choosing a strategy, we provide the following recommendations:

    • If your flow only makes use of tasks defined in the same file as the flow, or tasks that are part of prefect itself, then you can rely on the default provided prefecthq/prefect image.

    • If your flow requires a few extra dependencies found on PyPI, you can use the default prefecthq/prefect image and set prefect.deployments.steps.pip_install_requirements: in the pullstep to install these dependencies at runtime.

    • If the installation process requires compiling code or other expensive operations, you may be better off building a custom image instead.

    • If your flow (or flows) require extra dependencies or shared libraries, we recommend building a shared custom image with all the extra dependencies and shared task definitions you need. Your flows can then all rely on the same image, but have their source stored externally. This option can ease development, as the shared image only needs to be rebuilt when dependencies change, not when the flow source changes.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#next-steps","title":"Next steps","text":"

    We only served a single flow in this guide, but you can extend this setup to serve multiple flows in a single Docker image by updating your Python script to using flow.to_deployment and serve to serve multiple flows or the same flow with different configuration.

    To learn more about deploying flows, check out the Deployments concept doc!

    For advanced infrastructure requirements, such as executing each flow run within its own dedicated Docker container, learn more in the Work pool deployment guide.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/global-concurrency-limits/","title":"Global concurrency limits and rate limits","text":"

    Global concurrency limits allow you to manage execution efficiently, controlling how many tasks, flows, or other operations can run simultaneously. They are ideal when optimizing resource usage, preventing bottlenecks, and customizing task execution are priorities.

    Clarification on use of the term 'tasks'

    In the context of global concurrency and rate limits, \"tasks\" refers not specifically to Prefect tasks, but to concurrent units of work in general, such as those managed by an event loop or TaskGroup in asynchronous programming. These general \"tasks\" could include Prefect tasks when they are part of an asynchronous execution environment.

    Rate Limits ensure system stability by governing the frequency of requests or operations. They are suitable for preventing overuse, ensuring fairness, and handling errors gracefully.

    When selecting between Concurrency and Rate Limits, consider your primary goal. Choose Concurrency Limits for resource optimization and task management. Choose Rate Limits to maintain system stability and fair access to services.

    The core difference between a rate limit and a concurrency limit is the way in which slots are released. With a rate limit, slots are released at a controlled rate, controlled by slot_decay_per_second whereas with a concurrency limit, slots are released when the concurrency manager is exited.

    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#managing-global-concurrency-limits-and-rate-limits","title":"Managing Global concurrency limits and rate limits","text":"

    You can create, read, edit, and delete concurrency limits via the Prefect UI.

    When creating a concurrency limit, you can specify the following parameters:

    • Name: The name of the concurrency limit. This name is also how you'll reference the concurrency limit in your code. Special characters, such as /, %, &, >, <, are not allowed.
    • Concurrency Limit: The maximum number of slots that can be occupied on this concurrency limit.
    • Slot Decay Per Second: Controls the rate at which slots are released when the concurrency limit is used as a rate limit. This value must be configured when using the rate_limit function.
    • Active: Whether or not the concurrency limit is in an active state.
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#active-vs-inactive-limits","title":"Active vs inactive limits","text":"

    Global concurrency limits can be in either an active or inactive state.

    • Active: In this state, slots can be occupied, and code execution will be blocked when slots are unable to be acquired.
    • Inactive: In this state, slots will not be occupied, and code execution will not be blocked. Concurrency enforcement occurs only when you activate the limit.
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#slot-decay","title":"Slot decay","text":"

    Global concurrency limits can be configured with slot decay. This is used when the concurrency limit is used as a rate limit, and it governs the pace at which slots are released or become available for reuse after being occupied. These slots effectively represent the concurrency capacity within a specific concurrency limit. The concept is best understood as the rate at which these slots \"decay\" or refresh.

    To configure slot decay, you can set the slot_decay_per_second parameter when defining or adjusting a concurrency limit.

    For practical use, consider the following:

    • Higher values: Setting slot_decay_per_second to a higher value, such as 5.0, results in slots becoming available relatively quickly. In this scenario, a slot that was occupied by a task will free up after just 0.2 (1.0 / 5.0) seconds.

    • Lower values: Conversely, setting slot_decay_per_second to a lower value, like 0.1, causes slots to become available more slowly. In this scenario it would take 10 (1.0 / 0.1) seconds for a slot to become available again after occupancy

    Slot decay provides fine-grained control over the availability of slots, enabling you to optimize the rate of your workflow based on your specific requirements.

    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#using-the-concurrency-context-manager","title":"Using the concurrency context manager","text":"

    The concurrencycontext manager allows control over the maximum number of concurrent operations. You can select either the synchronous (sync) or asynchronous (async) version, depending on your use case. Here's how to use it:

    Concurrency limits are implicitly created

    When using the concurrency context manager, the concurrency limit you use will be created, in an inactive state, if it does not already exist.

    Sync

    from prefect import flow, task\nfrom prefect.concurrency.sync import concurrency\n\n\n@task\ndef process_data(x, y):\n    with concurrency(\"database\", occupy=1):\n        return x + y\n\n\n@flow\ndef my_flow():\n    for x, y in [(1, 2), (2, 3), (3, 4), (4, 5)]:\n        process_data.submit(x, y)\n\n\nif __name__ == \"__main__\":\n    my_flow()\n

    Async

    import asyncio\nfrom prefect import flow, task\nfrom prefect.concurrency.asyncio import concurrency\n\n\n@task\nasync def process_data(x, y):\n    async with concurrency(\"database\", occupy=1):\n        return x + y\n\n\n@flow\nasync def my_flow():\n    for x, y in [(1, 2), (2, 3), (3, 4), (4, 5)]:\n        await process_data.submit(x, y)\n\n\nif __name__ == \"__main__\":\n    asyncio.run(my_flow())\n
    1. The code imports the necessary modules and the concurrency context manager. Use the prefect.concurrency.sync module for sync usage and the prefect.concurrency.asyncio module for async usage.
    2. It defines a process_data task, taking x and y as input arguments. Inside this task, the concurrency context manager controls concurrency, using the database concurrency limit and occupying one slot. If another task attempts to run with the same limit and no slots are available, that task will be blocked until a slot becomes available.
    3. A flow named my_flow is defined. Within this flow, it iterates through a list of tuples, each containing pairs of x and y values. For each pair, the process_data task is submitted with the corresponding x and y values for processing.
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#using-rate_limit","title":"Using rate_limit","text":"

    The Rate Limit feature provides control over the frequency of requests or operations, ensuring responsible usage and system stability. Depending on your requirements, you can utilize rate_limit to govern both synchronous (sync) and asynchronous (async) operations. Here's how to make the most of it:

    Slot decay

    When using the rate_limit function the concurrency limit you use must have a slot decay configured.

    Sync

    from prefect import flow, task\nfrom prefect.concurrency.sync import rate_limit\n\n\n@task\ndef make_http_request():\n    rate_limit(\"rate-limited-api\")\n    print(\"Making an HTTP request...\")\n\n\n@flow\ndef my_flow():\n    for _ in range(10):\n        make_http_request.submit()\n\n\nif __name__ == \"__main__\":\n    my_flow()\n

    Async

    import asyncio\n\nfrom prefect import flow, task\nfrom prefect.concurrency.asyncio import rate_limit\n\n\n@task\nasync def make_http_request():\n    await rate_limit(\"rate-limited-api\")\n    print(\"Making an HTTP request...\")\n\n\n@flow\nasync def my_flow():\n    for _ in range(10):\n        await make_http_request.submit()\n\n\nif __name__ == \"__main__\":\n    asyncio.run(my_flow())\n
    1. The code imports the necessary modules and the rate_limit function. Use the prefect.concurrency.sync module for sync usage and the prefect.concurrency.asyncio module for async usage.
    2. It defines a make_http_request task. Inside this task, the rate_limit function is used to ensure that the requests are made at a controlled pace.
    3. A flow named my_flow is defined. Within this flow the make_http_request task is submitted 10 times.
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#using-concurrency-and-rate_limit-outside-of-a-flow","title":"Using concurrency and rate_limit outside of a flow","text":"

    concurreny and rate_limit can be used outside of a flow to control concurrency and rate limits for any operation.

    import asyncio\n\nfrom prefect.concurrency.asyncio import rate_limit\n\n\nasync def main():\n    for _ in range(10):\n        await rate_limit(\"rate-limited-api\")\n        print(\"Making an HTTP request...\")\n\n\n\nif __name__ == \"__main__\":\n    asyncio.run(main())\n
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#use-cases","title":"Use cases","text":"","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#throttling-task-submission","title":"Throttling task submission","text":"

    Throttling task submission to avoid overloading resources, to comply with external rate limits, or ensure a steady, controlled flow of work.

    In this scenario the rate_limit function is used to throttle the submission of tasks. The rate limit acts as a bottleneck, ensuring that tasks are submitted at a controlled rate, governed by the slot_decay_per_second setting on the associated concurrency limit.

    from prefect import flow, task\nfrom prefect.concurrency.sync import rate_limit\n\n\n@task\ndef my_task(i):\n    return i\n\n\n@flow\ndef my_flow():\n    for _ in range(100):\n        rate_limit(\"slow-my-flow\", occupy=1)\n        my_task.submit(1)\n\n\nif __name__ == \"__main__\":\n    my_flow()\n
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#managing-database-connections","title":"Managing database connections","text":"

    Managing the maximum number of concurrent database connections to avoid exhausting database resources.

    In this scenario we've setup a concurrency limit named database and given it a maximum concurrency limit that matches the maximum number of database connections we want to allow. We then use the concurrency context manager to control the number of database connections allowed at any one time.

    from prefect import flow, task, concurrency\nimport psycopg2\n\n@task\ndef database_query(query):\n    # Here we request a single slot on the 'database' concurrency limit. This\n    # will block in the case that all of the database connections are in use\n    # ensuring that we never exceed the maximum number of database connections.\n    with concurrency(\"database\", occupy=1):\n        connection = psycopg2.connect(\"<connection_string>\")\n        cursor = connection.cursor()\n        cursor.execute(query)\n        result = cursor.fetchall()\n        connection.close()\n        return result\n\n@flow\ndef my_flow():\n    queries = [\"SELECT * FROM table1\", \"SELECT * FROM table2\", \"SELECT * FROM table3\"]\n\n    for query in queries:\n        database_query.submit(query)\n\nif __name__ == \"__main__\":\n    my_flow()\n
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#parallel-data-processing","title":"Parallel data processing","text":"

    Limiting the maximum number of parallel processing tasks.

    In this scenario we want to limit the number of process_data tasks to five at any one time. We do this by using the concurrency context manager to request five slots on the data-processing concurrency limit. This will block until five slots are free and then submit five more tasks, ensuring that we never exceed the maximum number of parallel processing tasks.

    import asyncio\nfrom prefect.concurrency.sync import concurrency\n\n\nasync def process_data(data):\n    print(f\"Processing: {data}\")\n    await asyncio.sleep(1)\n    return f\"Processed: {data}\"\n\n\nasync def main():\n    data_items = list(range(100))\n    processed_data = []\n\n    while data_items:\n        with concurrency(\"data-processing\", occupy=5):\n            chunk = [data_items.pop() for _ in range(5)]\n            processed_data += await asyncio.gather(\n                *[process_data(item) for item in chunk]\n            )\n\n    print(processed_data)\n\n\nif __name__ == \"__main__\":\n    asyncio.run(main())\n
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/host/","title":"Hosting a Prefect server","text":"

    After you install Prefect you have a Python SDK client that can communicate with Prefect Cloud, the platform hosted by Prefect. You also have an API server backed by a database and a UI.

    In this section you'll learn how to host your own Prefect server.

    Spin up a local Prefect server UI with the prefect server start CLI command in the terminal:

    prefect server start\n

    Open the URL for the Prefect server UI (http://127.0.0.1:4200 by default) in a browser.

    Shut down the Prefect server with ctrl + c in the terminal.","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#differences-between-a-self-hosted-prefect-server-and-prefect-cloud","title":"Differences between a self-hosted Prefect server and Prefect Cloud","text":"

    A self-hosted Prefect server and Prefect Cloud share a common set of features. Prefect Cloud includes the following additional features:

    • Workspaces \u2014 isolated environments to organize your flows, deployments, and flow runs.
    • Automations \u2014 configure triggers, actions, and notifications in response to real-time monitoring events.
    • Email notifications \u2014 send email alerts from Prefect's servers based on automation triggers.
    • Service accounts \u2014 configure API access for running workers or executing flow runs on remote infrastructure.
    • Custom role-based access controls (RBAC) \u2014 assign users granular permissions to perform activities within an account or workspace.
    • Single Sign-on (SSO) \u2014 authentication using your identity provider.
    • Audit Log \u2014 a record of user activities to monitor security and compliance.

    You can read more about Prefect Cloud in the Cloud section.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#configuring-a-prefect-server-instance","title":"Configuring a Prefect server instance","text":"

    Go to your terminal session and run this command to set the API URL to point to a Prefect server instance:

    prefect config set PREFECT_API_URL=\"http://127.0.0.1:4200/api\"\n

    PREFECT_API_URL required when running Prefect inside a container

    You must set the API server address to use Prefect within a container, such as a Docker container.

    You can save the API server address in a Prefect profile. Whenever that profile is active, the API endpoint will be be at that address.

    See Profiles & Configuration for more information on profiles and configurable Prefect settings.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#prefect-database","title":"Prefect database","text":"

    The Prefect database persists data to track the state of your flow runs and related Prefect concepts, including:

    • Flow run and task run state
    • Run history
    • Logs
    • Deployments
    • Flow and task run concurrency limits
    • Storage blocks for flow and task results
    • Variables
    • Artifacts
    • Work pool status

    Currently Prefect supports the following databases:

    • SQLite: The default in Prefect, and our recommendation for lightweight, single-server deployments. SQLite requires essentially no setup.
    • PostgreSQL: Best for connecting to external databases, but does require additional setup (such as Docker). Prefect uses the pg_trgm extension, so it must be installed and enabled.
    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#using-the-database","title":"Using the database","text":"

    A local SQLite database is the default database and is configured upon Prefect installation. The database is located at ~/.prefect/prefect.db by default.

    To reset your database, run the CLI command:

    prefect server database reset -y\n

    This command will clear all data and reapply the schema.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#database-settings","title":"Database settings","text":"

    Prefect provides several settings for configuring the database. Here are the default settings:

    PREFECT_API_DATABASE_CONNECTION_URL='sqlite+aiosqlite:///${PREFECT_HOME}/prefect.db'\nPREFECT_API_DATABASE_ECHO='False'\nPREFECT_API_DATABASE_MIGRATE_ON_START='True'\nPREFECT_API_DATABASE_PASSWORD='None'\n

    You can save a setting to your active Prefect profile with prefect config set.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#configuring-a-postgresql-database","title":"Configuring a PostgreSQL database","text":"

    To connect Prefect to a PostgreSQL database, you can set the following environment variable:

    prefect config set PREFECT_API_DATABASE_CONNECTION_URL=\"postgresql+asyncpg://postgres:yourTopSecretPassword@localhost:5432/prefect\"\n

    The above environment variable assumes that:

    • You have a username called postgres
    • Your password is set to yourTopSecretPassword
    • Your database runs on the same host as the Prefect server instance, localhost
    • You use the default PostgreSQL port 5432
    • Your PostgreSQL instance has a database called prefect

    If you want to quickly start a PostgreSQL instance that can be used as your Prefect database, you can use the following command that will start a Docker container running PostgreSQL:

    docker run -d --name prefect-postgres -v prefectdb:/var/lib/postgresql/data -p 5432:5432 -e POSTGRES_USER=postgres -e POSTGRES_PASSWORD=yourTopSecretPassword -e POSTGRES_DB=prefect postgres:latest\n

    The above command:

    • Pulls the latest version of the official postgres Docker image, which is compatible with Prefect.
    • Starts a container with the name prefect-postgres.
    • Creates a database prefect with a user postgres and yourTopSecretPassword password.
    • Mounts the PostgreSQL data to a Docker volume called prefectdb to provide persistence if you ever have to restart or rebuild that container.

    You can inspect your profile to be sure that the environment variable has been set properly:

    prefect config view --show-sources\n

    Start the Prefect server and it should from now on use your PostgreSQL database instance:

    prefect server start\n
    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#in-memory-database","title":"In-memory database","text":"

    One of the benefits of SQLite is in-memory database support.

    To use an in-memory SQLite database, set the following environment variable:

    prefect config set PREFECT_API_DATABASE_CONNECTION_URL=\"sqlite+aiosqlite:///file::memory:?cache=shared&uri=true&check_same_thread=false\"\n

    Use SQLite database for testing only

    SQLite is only supported by Prefect for testing purposes and is not compatible with multiprocessing.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#database-versions","title":"Database versions","text":"

    The following database versions are required for use with Prefect:

    • SQLite 3.24 or newer
    • PostgreSQL 13.0 or newer
    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#migrations","title":"Migrations","text":"

    Prefect uses Alembic to manage database migrations. Alembic is a database migration tool for usage with the SQLAlchemy Database Toolkit for Python. Alembic provides a framework for generating and applying schema changes to a database.

    To apply migrations to your database you can run the following commands:

    To upgrade:

    prefect server database upgrade -y\n

    To downgrade:

    prefect server database downgrade -y\n

    You can use the -r flag to specify a specific migration version to upgrade or downgrade to. For example, to downgrade to the previous migration version you can run:

    prefect server database downgrade -y -r -1\n

    or to downgrade to a specific revision:

    prefect server database downgrade -y -r d20618ce678e\n

    See the contributing docs for information on how to create new database migrations.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#notifications","title":"Notifications","text":"

    When you use Prefect Cloud you gain access to a hosted platform with Workspace & User controls, Events, and Automations. Prefect Cloud has an option for automation notifications. The more limited Notifications option is provided for the self-hosted Prefect server.

    Notifications enable you to set up alerts that are sent when a flow enters any state you specify. When your flow and task runs changes state, Prefect notes the state change and checks whether the new state matches any notification policies. If it does, a new notification is queued.

    Prefect supports sending notifications via:

    • Slack message to a channel
    • Microsoft Teams message to a channel
    • Opsgenie to alerts
    • PagerDuty to alerts
    • Twilio to phone numbers
    • Email (requires your own server)

    Notifications in Prefect Cloud

    Prefect Cloud uses the robust Automations interface to enable notifications related to flow run state changes and work pool status.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#configure-notifications","title":"Configure notifications","text":"

    To configure a notification in a Prefect server, go to the Notifications page and select Create Notification or the + button.

    Notifications are structured just as you would describe them to someone. You can choose:

    • Which run states should trigger a notification.
    • Tags to filter which flow runs are covered by the notification.
    • Whether to send an email, a Slack message, Microsoft Teams message, or other services.

    For email notifications (supported on Prefect Cloud only), the configuration requires email addresses to which the message is sent.

    For Slack notifications, the configuration requires webhook credentials for your Slack and the channel to which the message is sent.

    For example, to get a Slack message if a flow with a daily-etl tag fails, the notification will read:

    If a run of any flow with daily-etl tag enters a failed state, send a notification to my-slack-webhook

    When the conditions of the notification are triggered, you\u2019ll receive a message:

    The fuzzy-leopard run of the daily-etl flow entered a failed state at 22-06-27 16:21:37 EST.

    On the Notifications page you can pause, edit, or delete any configured notification.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/logs/","title":"Logging","text":"

    Prefect enables you to log a variety of useful information about your flow and task runs, capturing information about your workflows for purposes such as monitoring, troubleshooting, and auditing.

    Prefect captures logs for your flow and task runs by default, even if you have not started a Prefect server with prefect server start.

    You can view and filter logs in the Prefect UI or Prefect Cloud, or access log records via the API.

    Prefect enables fine-grained customization of log levels for flows and tasks, including configuration for default levels and log message formatting.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-overview","title":"Logging overview","text":"

    Whenever you run a flow, Prefect automatically logs events for flow runs and task runs, along with any custom log handlers you have configured. No configuration is needed to enable Prefect logging.

    For example, say you created a simple flow in a file flow.py. If you create a local flow run with python flow.py, you'll see an example of the log messages created automatically by Prefect:

    16:45:44.534 | INFO    | prefect.engine - Created flow run 'gray-dingo' for flow\n'hello-flow'\n16:45:44.534 | INFO    | Flow run 'gray-dingo' - Using task runner 'SequentialTaskRunner'\n16:45:44.598 | INFO    | Flow run 'gray-dingo' - Created task run 'hello-task-54135dc1-0'\nfor task 'hello-task'\nHello world!\n16:45:44.650 | INFO    | Task run 'hello-task-54135dc1-0' - Finished in state\nCompleted(None)\n16:45:44.672 | INFO    | Flow run 'gray-dingo' - Finished in state\nCompleted('All states completed.')\n

    You can see logs for a flow run in the Prefect UI by navigating to the Flow Runs page and selecting a specific flow run to inspect.

    These log messages reflect the logging configuration for log levels and message formatters. You may customize the log levels captured and the default message format through configuration, and you can capture custom logging events by explicitly emitting log messages during flow and task runs.

    Prefect supports the standard Python logging levels CRITICAL, ERROR, WARNING, INFO, and DEBUG. By default, Prefect displays INFO-level and above events. You can configure the root logging level as well as specific logging levels for flow and task runs.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-configuration","title":"Logging configuration","text":"","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-settings","title":"Logging settings","text":"

    Prefect provides several settings for configuring logging level and loggers.

    By default, Prefect displays INFO-level and above logging records. You may change this level to DEBUG and DEBUG-level logs created by Prefect will be shown as well. You may need to change the log level used by loggers from other libraries to see their log records.

    You can override any logging configuration by setting an environment variable or Prefect Profile setting using the syntax PREFECT_LOGGING_[PATH]_[TO]_[KEY], with [PATH]_[TO]_[KEY] corresponding to the nested address of any setting.

    For example, to change the default logging levels for Prefect to DEBUG, you can set the environment variable PREFECT_LOGGING_LEVEL=\"DEBUG\".

    You may also configure the \"root\" Python logger. The root logger receives logs from all loggers unless they explicitly opt out by disabling propagation. By default, the root logger is configured to output WARNING level logs to the console. As with other logging settings, you can override this from the environment or in the logging configuration file. For example, you can change the level with the variable PREFECT_LOGGING_ROOT_LEVEL.

    You may adjust the log level used by specific handlers. For example, you could set PREFECT_LOGGING_HANDLERS_API_LEVEL=ERROR to have only ERROR logs reported to the Prefect API. The console handlers will still default to level INFO.

    There is a logging.yml file packaged with Prefect that defines the default logging configuration.

    You can customize logging configuration by creating your own version of logging.yml with custom settings, by either creating the file at the default location (/.prefect/logging.yml) or by specifying the path to the file with PREFECT_LOGGING_SETTINGS_PATH. (If the file does not exist at the specified location, Prefect ignores the setting and uses the default configuration.)

    See the Python Logging configuration documentation for more information about the configuration options and syntax used by logging.yml.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#prefect-loggers","title":"Prefect loggers","text":"

    To access the Prefect logger, import from prefect import get_run_logger. You can send messages to the logger in both flows and tasks.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-in-flows","title":"Logging in flows","text":"

    To log from a flow, retrieve a logger instance with get_run_logger(), then call the standard Python logging methods.

    from prefect import flow, get_run_logger\n\n@flow(name=\"log-example-flow\")\ndef logger_flow():\n    logger = get_run_logger()\n    logger.info(\"INFO level log message.\")\n

    Prefect automatically uses the flow run logger based on the flow context. If you run the above code, Prefect captures the following as a log event.

    15:35:17.304 | INFO    | Flow run 'mottled-marten' - INFO level log message.\n

    The default flow run log formatter uses the flow run name for log messages.

    Note

    Starting in 2.7.11, if you use a logger that sends logs to the API outside of a flow or task run, a warning will be displayed instead of an error. You can silence this warning by setting `PREFECT_LOGGING_TO_API_WHEN_MISSING_FLOW=ignore` or have the logger raise an error by setting the value to `error`.\n
    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-in-tasks","title":"Logging in tasks","text":"

    Logging in tasks works much as logging in flows: retrieve a logger instance with get_run_logger(), then call the standard Python logging methods.

    from prefect import flow, task, get_run_logger\n\n@task(name=\"log-example-task\")\ndef logger_task():\n    logger = get_run_logger()\n    logger.info(\"INFO level log message from a task.\")\n\n@flow(name=\"log-example-flow\")\ndef logger_flow():\n    logger_task()\n

    Prefect automatically uses the task run logger based on the task context. The default task run log formatter uses the task run name for log messages.

    15:33:47.179 | INFO   | Task run 'logger_task-80a1ffd1-0' - INFO level log message from a task.\n

    The underlying log model for task runs captures the task name, task run ID, and parent flow run ID, which are persisted to the database for reporting and may also be used in custom message formatting.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-print-statements","title":"Logging print statements","text":"

    Prefect provides the log_prints option to enable the logging of print statements at the task or flow level. When log_prints=True for a given task or flow, the Python builtin print will be patched to redirect to the Prefect logger for the scope of that task or flow.

    By default, tasks and subflows will inherit the log_prints setting from their parent flow, unless opted out with their own explicit log_prints setting.

    from prefect import task, flow\n\n@task\ndef my_task():\n    print(\"we're logging print statements from a task\")\n\n@flow(log_prints=True)\ndef my_flow():\n    print(\"we're logging print statements from a flow\")\n    my_task()\n

    Will output:

    15:52:11.244 | INFO    | prefect.engine - Created flow run 'emerald-gharial' for flow 'my-flow'\n15:52:11.812 | INFO    | Flow run 'emerald-gharial' - we're logging print statements from a flow\n15:52:11.926 | INFO    | Flow run 'emerald-gharial' - Created task run 'my_task-20c6ece6-0' for task 'my_task'\n15:52:11.927 | INFO    | Flow run 'emerald-gharial' - Executing 'my_task-20c6ece6-0' immediately...\n15:52:12.217 | INFO    | Task run 'my_task-20c6ece6-0' - we're logging print statements from a task\n
    from prefect import task, flow\n\n@task\ndef my_task(log_prints=False):\n    print(\"not logging print statements in this task\")\n\n@flow(log_prints=True)\ndef my_flow():\n    print(\"we're logging print statements from a flow\")\n    my_task()\n

    Using log_prints=False at the task level will output:

    15:52:11.244 | INFO    | prefect.engine - Created flow run 'emerald-gharial' for flow 'my-flow'\n15:52:11.812 | INFO    | Flow run 'emerald-gharial' - we're logging print statements from a flow\n15:52:11.926 | INFO    | Flow run 'emerald-gharial' - Created task run 'my_task-20c6ece6-0' for task 'my_task'\n15:52:11.927 | INFO    | Flow run 'emerald-gharial' - Executing 'my_task-20c6ece6-0' immediately...\nnot logging print statements in this task\n

    You can also configure this behavior globally for all Prefect flows, tasks, and subflows.

    prefect config set PREFECT_LOGGING_LOG_PRINTS=True\n
    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#formatters","title":"Formatters","text":"

    Prefect log formatters specify the format of log messages. You can see details of message formatting for different loggers in logging.yml. For example, the default formatting for task run log records is:

    \"%(asctime)s.%(msecs)03d | %(levelname)-7s | Task run %(task_run_name)r - %(message)s\"\n

    The variables available to interpolate in log messages varies by logger. In addition to the run context, message string, and any keyword arguments, flow and task run loggers have access to additional variables.

    The flow run logger has the following:

    • flow_run_name
    • flow_run_id
    • flow_name

    The task run logger has the following:

    • task_run_id
    • flow_run_id
    • task_run_name
    • task_name
    • flow_run_name
    • flow_name

    You can specify custom formatting by setting an environment variable or by modifying the formatter in a logging.yml file as described earlier. For example, to change the formatting for the flow runs formatter:

    PREFECT_LOGGING_FORMATTERS_STANDARD_FLOW_RUN_FMT=\"%(asctime)s.%(msecs)03d | %(levelname)-7s | %(flow_run_id)s - %(message)s\"\n

    The resulting messages, using the flow run ID instead of name, would look like this:

    10:40:01.211 | INFO    | e43a5a80-417a-41c4-a39e-2ef7421ee1fc - Created task run\n'othertask-1c085beb-3' for task 'othertask'\n
    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#styles","title":"Styles","text":"

    By default, Prefect highlights specific keywords in the console logs with a variety of colors.

    Highlighting can be toggled on/off with the PREFECT_LOGGING_COLORS setting, e.g.

    PREFECT_LOGGING_COLORS=False\n

    You can change what gets highlighted and also adjust the colors by updating the styles in a logging.yml file. Below lists the specific keys built-in to the PrefectConsoleHighlighter.

    URLs:

    • log.web_url
    • log.local_url

    Log levels:

    • log.info_level
    • log.warning_level
    • log.error_level
    • log.critical_level

    State types:

    • log.pending_state
    • log.running_state
    • log.scheduled_state
    • log.completed_state
    • log.cancelled_state
    • log.failed_state
    • log.crashed_state

    Flow (run) names:

    • log.flow_run_name
    • log.flow_name

    Task (run) names:

    • log.task_run_name
    • log.task_name

    You can also build your own handler with a custom highlighter. For example, to additionally highlight emails:

    1. Copy and paste the following into my_package_or_module.py (rename as needed) in the same directory as the flow run script, or ideally part of a Python package so it's available in site-packages to be accessed anywhere within your environment.
    import logging\nfrom typing import Dict, Union\n\nfrom rich.highlighter import Highlighter\n\nfrom prefect.logging.handlers import PrefectConsoleHandler\nfrom prefect.logging.highlighters import PrefectConsoleHighlighter\n\nclass CustomConsoleHighlighter(PrefectConsoleHighlighter):\n    base_style = \"log.\"\n    highlights = PrefectConsoleHighlighter.highlights + [\n        # ?P<email> is naming this expression as `email`\n        r\"(?P<email>[\\w-]+@([\\w-]+\\.)+[\\w-]+)\",\n    ]\n\nclass CustomConsoleHandler(PrefectConsoleHandler):\n    def __init__(\n        self,\n        highlighter: Highlighter = CustomConsoleHighlighter,\n        styles: Dict[str, str] = None,\n        level: Union[int, str] = logging.NOTSET,\n   ):\n        super().__init__(highlighter=highlighter, styles=styles, level=level)\n
    1. Update /.prefect/logging.yml to use my_package_or_module.CustomConsoleHandler and additionally reference the base_style and named expression: log.email.
        console_flow_runs:\n        level: 0\n        class: my_package_or_module.CustomConsoleHandler\n        formatter: flow_runs\n        styles:\n            log.email: magenta\n            # other styles can be appended here, e.g.\n            # log.completed_state: green\n
    1. Then on your next flow run, text that looks like an email will be highlighted--e.g. my@email.com is colored in magenta here.
    from prefect import flow, get_run_logger\n\n@flow\ndef log_email_flow():\n    logger = get_run_logger()\n    logger.info(\"my@email.com\")\n\nlog_email_flow()\n
    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#applying-markup-in-logs","title":"Applying markup in logs","text":"

    To use Rich's markup in Prefect logs, first configure PREFECT_LOGGING_MARKUP.

    PREFECT_LOGGING_MARKUP=True\n

    Then, the following will highlight \"fancy\" in red.

    from prefect import flow, get_run_logger\n\n@flow\ndef my_flow():\n    logger = get_run_logger()\n    logger.info(\"This is [bold red]fancy[/]\")\n\nmy_flow()\n

    Inaccurate logs could result

    Although this can be convenient, the downside is, if enabled, strings that contain square brackets may be inaccurately interpreted and lead to incomplete output, e.g. DROP TABLE [dbo].[SomeTable];\" outputs DROP TABLE .[SomeTable];.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#log-database-schema","title":"Log database schema","text":"

    Logged events are also persisted to the Prefect database. A log record includes the following data:

    Column Description id Primary key ID of the log record. created Timestamp specifying when the record was created. updated Timestamp specifying when the record was updated. name String specifying the name of the logger. level Integer representation of the logging level. flow_run_id ID of the flow run associated with the log record. If the log record is for a task run, this is the parent flow of the task. task_run_id ID of the task run associated with the log record. Null if logging a flow run event. message Log message. timestamp The client-side timestamp of this logged statement.

    For more information, see Log schema in the API documentation.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#including-logs-from-other-libraries","title":"Including logs from other libraries","text":"

    By default, Prefect won't capture log statements from libraries that your flows and tasks use. You can tell Prefect to include logs from these libraries with the PREFECT_LOGGING_EXTRA_LOGGERS setting.

    To use this setting, specify one or more Python library names to include, separated by commas. For example, if you want to make sure Prefect captures Dask and SciPy logging statements with your flow and task run logs:

    PREFECT_LOGGING_EXTRA_LOGGERS=dask,scipy\n

    You can set this setting as an environment variable or in a profile. See Settings for more details about how to use settings.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/managed-execution/","title":"Managed Execution","text":"

    Prefect Cloud can run your flows on your behalf with prefect:managed work pools. Flows run with this work pool do not require a worker or cloud provider account. Prefect handles the infrastructure and code execution for you.

    Managed execution is a great option for users who want to get started quickly, with no infrastructure setup.

    Managed Execution is in beta

    Managed Execution is currently in beta. Features are likely to change without warning.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#usage-guide","title":"Usage guide","text":"

    Run a flow with managed infrastructure in three steps.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#step-1","title":"Step 1","text":"

    Create a new work pool of type prefect:managed. you can do this via the UI wizard, or via the CLI

    prefect work-pool create my-managed-pool --type prefect:managed\n
    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#step-2","title":"Step 2","text":"

    Create a deployment using the flow deploy method or prefect.yaml.

    Specify the name of your managed work pool, as shown in this example that uses the deploy method:

    managed-execution.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n    source=\"https://github.com/desertaxle/demo.git\",\n    entrypoint=\"flow.py:my_flow\",\n    ).deploy(\n        name=\"test-managed-flow\",\n        work_pool_name=\"my-managed-pool\",\n    )\n

    With your CLI authenticated to your Prefect Cloud workspace, run the script to create your deployment:

    python managed-execution.py\n

    Note that this deployment uses flow code stored in a GitHub repository.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#step-3","title":"Step 3","text":"

    Run the deployment from the UI or from the CLI.

    That's it! You ran a flow on remote infrastructure without any infrastructure setup, worker, or cloud provider account.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#adding-dependencies","title":"Adding dependencies","text":"

    You can install Python package dependencies at runtime by configuring job_variables={\"pip_packages\": [\"pandas\", \"prefect-aws\"]} like this:

    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n    source=\"https://github.com/desertaxle/demo.git\",\n    entrypoint=\"flow.py:my_flow\",\n    ).deploy(\n        name=\"test-managed-flow\",\n        work_pool_name=\"my-managed-pool\",\n        job_variables={\"pip_packages\": [\"pandas\", \"prefect-aws\"]}\n    )\n

    Alternatively, you can specify a requirements.txt file and reference it in your prefect.yaml pull_step.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#limitations","title":"Limitations","text":"

    Managed execution requires Prefect 2.14.4 or newer.

    All limitations listed below may change without warning during the beta period. We will update this page as we make changes.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#concurrency-work-pools","title":"Concurrency & work pools","text":"

    Free tier accounts are limited to:

    • Maximum of 1 concurrent flow run per workspace across all prefect:managed pools.
    • Maximum of 1 managed execution work pool per workspace.

    Pro tier and above accounts are limited to:

    • Maximum of 10 concurrent flow runs per workspace across all prefect:managed pools.
    • Maximum of 5 managed execution work pools per workspace.
    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#images","title":"Images","text":"

    At this time, managed execution requires that you run the official Prefect Docker image: prefecthq/prefect:2-latest. However, as noted above, you can install Python package dependencies at runtime. If you need to use your own image, we recommend using another type of work pool.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#code-storage","title":"Code storage","text":"

    Flow code must be stored in an accessible remote location. This means git-based cloud providers such as GitHub, Bitbucket, or GitLab are supported. Remote block-based storage is also supported, so S3, GCS, and Azure Blob are additional code storage options.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#resources","title":"Resources","text":"

    Memory is limited to 2GB of RAM, which includes all operations such as dependency installation. Maximum job run time is 24 hours.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#usage-limits","title":"Usage limits","text":"

    Free tier accounts are limited to ten compute hours per workspace per month. Pro tier and above accounts are limited to 250 hours per workspace per month. you can view your compute hours quota usage on the work pools page.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#next-steps","title":"Next steps","text":"

    Read more about creating deployments in the deployment guide.

    If you find that you need more control over your infrastructure, such as the ability to run custom Docker images, serverless push work pools might be a good option. Read more here.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/migration-guide/","title":"Migrating from Prefect 1 to Prefect 2","text":"

    This guide is designed to help you migrate your workflows from Prefect 1 to Prefect 2.

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#what-stayed-the-same","title":"What stayed the same","text":"

    Prefect 2 still:

    • Has tasks and flows.
    • Orchestrates your flow runs and provides observability into their execution states.
    • Runs and inspects flow runs locally.
    • Provides a coordination plane for your dataflows based on the same principles.
    • Employs the same hybrid execution model, where Prefect doesn't store your flow code or data.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#what-changed","title":"What changed","text":"

    Prefect 2 requires modifications to your existing tasks, flows, and deployment patterns. We've organized this section into the following categories:

    • Simplified patterns \u2014 abstractions from Prefect 1 that are no longer necessary in the dynamic, DAG-free Prefect workflows that support running native Python code in your flows.
    • Conceptual and syntax changes that often clarify names and simplify familiar abstractions such as retries and caching.
    • New features enabled by the dynamic and flexible Prefect API.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#simplified-patterns","title":"Simplified patterns","text":"

    Since Prefect 2 allows running native Python code within the flow function, some abstractions are no longer necessary:

    • Parameter tasks: in Prefect 2, inputs to your flow function are automatically treated as parameters of your flow. You can define the parameter values in your flow code when you create your Deployment, or when you schedule an ad-hoc flow run. One benefit of Prefect parametrization is built-in type validation with pydantic.
    • Task-level state_handlers: in Prefect 2, you can build custom logic that reacts to task-run states within your flow function without the need for state_handlers. The page \" How to take action on a state change of a task run\" provides a further explanation and code examples.
    • Instead of using signals, Prefect 2 allows you to raise an arbitrary exception in your task or flow and return a custom state. For more details and examples, see How can I stop the task run based on a custom logic.
    • Conditional tasks such as case are no longer required. Use Python native if...else statements to build a conditional logic. The Discourse tag \"conditional-logic\" provides more resources.
    • Since you can use any context manager directly in your flow, a resource_manager is no longer necessary. As long as you point to your flow script in your Deployment, you can share database connections and any other resources between tasks in your flow. The Discourse page How to clean up resources used in a flow provides a full example.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#conceptual-and-syntax-changes","title":"Conceptual and syntax changes","text":"

    The changes listed below require you to modify your workflow code. The following table shows how Prefect 1 concepts have been implemented in Prefect 2. The last column contains references to additional resources that provide more details and examples.

    Concept Prefect 1 Prefect 2 Reference links Flow definition. with Flow(\"flow_name\") as flow: @flow(name=\"flow_name\") How can I define a flow? Flow executor that determines how to execute your task runs. Executor such as LocalExecutor. Task runner such as ConcurrentTaskRunner. What is the default TaskRunner (executor)? Configuration that determines how and where to execute your flow runs. Run configuration such as flow.run_config = DockerRun(). Create an infrastructure block such as a Docker Container and specify it as the infrastructure when creating a deployment. How can I run my flow in a Docker container? Assignment of schedules and default parameter values. Schedules are attached to the flow object and default parameter values are defined within the Parameter tasks. Schedules and default parameters are assigned to a flow\u2019s Deployment, rather than to a Flow object. How can I attach a schedule to a flow? Retries @task(max_retries=2, retry_delay=timedelta(seconds=5)) @task(retries=2, retry_delay_seconds=5) How can I specify the retry behavior for a specific task? Logger syntax. Logger is retrieved from prefect.context and can only be used within tasks. In Prefect 2, you can log not only from tasks, but also within flows. To get the logger object, use: prefect.get_run_logger(). How can I add logs to my flow? The syntax and contents of Prefect context. Context is a thread-safe way of accessing variables related to the flow run and task run. The syntax to retrieve it: prefect.context. Context is still available, but its content is much richer, allowing you to retrieve even more information about your flow runs and task runs. The syntax to retrieve it: prefect.context.get_run_context(). How to access Prefect context values? Task library. Included in the main Prefect Core repository. Separated into individual repositories per system, cloud provider, or technology. How to migrate Prefect 1 tasks to Prefect 2 integrations.","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#what-changed-in-dataflow-orchestration","title":"What changed in dataflow orchestration?","text":"

    Let\u2019s look at the differences in how Prefect 2 transitions your flow and task runs between various execution states.

    • In Prefect 2, the final state of a flow run that finished without errors is Completed, while in Prefect 1, this flow run has a Success state. You can find more about that topic here.
    • The decision about whether a flow run should be considered successful or not is no longer based on special reference tasks. Instead, your flow\u2019s return value determines the final state of a flow run. This link provides a more detailed explanation with code examples.
    • In Prefect 1, concurrency limits were only available to Prefect Cloud users. Prefect 2 provides customizable concurrency limits with the open-source Prefect server and Prefect Cloud. In Prefect 2, flow run concurrency limits are set on work pools.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#what-changed-in-flow-deployment-patterns","title":"What changed in flow deployment patterns?","text":"

    To deploy your Prefect 1 flows, you have to send flow metadata to the backend in a step called registration. Prefect 2 no longer requires flow pre-registration. Instead, you create a Deployment that specifies the entry point to your flow code and optionally specifies:

    • Where to run your flow (your Infrastructure, such as a DockerContainer, KubernetesJob, or ECSTask).
    • When to run your flow (an Interval, Cron, or RRule schedule).
    • How to run your flow (execution details such as parameters, flow deployment name, and more).
    • The work pool for your deployment. If no work pool is specified, a default work pool named default is used.

    The API is now implemented as a REST API rather than GraphQL. This page illustrates how you can interact with the API.

    In Prefect 1, the logical grouping of flows was based on projects. Prefect 2 provides a much more flexible way of organizing your flows, tasks, and deployments through customizable filters and\u00a0tags. This page provides more details on how to assign tags to various Prefect 2 objects.

    The role of agents has changed:

    • In Prefect 2, there is only one generic agent type. The agent polls a work pool looking for flow runs.
    • See this Discourse page for a more detailed discussion.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#new-features-introduced-in-prefect-2","title":"New features introduced in Prefect 2","text":"

    The following new components and capabilities are enabled by Prefect 2.

    • More flexibility thanks to the elimination of flow pre-registration.
    • More flexibility for flow deployments, including easier promotion of a flow through development, staging, and production environments.
    • Native async support.
    • Out-of-the-box pydantic validation.
    • Blocks allowing you to securely store UI-editable, type-checked configuration to external systems and an easy-to-use Key-Value Store. All those components are configurable in one place and provided as part of the open-source Prefect 2 product. In contrast, the concept of Secrets in Prefect 1 was much more narrow and only available in Prefect Cloud.
    • Notifications available in the open-source Prefect 2 version, as opposed to Cloud-only Automations in Prefect 1.
    • A first-class subflows concept: Prefect 1 only allowed the flow-of-flows orchestrator pattern. With Prefect 2 subflows, you gain a natural and intuitive way of organizing your flows into modular sub-components. For more details, see the following list of resources about subflows.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#orchestration-behind-the-api","title":"Orchestration behind the API","text":"

    Apart from new features, Prefect 2 simplifies many usage patterns and provides a much more seamless onboarding experience.

    Every time you run a flow, whether it is tracked by the API server or ad-hoc through a Python script, it is on the same UI page for easier debugging and observability.

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#code-as-workflows","title":"Code as workflows","text":"

    With Prefect 2, your functions\u00a0are\u00a0your flows and tasks. Prefect 2 automatically detects your flows and tasks without the need to define a rigid DAG structure. While use of tasks is encouraged to provide you the maximum visibility into your workflows, they are no longer required. You can add a single @flow decorator to your main function to transform any Python script into a Prefect workflow.

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#incremental-adoption","title":"Incremental adoption","text":"

    The built-in SQLite database automatically tracks all your locally executed flow runs. As soon as you start a Prefect server and open the Prefect UI in your browser (or authenticate your CLI with your Prefect Cloud workspace), you can see all your locally executed flow runs in the UI. You don't even need to start an agent.

    Then, when you want to move toward scheduled, repeatable workflows, you can build a deployment and send it to the server by running a CLI command or a Python script.

    • You can create a deployment to on remote infrastructure, where the run environment is defined by a reusable infrastructure block.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#fewer-ambiguities","title":"Fewer ambiguities","text":"

    Prefect 2 eliminates ambiguities in many ways. For example. there is no more confusion between Prefect Core and Prefect Server \u2014 Prefect 2 unifies those into a single open source product. This product is also much easier to deploy with no requirement for Docker or docker-compose.

    If you want to switch your backend to use Prefect Cloud for an easier production-level managed experience, Prefect profiles let you quickly connect to your workspace.

    In Prefect 1, there are several confusing ways you could implement caching. Prefect 2 resolves those ambiguities by providing a single cache_key_fn function paired with cache_expiration, allowing you to define arbitrary caching mechanisms \u2014 no more confusion about whether you need to use cache_for, cache_validator, or file-based caching using targets.

    For more details on how to configure caching, check out the following resources:

    • Caching docs
    • Time-based caching
    • Input-based caching

    A similarly confusing concept in Prefect 1 was distinguishing between the functional and imperative APIs. This distinction caused ambiguities with respect to how to define state dependencies between tasks. Prefect 1 users were often unsure whether they should use the functional upstream_tasks keyword argument or the imperative methods such as task.set_upstream(), task.set_downstream(), or flow.set_dependencies(). In Prefect 2, there is only the functional API.

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#next-steps","title":"Next steps","text":"

    We know migrations can be tough. We encourage you to take it step-by-step and experiment with the new features.

    To make the migration process easier for you:

    • We provided a detailed FAQ section allowing you to find the right information you need to move your workflows to Prefect 2. If you still have some open questions, feel free to create a new topic describing your migration issue.
    • We have dedicated resources in the Customer Success team to help you along your migration journey. Reach out to cs@prefect.io to discuss how we can help.
    • You can ask questions in our 20,000+ member Community Slack.

    Happy Engineering!

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/moving-data/","title":"Read and Write Data to and From Cloud Provider Storage","text":"

    Writing data to cloud-based storage and reading data from that storage is a common task in data engineering. In this guide we'll learn how to use Prefect to move data to and from AWS, Azure, and GCP blob storage.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#prerequisites","title":"Prerequisites","text":"
    • Prefect installed
    • Authenticated with Prefect Cloud (or self-hosted Prefect server instance)
    • A cloud provider account (e.g. AWS)
    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#install-relevant-prefect-integration-library","title":"Install relevant Prefect integration library","text":"

    In the CLI, install the Prefect integration library for your cloud provider:

    AWSAzureGCP

    prefect-aws provides blocks for interacting with AWS services.

    pip install -U prefect-aws\n

    prefect-azure provides blocks for interacting with Azure services.

     pip install -U prefect-azure\n

    prefect-gcp provides blocks for interacting with GCP services.

     pip install -U prefect-gcp\n
    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#register-the-block-types","title":"Register the block types","text":"

    Register the new block types with Prefect Cloud (or with your self-hosted Prefect server instance):

    AWSAzureGCP
    prefect block register -m prefect_aws  \n
    prefect block register -m prefect_azure \n
    prefect block register -m prefect_gcp\n

    We should see a message in the CLI that several block types were registered. If we check the UI, we should see the new block types listed.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#create-a-storage-bucket","title":"Create a storage bucket","text":"

    Create a storage bucket in the cloud provider account. Ensure the bucket is publicly accessible or create a user or service account with the appropriate permissions to fetch and write data to the bucket.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#create-a-credentials-block","title":"Create a credentials block","text":"

    If the bucket is private, there are several options to authenticate:

    1. At deployment runtime, ensure the runtime environment is authenticated.
    2. Create a block with configuration details and reference it when creating the storage block.

    If saving credential details in a block we can use a credentials block specific to the cloud provider or use a more generic secret block. We can create blocks via the UI or Python code. Below we'll use Python code to create a credentials block for our cloud provider.

    Credentials safety

    Reminder, don't store credential values in public locations such as public git platform repositories. In the examples below we use environment variables to store credential values.

    AWSAzureGCP
    import os\nfrom prefect_aws import AwsCredentials\n\nmy_aws_creds = AwsCredentials(\n    aws_access_key_id=\"123abc\",\n    aws_secret_access_key=os.environ.get(\"MY_AWS_SECRET_ACCESS_KEY\"),\n)\nmy_aws_creds.save(name=\"my-aws-creds-block\", overwrite=True)\n
    import os\nfrom prefect_azure import AzureBlobStorageCredentials\n\nmy_azure_creds = AzureBlobStorageCredentials(\n    connection_string=os.environ.get(\"MY_AZURE_CONNECTION_STRING\"),\n)\nmy_azure_creds.save(name=\"my-azure-creds-block\", overwrite=True)\n

    We recommend specifying the service account key file contents as a string, rather than the path to the file, because that file might not be available in your production environments.

    import os\nfrom prefect_gcp import GCPCredentials\n\nmy_gcp_creds = GCPCredentials(\n    service_account_info=os.environ.get(\"GCP_SERVICE_ACCOUNT_KEY_FILE_CONTENTS\"), \n)\nmy_gcp_creds.save(name=\"my-gcp-creds-block\", overwrite=True)\n

    Run the code to create the block. We should see a message that the block was created.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#create-a-storage-block","title":"Create a storage block","text":"

    Let's create a block for the chosen cloud provider using Python code or the UI. In this example we'll use Python code.

    AWSAzureGCP

    Note that the S3Bucket block is not the same as the S3 block that ships with Prefect. The S3Bucket block we use in this example is part of the prefect-aws library and provides additional functionality.

    We'll reference the credentials block created above.

    from prefect_aws import S3Bucket\n\ns3bucket = S3Bucket.create(\n    bucket=\"my-bucket-name\",\n    credentials=\"my-aws-creds-block\"\n    )\ns3bucket.save(name=\"my-s3-bucket-block\", overwrite=True)\n

    Note that the AzureBlobStorageCredentials block is not the same as the Azure block that ships with Prefect. The AzureBlobStorageCredentials block we use in this example is part of the prefect-azure library and provides additional functionality.

    Azure blob storage doesn't require a separate block, the connection string used in the AzureBlobStorageCredentials block can encode the information needed.

    Note that the GcsBucket block is not the same as the GCS block that ships with Prefect. The GcsBucket block is part of the prefect-gcp library and provides additional functionality. We'll use it here.

    We'll reference the credentials block created above.

    from prefect_gcp.cloud_storage import GcsBucket\n\ngcsbucket = GcsBucket(\n    bucket=\"my-bucket-name\", \n    credentials=\"my-gcp-creds-block\"\n    )\ngcsbucket.save(name=\"my-gcs-bucket-block\", overwrite=True)\n

    Run the code to create the block. We should see a message that the block was created.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#write-data","title":"Write data","text":"

    Use your new block inside a flow to write data to your cloud provider.

    AWSAzureGCP
    from pathlib import Path\nfrom prefect import flow\nfrom prefect_aws.s3 import S3Bucket\n\n@flow()\ndef upload_to_s3():\n    \"\"\"Flow function to upload data\"\"\"\n    path = Path(\"my_path_to/my_file.parquet\")\n    aws_block = S3Bucket.load(\"my-s3-bucket-block\")\n    aws_block.upload_from_path(from_path=path, to_path=path)\n\nif __name__ == \"__main__\":\n    upload_to_s3()\n
    from prefect import flow\nfrom prefect_azure import AzureBlobStorageCredentials\nfrom prefect_azure.blob_storage import blob_storage_upload\n\n@flow\ndef upload_to_azure():\n    \"\"\"Flow function to upload data\"\"\"\n    blob_storage_credentials = AzureBlobStorageCredentials.load(\n        name=\"my-azure-creds-block\"\n    )\n\n    with open(\"my_path_to/my_file.parquet\", \"rb\") as f:\n        blob_storage_upload(\n            data=f.read(),\n            container=\"my_container\",\n            blob=\"my_path_to/my_file.parquet\",\n            blob_storage_credentials=blob_storage_credentials,\n        )\n\nif __name__ == \"__main__\":\n    upload_to_azure()\n
    from pathlib import Path\nfrom prefect import flow\nfrom prefect_gcp.cloud_storage import GcsBucket\n\n@flow()\ndef upload_to_gcs():\n    \"\"\"Flow function to upload data\"\"\"\n    path = Path(\"my_path_to/my_file.parquet\")\n    gcs_block = GcsBucket.load(\"my-gcs-bucket-block\")\n    gcs_block.upload_from_path(from_path=path, to_path=path)\n\nif __name__ == \"__main__\":\n    upload_to_gcs()\n
    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#read-data","title":"Read data","text":"

    Use your block to read data from your cloud provider inside a flow.

    AWSAzureGCP
    from prefect import flow\nfrom prefect_aws import S3Bucket\n\n@flow\ndef download_from_s3():\n    \"\"\"Flow function to download data\"\"\"\n    s3_block = S3Bucket.load(\"my-s3-bucket-block\")\n    s3_block.get_directory(\n        from_path=\"my_path_to/my_file.parquet\", \n        local_path=\"my_path_to/my_file.parquet\"\n    )\n\nif __name__ == \"__main__\":\n    download_from_s3()\n
    from prefect import flow\nfrom prefect_azure import AzureBlobStorageCredentials\nfrom prefect_azure.blob_storage import blob_storage_download\n\n@flow\ndef download_from_azure():\n    \"\"\"Flow function to download data\"\"\"\n    blob_storage_credentials = AzureBlobStorageCredentials.load(\n        name=\"my-azure-creds-block\"\n    )\n    blob_storage_download(\n        blob=\"my_path_to/my_file.parquet\",\n        container=\"my_container\",\n        blob_storage_credentials=blob_storage_credentials,\n    )\n\nif __name__ == \"__main__\":\n    download_from_azure()\n
    from prefect import flow\nfrom prefect_gcp.cloud_storage import GcsBucket\n\n@flow\ndef download_from_gcs():\n    gcs_block = GcsBucket.load(\"my-gcs-bucket-block\")\n    gcs_block.get_directory(\n        from_path=\"my_path_to/my_file.parquet\", \n        local_path=\"my_path_to/my_file.parquet\"\n    )\n\nif __name__ == \"__main__\":\n    download_from_gcs()\n

    In this guide we've seen how to use Prefect to read data from and write data to cloud providers!

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#next-steps","title":"Next steps","text":"

    Check out the prefect-aws, prefect-azure, and prefect-gcp docs to see additional methods for interacting with cloud storage providers. Each library also contains blocks for interacting with other cloud-provider services.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/prefect-deploy/","title":"Deploying Flows to Work Pools and Workers","text":"

    In this guide, we will configure a deployment that uses a work pool for dynamically provisioned infrastructure.

    All Prefect flow runs are tracked by the API. The API does not require prior registration of flows. With Prefect, you can call a flow locally or on a remote environment and it will be tracked.

    A deployment turns your workflow into an application that can be interacted with and managed via the Prefect API. A deployment enables you to:

    • Schedule flow runs.
    • Specify event triggers for flow runs.
    • Assign one or more tags to organize your deployments and flow runs. You can use those tags as filters in the Prefect UI.
    • Assign custom parameter values for flow runs based on the deployment.
    • Create ad-hoc flow runs from the API or Prefect UI.
    • Upload flow files to a defined storage location for retrieval at run time.

    Deployments created with .serve

    A deployment created with the Python flow.serve method or the serve function runs flows in a subprocess on the same machine where the deployment is created. It does not use a work pool or worker.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#work-pool-based-deployments","title":"Work pool-based deployments","text":"

    A work pool-based deployment is useful when you want to dynamically scale the infrastructure where your flow code runs. Work pool-based deployments contain information about the infrastructure type and configuration for your workflow execution.

    Work pool-based deployment infrastructure options include the following:

    • Process - runs flow in a subprocess. In most cases, you're better off using .serve.
    • Docker - runs flows in an ephemeral Docker container.
    • Kubernetes - runs flows as a Kubernetes Job.
    • Serverless Cloud Provider options - runs flows in a Docker container in a serverless cloud provider environment, such as AWS ECS, Azure Container Instance, Google Cloud Run, or Vertex AI.

    The following diagram provides a high-level overview of the conceptual elements involved in defining a work-pool based deployment that is polled by a worker and executes a flow run based on that deployment.

    %%{\n  init: {\n    'theme': 'base',\n    'themeVariables': {\n      'fontSize': '19px'\n    }\n  }\n}%%\n\nflowchart LR\n    F(\"<div style='margin: 5px 10px 5px 5px;'>Flow Code</div>\"):::yellow -.-> A(\"<div style='margin: 5px 10px 5px 5px;'>Deployment Definition</div>\"):::gold\n    subgraph Server [\"<div style='width: 150px; text-align: center; margin-top: 5px;'>Prefect API</div>\"]\n        D(\"<div style='margin: 5px 10px 5px 5px;'>Deployment</div>\"):::green\n    end\n    subgraph Remote Storage [\"<div style='width: 160px; text-align: center; margin-top: 5px;'>Remote Storage</div>\"]\n        B(\"<div style='margin: 5px 6px 5px 5px;'>Flow</div>\"):::yellow\n    end\n    subgraph Infrastructure [\"<div style='width: 150px; text-align: center; margin-top: 5px;'>Infrastructure</div>\"]\n        G(\"<div style='margin: 5px 10px 5px 5px;'>Flow Run</div>\"):::blue\n    end\n\n    A --> D\n    D --> E(\"<div style='margin: 5px 10px 5px 5px;'>Worker</div>\"):::red\n    B -.-> E\n    A -.-> B\n    E -.-> G\n\n    classDef gold fill:goldenrod,stroke:goldenrod,stroke-width:4px,color:black\n    classDef yellow fill:gold,stroke:gold,stroke-width:4px,color:black\n    classDef gray fill:lightgray,stroke:lightgray,stroke-width:4px\n    classDef blue fill:blue,stroke:blue,stroke-width:4px,color:white\n    classDef green fill:green,stroke:green,stroke-width:4px,color:white\n    classDef red fill:red,stroke:red,stroke-width:4px,color:white\n    classDef dkgray fill:darkgray,stroke:darkgray,stroke-width:4px,color:white

    The work pool types above require a worker to be running on your infrastructure to poll a work pool for scheduled flow runs.

    Additional work pool options available with Prefect Cloud

    Prefect Cloud offers other flavors of work pools that don't require a worker:

    • Push Work Pools - serverless cloud options that don't require a worker because Prefect Cloud submits them to your serverless cloud infrastructure on your behalf. Prefect can auto-provision your cloud infrastructure for you and set it up to use your work pool.

    • Managed Execution Prefect Cloud submits and runs your deployment on serverless infrastructure. No cloud provider account required.

    In this guide, we focus on deployments that require a worker.

    Work pool-based deployments that use a worker also allow you to assign a work queue name to prioritize work and allow you to limit concurrent runs at the work pool level.

    When creating a deployment that uses a work pool and worker, we must answer two basic questions:

    • What instructions does a worker need to set up an execution environment for our flow? For example, a flow may have Python package requirements, unique Kubernetes settings, or Docker networking configuration.
    • How should the flow code be accessed?
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#creating-work-pool-based-deployments","title":"Creating work pool-based deployments","text":"

    The tutorial shows how you can create a deployment with a long-running process using .serve and how to move to a work-pool-based deployment setup with .deploy. See the discussion of when you might want to move to work-pool-based deployments there.

    In this guide, we show how to use .deploy in more depth and discuss prefect.yaml, a YAML-based alternative for managing deployments.

    Use the tabs below to explore these two deployment creation options.

    .deployprefect.yaml

    The prefect.yaml file is a YAML file describing base settings for your deployments, procedural steps for preparing deployments, and instructions for preparing the execution environment for a deployment run.

    You can initialize your deployment configuration, which creates the prefect.yaml file, by running the CLI command prefect init in any directory or repository that stores your flow code.

    Deployment configuration recipes

    Prefect ships with many off-the-shelf \"recipes\" that allow you to get started with more structure within your prefect.yaml file; run prefect init to be prompted with available recipes in your installation. You can provide a recipe name in your initialization command with the --recipe flag, otherwise Prefect will attempt to guess an appropriate recipe based on the structure of your working directory (for example if you initialize within a git repository, Prefect will use the git recipe).

    The prefect.yaml file contains deployment configuration for deployments created from this file, default instructions for how to build and push any necessary code artifacts (such as Docker images), and default instructions for pulling a deployment in remote execution environments (e.g., cloning a GitHub repository).

    Any deployment configuration can be overridden via options available on the prefect deploy CLI command when creating a deployment.

    The base structure for prefect.yaml is as follows:

    # generic metadata\nprefect-version: null\nname: null\n\n# preparation steps\nbuild: null\npush: null\n\n# runtime steps\npull: null\n\n# deployment configurations\ndeployments:\n- # base metadata\n    name: null\n    version: null\n    tags: []\n    description: null\n    schedule: null\n\n    # flow-specific fields\n    entrypoint: null\n    parameters: {}\n\n    # infra-specific fields\n    work_pool:\n    name: null\n    work_queue_name: null\n    job_variables: {}\n

    The metadata fields are always pre-populated for you. These fields are for bookkeeping purposes only. The other sections are pre-populated based on recipe; if no recipe is provided, Prefect will attempt to guess an appropriate one based on local configuration.

    You can create deployments via the CLI command prefect deploy without ever needing to alter the deployments section of your prefect.yaml file \u2014 the prefect deploy command will help in deployment creation via interactive prompts. The prefect.yaml file facilitates version-controlling your deployment configuration and managing multiple deployments.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#automatically-bake-your-code-into-a-docker-image","title":"Automatically bake your code into a Docker image","text":"

    You can create a deployment from Python code by calling the .deploy method on a flow.

    buy.py
    from prefect import flow\n\n\n@flow(log_prints=True)\ndef buy():\n    print(\"Buying securities\")\n\n\nif __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-code-baked-into-an-image-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"my_registry/my_image:my_image_tag\"\n    )\n

    Make sure you have the work pool created in the Prefect Cloud workspace you are authenticated to or on your running self-hosted server instance. Then run the script to create a deployment (in future examples this step will be omitted for brevity):

    python buy.py\n

    You should see messages in your terminal that Docker is building your image. When the deployment build succeeds you will see helpful information in your terminal showing you how to start a worker for your deployment and how to run your deployment. Your deployment will be visible on the Deployments page in the UI.

    By default, .deploy will build a Docker image with your flow code baked into it and push the image to the Docker Hub registry specified in the image argument`.

    Authentication to Docker Hub

    You need your environment to be authenticated to your Docker registry to push an image to it.

    You can specify a registry other than Docker Hub by providing the full registry path in the image argument.

    Warning

    If building a Docker image, the environment in which you are creating the deployment needs to have Docker installed and running.

    To avoid pushing to a registry, set push=False in the .deploy method.

    if __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-code-baked-into-an-image-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"my_registry/my_image:my_image_tag\",\n        push=False\n    )\n

    To avoid building an image, set build=False in the .deploy method.

    if __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-code-baked-into-an-image-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"discdiver/no-build-image:1.0\",\n        build=False\n    )\n

    The specified image will need to be available in your deployment's execution environment for your flow code to be accessible.

    Prefect generates a Dockerfile for you that will build an image based off of one of Prefect's published images. The generated Dockerfile will copy the current directory into the Docker image and install any dependencies listed in a requirements.txt file.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#automatically-build-a-custom-docker-image-with-a-local-dockerfile","title":"Automatically build a custom Docker image with a local Dockerfile","text":"

    If you want to use a custom Dockerfile, you can specify the path to the Dockerfile with the DeploymentImage class:

    custom_dockerfile.py
    from prefect import flow\nfrom prefect.deployments import DeploymentImage\n\n\n@flow(log_prints=True)\ndef buy():\n    print(\"Selling securities\")\n\n\nif __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-custom-dockerfile-deployment\",\", \n        work_pool_name=\"my-docker-pool\", \n        image=DeploymentImage(\n            name=\"my_image\",\n            tag=\"deploy-guide\",\n            dockerfile=\"Dockerfile\"\n    ),\n    push=False\n)\n

    The DeploymentImage object allows for a great deal of image customization.

    For example, you can install a private Python package from GCP's artifact registry like this:

    Create a custom base Dockerfile.

    FROM python:3.10\n\nARG AUTHED_ARTIFACT_REG_URL\nCOPY ./requirements.txt /requirements.txt\n\nRUN pip install --extra-index-url ${AUTHED_ARTIFACT_REG_URL} -r /requirements.txt\n

    Create our deployment by leveraging the DeploymentImage class.

    private-package.py
    from prefect import flow\nfrom prefect.deployments.runner import DeploymentImage\nfrom prefect.blocks.system import Secret\nfrom my_private_package import do_something_cool\n\n\n@flow(log_prints=True)\ndef my_flow():\n    do_something_cool()\n\n\nif __name__ == \"__main__\":\n    artifact_reg_url: Secret = Secret.load(\"artifact-reg-url\")\n\n    my_flow.deploy(\n        name=\"my-deployment\",\n        work_pool_name=\"k8s-demo\",\n        image=DeploymentImage(\n            name=\"my-image\",\n            tag=\"test\",\n            dockerfile=\"Dockerfile\",\n            buildargs={\"AUTHED_ARTIFACT_REG_URL\": artifact_reg_url.get()},\n        ),\n    )\n

    Note that we used a Prefect Secret block to load the URL configuration for the artifact registry above.

    See all the optional keyword arguments for the DeploymentImage class here.

    Default Docker namespace

    You can set the PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE setting to append a default Docker namespace to all images you build with .deploy. This is great if you use a private registry to store your images.

    To set a default Docker namespace for your current profile run:

    prefect config set PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE=<docker-registry-url>/<organization-or-username>\n

    Once set, you can omit the namespace from your image name when creating a deployment:

    with_default_docker_namespace.py
    if __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-code-baked-into-an-image-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"my_image:my_image_tag\"\n    )\n

    The above code will build an image with the format <docker-registry-url>/<organization-or-username>/my_image:my_image_tag when PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE is set.

    While baking code into Docker images is a popular deployment option, many teams decide to store their workflow code in git-based storage, such as GitHub, Bitbucket, or Gitlab. Let's see how to do that next.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#store-your-code-in-git-based-cloud-storage","title":"Store your code in git-based cloud storage","text":"

    If you don't specify an image argument for .deploy, then you need to specify where to pull the flow code from at runtime with the from_source method.

    Here's how we can pull our flow code from a GitHub repository.

    git_storage.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        \"https://github.com/my_github_account/my_repo/my_file.git\",\n        entrypoint=\"flows/no-image.py:hello_world\",\n    ).deploy(\n        name=\"no-image-deployment\",\n        work_pool_name=\"my_pool\",\n        build=False\n    )\n

    The entrypoint is the path to the file the flow is located in and the function name, separated by a colon.

    Alternatively, you could specify a git-based cloud storage URL for a Bitbucket or Gitlab repository.

    Note

    If you don't specify an image as part of your deployment creation, the image specified in the work pool will be used to run your flow.

    After creating a deployment you might change your flow code. Generally, you can just push your code to GitHub, without rebuilding your deployment. The exception is if something that the server needs to know about changes, such as the flow entrypoint parameters. Rerunning the Python script with .deploy will update your deployment on the server with the new flow code.

    If you want to pull your flow code from private git-based storage,

    If you need to provide additional configuration, such as specifying a private repository, you can provide a GitRepository object instead of a URL:

    private_git_storage.py
    from prefect import flow\nfrom prefect.runner.storage import GitRepository\nfrom prefect.blocks.system import Secret\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=GitRepository(\n        url=\"https://github.com/org/private-repo.git\",\n        branch=\"dev\",\n        credentials={\n            \"access_token\": Secret.load(\"github-access-token\")\n        }\n    ),\n    entrypoint=\"flows/no-image.py:hello_world\",\n    ).deploy(\n        name=\"private-git-storage-deployment\",\n        work_pool_name=\"my_pool\",\n        build=False\n    )\n

    Note the use of the Secret block to load the GitHub access token. Alternatively, you could provide a username and password to the username and password fields of the credentials argument.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#store-your-code-in-cloud-provider-storage","title":"Store your code in cloud provider storage","text":"

    Another option for flow code storage is any fsspec-supported storage location, such as AWS S3, GCP GCS, or Azure Blob Storage.

    For example, you can pass the S3 bucket path to source.

    s3_storage.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=\"s3://my-bucket/my-folder\",\n        entrypoint=\"flows.py:my_flow\",\n    ).deploy(\n        name=\"deployment-from-aws-flow\",\n        work_pool=\"my_pool\",\n    )\n

    In the example above your credentials will be auto-discovered from your deployment creation environment and credentials will need to be available in your runtime environment.

    If you need additional configuration for your cloud-based storage - for example, with a private S3 Bucket - we recommend using a storage block. A storage block also ensures your credentials will be available in both your deployment creation environment and your execution environment.

    Here's an example that uses an S3Bucket block from the prefect-aws library.

    s3_storage_auth.py
    from prefect import flow\nfrom prefect_aws.s3 import S3Bucket\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=S3Bucket.load(\"my-code-storage\"), entrypoint=\"my_file.py:my_flow\"\n    ).deploy(name=\"test-s3\", work_pool=\"my_pool\")\n

    If you are familiar with the deployment creation mechanics with .serve, you will notice that .deploy is very similar. .deploy just requires a work pool name and has a number of parameters dealing with flow-code storage for Docker images.

    Unlike .serve, if you don't specify an image to use for your flow, you must to specify where to pull the flow code from at runtime with the from_source method, whereas from_source is optional with .serve.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#additional-configuration-with-deploy","title":"Additional configuration with .deploy","text":"

    Our examples thus far have explored options for where to store flow code. Let's turn our attention to other deployment configuration options.

    To pass parameters to your flow, you can use the parameters argument in the .deploy method. Just pass in a dictionary of key-value pairs.

    pass_params.py
    from prefect import flow\n\n@flow\ndef hello_world(name: str):\n    print(f\"Hello, {name}!\")\n\nif __name__ == \"__main__\":\n    hello_world.deploy(\n        name=\"pass-params-deployment\",\n        work_pool_name=\"my_pool\",\n        parameters=dict(name=\"Prefect\"),\n        image=\"my_registry/my_image:my_image_tag\",\n    )\n

    The job_variables parameter allows you to fine-tune the infrastructure settings for a deployment. The values passed in override default values in the specified work pool's base job template.

    You can override environment variables, such as image_pull_policy and image, for a specific deployment with the job_variables argument.

    job_var_image_pull.py
    if __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-deployment-never-pull\", \n        work_pool_name=\"my-docker-pool\", \n        job_variables={\"image_pull_policy\": \"Never\"},\n        image=\"my-image:my-tag\"\",\n        push=False\n    )\n

    Similarly, you can override the environment variables specified in a work pool through the job_variables parameter:

    job_var_env_vars.py
    if __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-deployment-never-pull\", \n        work_pool_name=\"my-docker-pool\", \n        job_variables={\"env\": {\"EXTRA_PIP_PACKAGES\": \"boto3\"} },\n        image=\"my-image:my-tag\"\",\n        push=False\n    )\n

    The dictionary key \"EXTRA_PIP_PACKAGES\" denotes a special environment variable that Prefect will use to install additional Python packages at runtime. This approach is an alternative to building an image with a custom requirements.txt copied into it.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-actions","title":"Deployment actions","text":"

    Deployment actions defined in your prefect.yaml file control the lifecycle of the creation and execution of your deployments. The three actions available are build, push, and pull. pull is the only required deployment action \u2014 it is used to define how Prefect will pull your deployment in remote execution environments.

    Each action is defined as a list of steps that are executing in sequence.

    Each step has the following format:

    section:\n- prefect_package.path.to.importable.step:\n    id: \"step-id\" # optional\n    requires: \"pip-installable-package-spec\" # optional\n    kwarg1: value\n    kwarg2: more-values\n

    Every step can optionally provide a requires field that Prefect will use to auto-install in the event that the step cannot be found in the current environment. Each step can also specify an id for the step which is used when referencing step outputs in later steps. The additional fields map directly onto Python keyword arguments to the step function. Within a given section, steps always run in the order that they are provided within the prefect.yaml file.

    Deployment Instruction Overrides

    build, push, and pull sections can all be overridden on a per-deployment basis by defining build, push, and pull fields within a deployment definition in the prefect.yaml file.

    The prefect deploy command will use any build, push, or pull instructions provided in a deployment's definition in the prefect.yaml file.

    This capability is useful with multiple deployments that require different deployment instructions.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#the-build-action","title":"The build action","text":"

    The build section of prefect.yaml is where any necessary side effects for running your deployments are built - the most common type of side effect produced here is a Docker image. If you initialize with the docker recipe, you will be prompted to provide required information, such as image name and tag:

    prefect init --recipe docker\n>> image_name: < insert image name here >\n>> tag: < insert image tag here >\n

    Use --field to avoid the interactive experience

    We recommend that you only initialize a recipe when you are first creating your deployment structure, and afterwards store your configuration files within version control. However, sometimes you may need to initialize programmatically and avoid the interactive prompts. To do so, provide all required fields for your recipe using the --field flag:

    prefect init --recipe docker \\\n    --field image_name=my-repo/my-image \\\n    --field tag=my-tag\n

    build:\n- prefect_docker.deployments.steps.build_docker_image:\n    requires: prefect-docker>=0.3.0\n    image_name: my-repo/my-image\n    tag: my-tag\n    dockerfile: auto\n    push: true\n

    Once you've confirmed that these fields are set to their desired values, this step will automatically build a Docker image with the provided name and tag and push it to the repository referenced by the image name. As the prefect-docker package documentation notes, this step produces a few fields that can optionally be used in future steps or within prefect.yaml as template values. It is best practice to use {{ image }} within prefect.yaml (specifically the work pool's job variables section) so that you don't risk having your build step and deployment specification get out of sync with hardcoded values.

    Note

    Note that in the build step example above, we relied on the prefect-docker package; in cases that deal with external services, additional packages are often required and will be auto-installed for you.

    Pass output to downstream steps

    Each deployment action can be composed of multiple steps. For example, if you wanted to build a Docker image tagged with the current commit hash, you could use the run_shell_script step and feed the output into the build_docker_image step:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        id: get-commit-hash\n        script: git rev-parse --short HEAD\n        stream_output: false\n    - prefect_docker.deployments.steps.build_docker_image:\n        requires: prefect-docker\n        image_name: my-image\n        image_tag: \"{{ get-commit-hash.stdout }}\"\n        dockerfile: auto\n

    Note that the id field is used in the run_shell_script step so that its output can be referenced in the next step.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#the-push-action","title":"The push action","text":"

    The push section is most critical for situations in which code is not stored on persistent filesystems or in version control. In this scenario, code is often pushed and pulled from a Cloud storage bucket of some kind (e.g., S3, GCS, Azure Blobs, etc.). The push section allows users to specify and customize the logic for pushing this code repository to arbitrary remote locations.

    For example, a user wishing to store their code in an S3 bucket and rely on default worker settings for its runtime environment could use the s3 recipe:

    prefect init --recipe s3\n>> bucket: < insert bucket name here >\n

    Inspecting our newly created prefect.yaml file we find that the push and pull sections have been templated out for us as follows:

    push:\n- prefect_aws.deployments.steps.push_to_s3:\n    id: push-code\n    requires: prefect-aws>=0.3.0\n    bucket: my-bucket\n    folder: project-name\n    credentials: null\n\npull:\n- prefect_aws.deployments.steps.pull_from_s3:\n    requires: prefect-aws>=0.3.0\n    bucket: my-bucket\n    folder: \"{{ push-code.folder }}\"\n    credentials: null\n

    The bucket has been populated with our provided value (which also could have been provided with the --field flag); note that the folder property of the push step is a template - the pull_from_s3 step outputs both a bucket value as well as a folder value that can be used to template downstream steps. Doing this helps you keep your steps consistent across edits.

    As discussed above, if you are using blocks, the credentials section can be templated with a block reference for secure and dynamic credentials access:

    push:\n- prefect_aws.deployments.steps.push_to_s3:\n    requires: prefect-aws>=0.3.0\n    bucket: my-bucket\n    folder: project-name\n    credentials: \"{{ prefect.blocks.aws-credentials.dev-credentials }}\"\n

    Anytime you run prefect deploy, this push section will be executed upon successful completion of your build section. For more information on the mechanics of steps, see below.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#the-pull-action","title":"The pull action","text":"

    The pull section is the most important section within the prefect.yaml file. It contains instructions for preparing your flows for a deployment run. These instructions will be executed each time a deployment created within this folder is run via a worker.

    There are three main types of steps that typically show up in a pull section:

    • set_working_directory: this step simply sets the working directory for the process prior to importing your flow
    • git_clone: this step clones the provided repository on the provided branch
    • pull_from_{cloud}: this step pulls the working directory from a Cloud storage location (e.g., S3)

    Use block and variable references

    All block and variable references within your pull step will remain unresolved until runtime and will be pulled each time your deployment is run. This allows you to avoid storing sensitive information insecurely; it also allows you to manage certain types of configuration from the API and UI without having to rebuild your deployment every time.

    Below is an example of how to use an existing GitHubCredentials block to clone a private GitHub repository:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/org/repo.git\n        credentials: \"{{ prefect.blocks.github-credentials.my-credentials }}\"\n

    Alternatively, you can specify a BitBucketCredentials or GitLabCredentials block to clone from Bitbucket or GitLab. In lieu of a credentials block, you can also provide a GitHub, GitLab, or Bitbucket token directly to the 'access_token` field. You can use a Secret block to do this securely:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://bitbucket.org/org/repo.git\n        access_token: \"{{ prefect.blocks.secret.bitbucket-token }}\"\n
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#utility-steps","title":"Utility steps","text":"

    Utility steps can be used within a build, push, or pull action to assist in managing the deployment lifecycle:

    • run_shell_script allows for the execution of one or more shell commands in a subprocess, and returns the standard output and standard error of the script. This step is useful for scripts that require execution in a specific environment, or those which have specific input and output requirements.

    Here is an example of retrieving the short Git commit hash of the current repository to use as a Docker image tag:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        id: get-commit-hash\n        script: git rev-parse --short HEAD\n        stream_output: false\n    - prefect_docker.deployments.steps.build_docker_image:\n        requires: prefect-docker>=0.3.0\n        image_name: my-image\n        tag: \"{{ get-commit-hash.stdout }}\"\n        dockerfile: auto\n

    Provided environment variables are not expanded by default

    To expand environment variables in your shell script, set expand_env_vars: true in your run_shell_script step. For example:

    - prefect.deployments.steps.run_shell_script:\n    id: get-user\n    script: echo $USER\n    stream_output: true\n    expand_env_vars: true\n

    Without expand_env_vars: true, the above step would return a literal string $USER instead of the current user.

    • pip_install_requirements installs dependencies from a requirements.txt file within a specified directory.

    Below is an example of installing dependencies from a requirements.txt file after cloning:

    pull:\n    - prefect.deployments.steps.git_clone:\n        id: clone-step  # needed in order to be referenced in subsequent steps\n        repository: https://github.com/org/repo.git\n    - prefect.deployments.steps.pip_install_requirements:\n        directory: {{ clone-step.directory }}  # `clone-step` is a user-provided `id` field\n        requirements_file: requirements.txt\n

    Below is an example that retrieves an access token from a 3rd party Key Vault and uses it in a private clone step:

    pull:\n- prefect.deployments.steps.run_shell_script:\n    id: get-access-token\n    script: az keyvault secret show --name <secret name> --vault-name <secret vault> --query \"value\" --output tsv\n    stream_output: false\n- prefect.deployments.steps.git_clone:\n    repository: https://bitbucket.org/samples/deployments.git\n    branch: master\n    access_token: \"{{ get-access-token.stdout }}\"\n

    You can also run custom steps by packaging them. In the example below, retrieve_secrets is a custom python module that has been packaged into the default working directory of a Docker image (which is /opt/prefect by default). main is the function entry point, which returns an access token (e.g. return {\"access_token\": access_token}) like the preceding example, but utilizing the Azure Python SDK for retrieval.

    - retrieve_secrets.main:\n    id: get-access-token\n- prefect.deployments.steps.git_clone:\n    repository: https://bitbucket.org/samples/deployments.git\n    branch: master\n    access_token: '{{ get-access-token.access_token }}'\n
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#templating-options","title":"Templating options","text":"

    Values that you place within your prefect.yaml file can reference dynamic values in several different ways:

    • step outputs: every step of both build and push produce named fields such as image_name; you can reference these fields within prefect.yaml and prefect deploy will populate them with each call. References must be enclosed in double brackets and be of the form \"{{ field_name }}\"
    • blocks: Prefect blocks can also be referenced with the special syntax {{ prefect.blocks.block_type.block_slug }}. It is highly recommended that you use block references for any sensitive information (such as a GitHub access token or any credentials) to avoid hardcoding these values in plaintext
    • variables: Prefect variables can also be referenced with the special syntax {{ prefect.variables.variable_name }}. Variables can be used to reference non-sensitive, reusable pieces of information such as a default image name or a default work pool name.
    • environment variables: you can also reference environment variables with the special syntax {{ $MY_ENV_VAR }}. This is especially useful for referencing environment variables that are set at runtime.

    As an example, consider the following prefect.yaml file:

    build:\n- prefect_docker.deployments.steps.build_docker_image:\n    id: build-image\n    requires: prefect-docker>=0.3.0\n    image_name: my-repo/my-image\n    tag: my-tag\n    dockerfile: auto\n    push: true\n\ndeployments:\n- # base metadata\n    name: null\n    version: \"{{ build-image.tag }}\"\n    tags:\n        - \"{{ $my_deployment_tag }}\"\n        - \"{{ prefect.variables.some_common_tag }}\"\n    description: null\n    schedule: null\n\n    # flow-specific fields\n    entrypoint: null\n    parameters: {}\n\n    # infra-specific fields\n    work_pool:\n        name: \"my-k8s-work-pool\"\n        work_queue_name: null\n        job_variables:\n            image: \"{{ build-image.image }}\"\n            cluster_config: \"{{ prefect.blocks.kubernetes-cluster-config.my-favorite-config }}\"\n

    So long as our build steps produce fields called image_name and tag, every time we deploy a new version of our deployment, the {{ build-image.image }} variable will be dynamically populated with the relevant values.

    Docker step

    The most commonly used build step is prefect_docker.deployments.steps.build_docker_image which produces both the image_name and tag fields.

    For an example, check out the deployments tutorial.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-configurations","title":"Deployment Configurations","text":"

    You can create multiple deployments from one or more python file that use .deploy. Similarly a prefect.yaml file can have multiple deployment configurations that control the behavior of created deployments.

    These deployments can be managed independently of one another, allowing you to deploy the same flow with different configurations in the same codebase.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#working-with-multiple-deployments","title":"Working with multiple deployments","text":".deployprefect.yaml

    To create multiple work pool-based deployments at once you can use the deploy function, which is analogous to the serve function.

    from prefect import deploy, flow\n\n@flow(log_prints=True)\ndef buy():\n    print(\"Buying securities\")\n\n\nif __name__ == \"__main__\":\n    deploy(\n        buy.to_deployment(name=\"dev-deploy\", work_pool_name=\"my-dev-work-pool\"),\n        buy.to_deployment(name=\"prod-deploy\", work_pool_name=\"my-prod-work-pool\"),\n        image=\"my-registry/my-image:dev\",\n        push=False,\n    )\n

    Note that in the example above we created two deployments from the same flow, but with different work pools. Alternatively, we could have created two deployments from different flows.

    from prefect import deploy, flow\n\n@flow(log_prints=True)\ndef buy():\n    print(\"Buying securities.\")\n\n@flow(log_prints=True)\ndef sell():\n    print(\"Selling securities.\")\n\n\nif __name__ == \"__main__\":\n    deploy(\n        buy.to_deployment(name=\"buy-deploy\"),\n        sell.to_deployment(name=\"sell-deploy\"),\n        work_pool_name=\"my-dev-work-pool\"\n        image=\"my-registry/my-image:dev\",\n        push=False,\n    )\n

    In the example above the code for both flows gets baked into the same image.

    We can specify that one or more flows should be pulled from a remote location at runtime by using the from_source method. Here's an example of deploying two flows, one defined locally and one defined in a remote repository:

    from prefect import deploy, flow\n\n\n@flow(log_prints=True)\ndef local_flow():\n    print(\"I'm a flow!\")\n\nif __name__ == \"__main__\":\n    deploy(\n        local_flow.to_deployment(name=\"example-deploy-local-flow\"),\n        flow.from_source(\n            source=\"https://github.com/org/repo.git\",\n            entrypoint=\"flows.py:my_flow\",\n        ).to_deployment(\n            name=\"example-deploy-remote-flow\",\n        ),\n        work_pool_name=\"my-work-pool\",\n        image=\"my-registry/my-image:dev\",\n    )\n

    You could pass any number of flows to the deploy function. This behavior is useful if using a monorepo approach to your workflows.

    Prefect supports multiple deployment declarations within the prefect.yaml file. This method of declaring multiple deployments allows the configuration for all deployments to be version controlled and deployed with a single command.

    New deployment declarations can be added to the prefect.yaml file by adding a new entry to the deployments list. Each deployment declaration must have a unique name field which is used to select deployment declarations when using the prefect deploy command.

    For example, consider the following prefect.yaml file:

    build: ...\npush: ...\npull: ...\n\ndeployments:\n- name: deployment-1\n    entrypoint: flows/hello.py:my_flow\n    parameters:\n        number: 42,\n        message: Don't panic!\n    work_pool:\n        name: my-process-work-pool\n        work_queue_name: primary-queue\n\n- name: deployment-2\n    entrypoint: flows/goodbye.py:my_other_flow\n    work_pool:\n        name: my-process-work-pool\n        work_queue_name: secondary-queue\n\n- name: deployment-3\n    entrypoint: flows/hello.py:yet_another_flow\n    work_pool:\n        name: my-docker-work-pool\n        work_queue_name: tertiary-queue\n

    This file has three deployment declarations, each referencing a different flow. Each deployment declaration has a unique name field and can be deployed individually by using the --name flag when deploying.

    For example, to deploy deployment-1 you would run:

    prefect deploy --name deployment-1\n

    To deploy multiple deployments you can provide multiple --name flags:

    prefect deploy --name deployment-1 --name deployment-2\n

    To deploy multiple deployments with the same name, you can prefix the deployment name with its flow name:

    prefect deploy --name my_flow/deployment-1 --name my_other_flow/deployment-1\n

    To deploy all deployments you can use the --all flag:

    prefect deploy --all\n

    To deploy deployments that match a pattern you can run:

    prefect deploy -n my-flow/* -n *dev/my-deployment -n dep*prod\n

    The above command will deploy all deployments from the flow my-flow, all flows ending in dev with a deployment named my-deployment, and all deployments starting with dep and ending in prod.

    CLI Options When Deploying Multiple Deployments

    When deploying more than one deployment with a single prefect deploy command, any additional attributes provided via the CLI will be ignored.

    To provide overrides to a deployment via the CLI, you must deploy that deployment individually.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#reusing-configuration-across-deployments","title":"Reusing configuration across deployments","text":"

    Because a prefect.yaml file is a standard YAML file, you can use YAML aliases to reuse configuration across deployments.

    This functionality is useful when multiple deployments need to share the work pool configuration, deployment actions, or other configurations.

    You can declare a YAML alias by using the &{alias_name} syntax and insert that alias elsewhere in the file with the *{alias_name} syntax. When aliasing YAML maps, you can also override specific fields of the aliased map by using the <<: *{alias_name} syntax and adding additional fields below.

    We recommend adding a definitions section to your prefect.yaml file at the same level as the deployments section to store your aliases.

    For example, consider the following prefect.yaml file:

    build: ...\npush: ...\npull: ...\n\ndefinitions:\n    work_pools:\n        my_docker_work_pool: &my_docker_work_pool\n            name: my-docker-work-pool\n            work_queue_name: default\n            job_variables:\n                image: \"{{ build-image.image }}\"\n    schedules:\n        every_ten_minutes: &every_10_minutes\n            interval: 600\n    actions:\n        docker_build: &docker_build\n            - prefect_docker.deployments.steps.build_docker_image: &docker_build_config\n                id: build-image\n                requires: prefect-docker>=0.3.0\n                image_name: my-example-image\n                tag: dev\n                dockerfile: auto\n                push: true\n\ndeployments:\n- name: deployment-1\n    entrypoint: flows/hello.py:my_flow\n    schedule: *every_10_minutes\n    parameters:\n        number: 42,\n        message: Don't panic!\n    work_pool: *my_docker_work_pool\n    build: *docker_build # Uses the full docker_build action with no overrides\n\n- name: deployment-2\n    entrypoint: flows/goodbye.py:my_other_flow\n    work_pool: *my_docker_work_pool\n    build:\n        - prefect_docker.deployments.steps.build_docker_image:\n            <<: *docker_build_config # Uses the docker_build_config alias and overrides the dockerfile field\n            dockerfile: Dockerfile.custom\n\n- name: deployment-3\n    entrypoint: flows/hello.py:yet_another_flow\n    schedule: *every_10_minutes\n    work_pool:\n        name: my-process-work-pool\n        work_queue_name: primary-queue\n

    In the above example, we are using YAML aliases to reuse work pool, schedule, and build configuration across multiple deployments:

    • deployment-1 and deployment-2 are using the same work pool configuration
    • deployment-1 and deployment-3 are using the same schedule
    • deployment-1 and deployment-2 are using the same build deployment action, but deployment-2 is overriding the dockerfile field to use a custom Dockerfile
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-declaration-reference","title":"Deployment declaration reference","text":"","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-fields","title":"Deployment fields","text":"

    Below are fields that can be added to each deployment declaration.

    Property Description name The name to give to the created deployment. Used with the prefect deploy command to create or update specific deployments. version An optional version for the deployment. tags A list of strings to assign to the deployment as tags. description An optional description for the deployment. schedule An optional schedule to assign to the deployment. Fields for this section are documented in the Schedule Fields section. triggers An optional array of triggers to assign to the deployment entrypoint Required path to the .py file containing the flow you want to deploy (relative to the root directory of your development folder) combined with the name of the flow function. Should be in the format path/to/file.py:flow_function_name. parameters Optional default values to provide for the parameters of the deployed flow. Should be an object with key/value pairs. enforce_parameter_schema Boolean flag that determines whether the API should validate the parameters passed to a flow run against the parameter schema generated for the deployed flow. work_pool Information on where to schedule flow runs for the deployment. Fields for this section are documented in the Work Pool Fields section.","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#schedule-fields","title":"Schedule fields","text":"

    Below are fields that can be added to a deployment declaration's schedule section.

    Property Description interval Number of seconds indicating the time between flow runs. Cannot be used in conjunction with cron or rrule. anchor_date Datetime string indicating the starting or \"anchor\" date to begin the schedule. If no anchor_date is supplied, the current UTC time is used. Can only be used with interval. timezone String name of a time zone, used to enforce localization behaviors like DST boundaries. See the IANA Time Zone Database for valid time zones. cron A valid cron string. Cannot be used in conjunction with interval or rrule. day_or Boolean indicating how croniter handles day and day_of_week entries. Must be used with cron. Defaults to True. rrule String representation of an RRule schedule. See the rrulestr examples for syntax. Cannot be used in conjunction with interval or cron.

    For more information about schedules, see the Schedules concept doc.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#work-pool-fields","title":"Work pool fields","text":"

    Below are fields that can be added to a deployment declaration's work_pool section.

    Property Description name The name of the work pool to schedule flow runs in for the deployment. work_queue_name The name of the work queue within the specified work pool to schedule flow runs in for the deployment. If not provided, the default queue for the specified work pool will be used. job_variables Values used to override the default values in the specified work pool's base job template. Maps directly to a created deployments infra_overrides attribute.","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-mechanics","title":"Deployment mechanics","text":"

    Anytime you run prefect deploy in a directory that contains a prefect.yaml file, the following actions are taken in order:

    • The prefect.yaml file is loaded. First, the build section is loaded and all variable and block references are resolved. The steps are then run in the order provided.
    • Next, the push section is loaded and all variable and block references are resolved; the steps within this section are then run in the order provided
    • Next, the pull section is templated with any step outputs but is not run. Note that block references are not hydrated for security purposes - block references are always resolved at runtime
    • Next, all variable and block references are resolved with the deployment declaration. All flags provided via the prefect deploy CLI are then overlaid on the values loaded from the file.
    • The final step occurs when the fully realized deployment specification is registered with the Prefect API

    Deployment Instruction Overrides

    The build, push, and pull sections in deployment definitions take precedence over the corresponding sections above them in prefect.yaml.

    Each time a step is run, the following actions are taken in order:

    • The step's inputs and block / variable references are resolved (see the templating documentation above for more details).
    • The step's function is imported; if it cannot be found, the special requires keyword is used to install the necessary packages
    • The step's function is called with the resolved inputs.
    • The step's output is returned and used to resolve inputs for subsequent steps.
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#next-steps","title":"Next steps","text":"

    Now that you are familiar with creating deployments, you may want to explore push work pools or Kubernetes work pools.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/runtime-context/","title":"Runtime context","text":"

    Prefect tracks information about the current flow or task run with a run context. The run context can be thought of as a global variable that allows the Prefect engine to determine relationships between your runs, e.g. which flow your task is called from.

    The run context itself contains many internal objects used by Prefect to manage execution of your run and is only available in specific situations. For this reason, we expose a simple interface that only includes the items you care about and dynamically retrieves additional information when necessary. We call this the \"runtime context\" as it contains information that can only be accessed when a run is happening.

    Mock values via environment variable

    Oftentimes, you may want to mock certain values for testing purposes. For example, by manually setting an ID or a scheduled start time to ensure your code is functioning properly. Starting in version 2.10.3, you can mock values in runtime via environment variable using the schema PREFECT__RUNTIME__{SUBMODULE}__{KEY_NAME}=value: ```bash $ export PREFECT__RUNTIME__TASK_RUN__FAKE_KEY='foo' $ python -c 'from prefect.runtime import task_run; print(task_run.fake_key)' # \"foo\"

    `` </div> If the environment variable mocks an existing runtime attribute, the value is cast to the same type. This works for runtime attributes of basic types (bool,int,floatandstr) andpendulum.DateTime. For complex types likelistordict`, we suggest mocking them using monkeypatch or a similar tool.

    ","tags":["flows","subflows","tasks","deployments"],"boost":2},{"location":"guides/runtime-context/#accessing-runtime-information","title":"Accessing runtime information","text":"

    The prefect.runtime module is the home for all runtime context access. Each major runtime concept has its own submodule:

    • deployment: Access information about the deployment for the current run
    • flow_run: Access information about the current flow run
    • task_run: Access information about the current task run

    For example:

    from prefect import flow, task\nfrom prefect import runtime\n\n@flow(log_prints=True)\ndef my_flow(x):\n    print(\"My name is\", runtime.flow_run.name)\n    print(\"I belong to deployment\", runtime.deployment.name)\n    my_task(2)\n\n@task\ndef my_task(y):\n    print(\"My name is\", runtime.task_run.name)\n    print(\"Flow run parameters:\", runtime.flow_run.parameters)\n\nmy_flow(1)\n

    Outputs:

    $ python my_runtime_info.py\n10:08:02.948 | INFO    | prefect.engine - Created flow run 'solid-gibbon' for flow 'my-flow'\n10:08:03.555 | INFO    | Flow run 'solid-gibbon' - My name is solid-gibbon\n10:08:03.558 | INFO    | Flow run 'solid-gibbon' - I belong to deployment None\n10:08:03.703 | INFO    | Flow run 'solid-gibbon' - Created task run 'my_task-0' for task 'my_task'\n10:08:03.704 | INFO    | Flow run 'solid-gibbon' - Executing 'my_task-0' immediately...\n10:08:04.006 | INFO    | Task run 'my_task-0' - My name is my_task-0\n10:08:04.007 | INFO    | Task run 'my_task-0' - Flow run parameters: {'x': 1}\n10:08:04.105 | INFO    | Task run 'my_task-0' - Finished in state Completed()\n10:08:04.968 | INFO    | Flow run 'solid-gibbon' - Finished in state Completed('All states completed.')\n

    Above, we demonstrate access to information about the current flow run, task run, and deployment. When run without a deployment (via python my_runtime_info.py), you should see \"I belong to deployment None\" logged. When information is not available, the runtime will always return an empty value. Because this flow was run without a deployment, there is no deployment data. If this flow were deployed and executed by a worker, we'd see the name of the deployment instead.

    See the runtime API reference for a full list of available attributes.

    ","tags":["flows","subflows","tasks","deployments"],"boost":2},{"location":"guides/runtime-context/#accessing-the-run-context-directly","title":"Accessing the run context directly","text":"

    The current run context can be accessed with prefect.context.get_run_context(). This will raise an exception if no run context is available, meaning you are not in a flow or task run. If a task run context is available, it will be returned even if a flow run context is available.

    Alternatively, you can access the flow run or task run context explicitly. This will, for example, allow you to access the flow run context from a task run. Note that we do not send the flow run context to distributed task workers because the context is costly to serialize and deserialize.

    from prefect.context import FlowRunContext, TaskRunContext\n\nflow_run_ctx = FlowRunContext.get()\ntask_run_ctx = TaskRunContext.get()\n

    Unlike get_run_context, this will not raise an error if the context is not available. Instead, it will return None.

    ","tags":["flows","subflows","tasks","deployments"],"boost":2},{"location":"guides/settings/","title":"Profiles & Configuration","text":"

    Prefect's settings are documented and type-validated. By modifying the default settings, you can customize various aspects of the system.

    Settings can be viewed from the CLI or the UI.

    You can override the setting for a profile with environment variables.

    Prefect profiles are groups of settings that you can persist on your machine. When you change profiles, all of the settings configured in the profile are applied. You can apply profiles to individual commands or set a profile for your environment.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#commonly-configured-settings","title":"Commonly configured settings","text":"

    This section describes some commonly configured settings for Prefect installations. See Configuring settings for details on setting and unsetting configuration values.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#prefect_api_url","title":"PREFECT_API_URL","text":"

    The PREFECT_API_URL value specifies the API endpoint of your Prefect Cloud workspace or a Prefect server instance.

    For example, using a local Prefect server instance.

    PREFECT_API_URL=\"http://127.0.0.1:4200/api\"\n

    Using Prefect Cloud:

    PREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]\"\n

    View your Account ID and Workspace ID in your browser URL when logged into Prefect Cloud. For example: https://app.prefect.cloud/account/abc-my-account-id-is-here/workspaces/123-my-workspace-id-is-here.

    PREFECT_API_URL setting for workers

    When using workers and work pools (agent and block-based deployments are legacy) that can create flow runs for deployments in remote environments, PREFECT_API_URL must be set for the environment in which your worker is running.

    If you want the worker to communicate with Prefect Cloud or a Prefect server instance from a remote execution environment such as a VM or Docker container, you must configure PREFECT_API_URL in that environment.

    Running the Prefect UI behind a reverse proxy

    When using a reverse proxy (such as Nginx or Traefik) to proxy traffic to a locally-hosted Prefect UI instance, the Prefect server also needs to be configured to know how to connect to the API. The PREFECT_UI_API_URL should be set to the external proxy URL (e.g. if your external URL is https://prefect-server.example.com/ then set PREFECT_UI_API_URL=https://prefect-server.example.com/api for the Prefect server process). You can also accomplish this by setting PREFECT_API_URL to the API URL, as this setting is used as a fallback if PREFECT_UI_API_URL is not set.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#prefect_api_key","title":"PREFECT_API_KEY","text":"

    The PREFECT_API_KEY value specifies the API key used to authenticate with your Prefect Cloud workspace.

    PREFECT_API_KEY=\"[API-KEY]\"\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#prefect_home","title":"PREFECT_HOME","text":"

    The PREFECT_HOME value specifies the local Prefect directory for configuration files, profiles, and the location of the default Prefect SQLite database.

    PREFECT_HOME='~/.prefect'\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#prefect_local_storage_path","title":"PREFECT_LOCAL_STORAGE_PATH","text":"

    The PREFECT_LOCAL_STORAGE_PATH value specifies the default location of local storage for flow runs.

    PREFECT_LOCAL_STORAGE_PATH='${PREFECT_HOME}/storage'\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#database-settings","title":"Database settings","text":"

    Prefect provides several self-hosting database configuration settings you can read about here.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#logging-settings","title":"Logging settings","text":"

    Prefect provides several logging configuration settings that you can read about in the logging docs.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#configuring-settings","title":"Configuring settings","text":"

    The prefect config CLI commands enable you to view, set, and unset settings.

    Command Description set Change the value for a setting. unset Restore the default value for a setting. view Display the current settings.","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#viewing-settings-from-the-cli","title":"Viewing settings from the CLI","text":"

    The prefect config view command will display settings that override default values.

    $ prefect config view\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='DEBUG'\n

    You can show the sources of values with --show-sources:

    $ prefect config view --show-sources\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='DEBUG' (from env)\n

    You can also include default values with --show-defaults:

    $ prefect config view --show-defaults\nPREFECT_PROFILE='default'\nPREFECT_AGENT_PREFETCH_SECONDS='10' (from defaults)\nPREFECT_AGENT_QUERY_INTERVAL='5.0' (from defaults)\nPREFECT_API_KEY='None' (from defaults)\nPREFECT_API_REQUEST_TIMEOUT='60.0' (from defaults)\nPREFECT_API_URL='None' (from defaults)\n...\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#setting-and-clearing-values","title":"Setting and clearing values","text":"

    The prefect config set command lets you change the value of a default setting.

    A commonly used example is setting the PREFECT_API_URL, which you may need to change when interacting with different Prefect server instances or Prefect Cloud.

    # use a local Prefect server\nprefect config set PREFECT_API_URL=\"http://127.0.0.1:4200/api\"\n\n# use Prefect Cloud\nprefect config set PREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]\"\n

    If you want to configure a setting to use its default value, use the prefect config unset command.

    prefect config unset PREFECT_API_URL\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#overriding-defaults-with-environment-variables","title":"Overriding defaults with environment variables","text":"

    All settings have keys that match the environment variable that can be used to override them.

    For example, configuring the home directory:

    # environment variable\nexport PREFECT_HOME=\"/path/to/home\"\n
    # python\nimport prefect.settings\nprefect.settings.PREFECT_HOME.value()  # PosixPath('/path/to/home')\n

    Configuring the server's port:

    # environment variable\nexport PREFECT_SERVER_API_PORT=4242\n
    # python\nprefect.settings.PREFECT_SERVER_API_PORT.value()  # 4242\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#configuration-profiles","title":"Configuration profiles","text":"

    Prefect allows you to persist settings instead of setting an environment variable each time you open a new shell. Settings are persisted to profiles, which allow you to move between groups of settings quickly.

    The prefect profile CLI commands enable you to create, review, and manage profiles.

    Command Description create Create a new profile. delete Delete the given profile. inspect Display settings from a given profile; defaults to active. ls List profile names. rename Change the name of a profile. use Switch the active profile.

    If you configured settings for a profile, prefect profile inspect displays those settings:

    $ prefect profile inspect\nPREFECT_PROFILE = \"default\"\nPREFECT_API_KEY = \"pnu_XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX\"\nPREFECT_API_URL = \"http://127.0.0.1:4200/api\"\n

    You can pass the name of a profile to view its settings:

    $ prefect profile create test\n$ prefect profile inspect test\nPREFECT_PROFILE=\"test\"\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#creating-and-removing-profiles","title":"Creating and removing profiles","text":"

    Create a new profile with no settings:

    $ prefect profile create test\nCreated profile 'test' at /Users/terry/.prefect/profiles.toml.\n

    Create a new profile foo with settings cloned from an existing default profile:

    $ prefect profile create foo --from default\nCreated profile 'cloud' matching 'default' at /Users/terry/.prefect/profiles.toml.\n

    Rename a profile:

    $ prefect profile rename temp test\nRenamed profile 'temp' to 'test'.\n

    Remove a profile:

    $ prefect profile delete test\nRemoved profile 'test'.\n

    Removing the default profile resets it:

    $ prefect profile delete default\nReset profile 'default'.\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#change-values-in-profiles","title":"Change values in profiles","text":"

    Set a value in the current profile:

    $ prefect config set VAR=X\nSet variable 'VAR' to 'X'\nUpdated profile 'default'\n

    Set multiple values in the current profile:

    $ prefect config set VAR2=Y VAR3=Z\nSet variable 'VAR2' to 'Y'\nSet variable 'VAR3' to 'Z'\nUpdated profile 'default'\n

    You can set a value in another profile by passing the --profile NAME option to a CLI command:

    $ prefect --profile \"foo\" config set VAR=Y\nSet variable 'VAR' to 'Y'\nUpdated profile 'foo'\n

    Unset values in the current profile to restore the defaults:

    $ prefect config unset VAR2 VAR3\nUnset variable 'VAR2'\nUnset variable 'VAR3'\nUpdated profile 'default'\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#inspecting-profiles","title":"Inspecting profiles","text":"

    See a list of available profiles:

    $ prefect profile ls\n* default\ncloud\ntest\nlocal\n

    View all settings for a profile:

    $ prefect profile inspect cloud\nPREFECT_API_URL='https://api.prefect.cloud/api/accounts/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxx\nx/workspaces/xxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx'\nPREFECT_API_KEY='xxx_xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'          \n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#using-profiles","title":"Using profiles","text":"

    The profile default is used by default. There are several methods to switch to another profile.

    The recommended method is to use the prefect profile use command with the name of the profile:

    $ prefect profile use foo\nProfile 'test' now active.\n

    Alternatively, you may set the environment variable PREFECT_PROFILE to the name of the profile:

    export PREFECT_PROFILE=foo\n

    Or, specify the profile in the CLI command for one-time usage:

    prefect --profile \"foo\" ...\n

    Note that this option must come before the subcommand. For example, to list flow runs using the profile foo:

    prefect --profile \"foo\" flow-run ls\n

    You may use the -p flag as well:

    prefect -p \"foo\" flow-run ls\n

    You may also create an 'alias' to automatically use your profile:

    $ alias prefect-foo=\"prefect --profile 'foo' \"\n# uses our profile!\n$ prefect-foo config view  \n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#conflicts-with-environment-variables","title":"Conflicts with environment variables","text":"

    If setting the profile from the CLI with --profile, environment variables that conflict with settings in the profile will be ignored.

    In all other cases, environment variables will take precedence over the value in the profile.

    For example, a value set in a profile will be used by default:

    $ prefect config set PREFECT_LOGGING_LEVEL=\"ERROR\"\n$ prefect config view --show-sources\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='ERROR' (from profile)\n

    But, setting an environment variable will override the profile setting:

    $ export PREFECT_LOGGING_LEVEL=\"DEBUG\"\n$ prefect config view --show-sources\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='DEBUG' (from env)\n

    Unless the profile is explicitly requested when using the CLI:

    $ prefect --profile default config view --show-sources\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='ERROR' (from profile)\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#profile-files","title":"Profile files","text":"

    Profiles are persisted to the PREFECT_PROFILES_PATH, which can be changed with an environment variable.

    By default, it is stored in your PREFECT_HOME directory:

    $ prefect config view --show-defaults | grep PROFILES_PATH\nPREFECT_PROFILES_PATH='~/.prefect/profiles.toml'\n

    The TOML format is used to store profile data.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/state-change-hooks/","title":"State Change Hooks","text":"

    You've read about how state change hooks execute code in response to changes in flow or task run states, enabling you to define actions for specific state transitions in a workflow. Now let's see some real-world use cases!

    ","tags":["state change","hooks","triggers"],"boost":2},{"location":"guides/state-change-hooks/#example-use-cases","title":"Example use cases","text":"","tags":["state change","hooks","triggers"],"boost":2},{"location":"guides/state-change-hooks/#send-a-notification-when-a-flow-run-fails","title":"Send a notification when a flow run fails","text":"

    State change hooks enable you to customize messages sent when tasks transition between states, such as sending notifications containing sensitive information when tasks enter a Failed state. Let's run a client-side hook upon a flow run entering a Failed state.

    from prefect import flow\nfrom prefect.blocks.core import Block\nfrom prefect.settings import PREFECT_API_URL\n\ndef notify_slack(flow, flow_run, state):\n    slack_webhook_block = Block.load(\n        \"slack-webhook/my-slack-webhook\"\n    )\n\n    slack_webhook_block.notify(\n        (\n            f\"Your job {flow_run.name} entered {state.name} \"\n            f\"with message:\\n\\n\"\n            f\"See <https://{PREFECT_API_URL.value()}/flow-runs/\"\n            f\"flow-run/{flow_run.id}|the flow run in the UI>\\n\\n\"\n            f\"Tags: {flow_run.tags}\\n\\n\"\n            f\"Scheduled start: {flow_run.expected_start_time}\"\n        )\n    )\n\n@flow(on_failure=[notify_slack], retries=1)\ndef failing_flow():\n    raise ValueError(\"oops!\")\n\nif __name__ == \"__main__\":\n    failing_flow()\n

    Note that because we've configured retries here, the on_failure hook will not run until all retries have completed, when the flow run finally enters a Failed state.

    ","tags":["state change","hooks","triggers"],"boost":2},{"location":"guides/state-change-hooks/#delete-a-cloud-run-job-when-a-flow-crashes","title":"Delete a Cloud Run job when a flow crashes","text":"

    State change hooks can aid in managing infrastructure cleanup in scenarios where tasks spin up individual infrastructure resources independently of Prefect. When a flow run crashes, tasks may exit abruptly, resulting in the potential omission of cleanup logic within the tasks. State change hooks can be used to ensure infrastructure is properly cleaned up even when a flow run enters a Crashed state.

    Let's create a hook that deletes a Cloud Run job if the flow run crashes.

    import os\nfrom prefect import flow, task\nfrom prefect.blocks.system import String\nfrom prefect.client import get_client\nimport prefect.runtime\n\nasync def delete_cloud_run_job(flow, flow_run, state):\n    \"\"\"Flow run state change hook that deletes a Cloud Run Job if\n    the flow run crashes.\"\"\"\n\n    # retrieve Cloud Run job name\n    cloud_run_job_name = await String.load(\n        name=\"crashing-flow-cloud-run-job\"\n    )\n\n    # delete Cloud Run job\n    delete_job_command = f\"yes | gcloud beta run jobs delete \n    {cloud_run_job_name.value} --region us-central1\"\n    os.system(delete_job_command)\n\n    # clean up the Cloud Run job string block as well\n    async with get_client() as client:\n        block_document = await client.read_block_document_by_name(\n            \"crashing-flow-cloud-run-job\", block_type_slug=\"string\"\n        )\n        await client.delete_block_document(block_document.id)\n\n@task\ndef my_task_that_crashes():\n    raise SystemExit(\"Crashing on purpose!\")\n\n@flow(on_crashed=[delete_cloud_run_job])\ndef crashing_flow():\n    \"\"\"Save the flow run name (i.e. Cloud Run job name) as a \n    String block. It then executes a task that ends up crashing.\"\"\"\n    flow_run_name = prefect.runtime.flow_run.name\n    cloud_run_job_name = String(value=flow_run_name)\n    cloud_run_job_name.save(\n        name=\"crashing-flow-cloud-run-job\", overwrite=True\n    )\n\n    my_task_that_crashes()\n\nif __name__ == \"__main__\":\n    crashing_flow()\n
    ","tags":["state change","hooks","triggers"],"boost":2},{"location":"guides/testing/","title":"Testing","text":"

    Once you have some awesome flows, you probably want to test them!

    ","tags":["testing","unit testing","development"],"boost":2},{"location":"guides/testing/#unit-testing-flows","title":"Unit testing flows","text":"

    Prefect provides a simple context manager for unit tests that allows you to run flows and tasks against a temporary local SQLite database.

    from prefect import flow\nfrom prefect.testing.utilities import prefect_test_harness\n\n@flow\ndef my_favorite_flow():\n    return 42\n\ndef test_my_favorite_flow():\n  with prefect_test_harness():\n      # run the flow against a temporary testing database\n      assert my_favorite_flow() == 42\n

    For more extensive testing, you can leverage prefect_test_harness as a fixture in your unit testing framework. For example, when using pytest:

    from prefect import flow\nimport pytest\nfrom prefect.testing.utilities import prefect_test_harness\n\n@pytest.fixture(autouse=True, scope=\"session\")\ndef prefect_test_fixture():\n    with prefect_test_harness():\n        yield\n\n@flow\ndef my_favorite_flow():\n    return 42\n\ndef test_my_favorite_flow():\n    assert my_favorite_flow() == 42\n

    Note

    In this example, the fixture is scoped to run once for the entire test session. In most cases, you will not need a clean database for each test and just want to isolate your test runs to a test database. Creating a new test database per test creates significant overhead, so we recommend scoping the fixture to the session. If you need to isolate some tests fully, you can use the test harness again to create a fresh database.

    ","tags":["testing","unit testing","development"],"boost":2},{"location":"guides/testing/#unit-testing-tasks","title":"Unit testing tasks","text":"

    To test an individual task, you can access the original function using .fn:

    from prefect import flow, task\n\n@task\ndef my_favorite_task():\n    return 42\n\n@flow\ndef my_favorite_flow():\n    val = my_favorite_task()\n    return val\n\ndef test_my_favorite_task():\n    assert my_favorite_task.fn() == 42\n

    Disable logger

    If your task makes uses a logger, you can disable the logger in order to avoid the RuntimeError raised from a missing flow context.

    from prefect.logging import disable_run_logger\n\ndef test_my_favorite_task():\n    with disable_run_logger():\n        assert my_favorite_task.fn() == 42\n

    ","tags":["testing","unit testing","development"],"boost":2},{"location":"guides/troubleshooting/","title":"Troubleshooting","text":"

    Don't Panic! If you experience an error with Prefect, there are many paths to understanding and resolving it. The first troubleshooting step is confirming that you are running the latest version of Prefect. If you are not, be sure to upgrade to the latest version, since the issue may have already been fixed. Beyond that, there are several categories of errors:

    • The issue may be in your flow code, in which case you should carefully read the logs.
    • The issue could be with how you are authenticated, and whether or not you are connected to Cloud.
    • The issue might have to do with how your code is executed.
    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#upgrade","title":"Upgrade","text":"

    Prefect is constantly evolving, adding new features and fixing bugs. Chances are that a patch has already been identified and released. Search existing issues for similar reports and check out the Release Notes. Upgrade to the newest version with the following command:

    pip install --upgrade prefect\n

    Different components may use different versions of Prefect:

    • Cloud will generally always be the newest version. Cloud is continuously deployed by the Prefect team. When using a self-hosted server, you can control this version.
    • Workers and agents typically don't change versions frequently, and are usually whatever the latest version was at the time of creation. Workers and agents provision infrastructure for flow runs, so upgrading them may help with infrastructure problems.
    • Flows could use a different version than the worker or agent that created them, especially when running in different environments. Suppose your worker and flow both use the latest official Docker image, but your worker was created a month ago. Your worker will often be on an older version than your flow.

    Integration Versions

    Keep in mind that integrations are versioned and released independently of the core Prefect library. They should be upgraded simultaneously with the core library, using the same method.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#logs","title":"Logs","text":"

    In many cases, there will be an informative stack trace in Prefect's logs. Read it carefully, locate the source of the error, and try to identify the cause.

    There are two types of logs:

    • Flow and task logs are always scoped to a flow. They are sent to Prefect and are viewable in the UI.
    • Worker and agent logs are not scoped to a flow and may have more information on what happened before the flow started. These logs are generally only available where the worker or agent is running.

    If your flow and task logs are empty, there may have been an infrastructure issue that prevented your flow from starting. Check your worker logs for more details.

    If there is no clear indication of what went wrong, try updating the logging level from the default INFO level to the DEBUG level. Settings such as the logging level are propagated from the worker environment to the flow run environment and can be set via environment variables or the prefect config set CLI:

    # Using the CLI\nprefect config set PREFECT_LOGGING_LEVEL=DEBUG\n\n# Using environment variables\nexport PREFECT_LOGGING_LEVEL=DEBUG\n

    The DEBUG logging level produces a high volume of logs so consider setting it back to INFO once any issues are resolved.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#cloud","title":"Cloud","text":"

    When using Prefect Cloud, there are the additional concerns of authentication and authorization. The Prefect API authenticates users and service accounts - collectively known as actors - with API keys. Missing, incorrect, or expired API keys will result in a 401 response with detail Invalid authentication credentials. Use the following command to check your authentication, replacing $PREFECT_API_KEY with your API key:

    curl -s -H \"Authorization: Bearer $PREFECT_API_KEY\" \"https://api.prefect.cloud/api/me/\"\n

    Users vs Service Accounts

    Service accounts - sometimes referred to as bots - represent non-human actors that interact with Prefect such as workers and CI/CD systems. Each human that interacts with Prefect should be represented as a user. User API keys start with pnu_ and service account API keys start with pnb_.

    Supposing the response succeeds, let's check our authorization. Actors can be members of workspaces. An actor attempting an action in a workspace they are not a member of will result in a 404 response. Use the following command to check your actor's workspace memberships:

    curl -s -H \"Authorization: Bearer $PREFECT_API_KEY\" \"https://api.prefect.cloud/api/me/workspaces\"\n

    Formatting JSON

    Python comes with a helpful tool for formatting JSON. Append the following to the end of the command above to make the output more readable: | python -m json.tool

    Make sure your actor is a member of the workspace you are working in. Within a workspace, an actor has a role which grants them certain permissions. Insufficient permissions will result in an error. For example, starting an agent or worker with the Viewer role, will result in errors.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#execution","title":"Execution","text":"

    Prefect flows can be executed locally by the user, or remotely by a worker or agent. Local execution generally means that you - the user - run your flow directly with a command like python flow.py. Remote execution generally means that a worker runs your flow via a deployment, optionally on different infrastructure.

    With remote execution, the creation of your flow run happens separately from its execution. Flow runs are assigned to a work pool and a work queue. For flow runs to execute, a worker must be subscribed to the work pool and work queue, otherwise the flow runs will go from Scheduled to Late. Ensure that your work pool and work queue have a subscribed worker.

    Local and remote execution can also differ in their treatment of relative imports. If switching from local to remote execution results in local import errors, try replicating the behavior by executing the flow locally with the -m flag (i.e. python -m flow instead of python flow.py). Read more about -m here.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#api-tests-return-an-unexpected-307-redirected","title":"API tests return an unexpected 307 Redirected","text":"

    Summary: Requests require a trailing / in the request URL.

    If you write a test that does not include a trailing / when making a request to a specific endpoint:

    async def test_example(client):\n    response = await client.post(\"/my_route\")\n    assert response.status_code == 201\n

    You'll see a failure like:

    E       assert 307 == 201\nE        +  where 307 = <Response [307 Temporary Redirect]>.status_code\n

    To resolve this, include the trailing /:

    async def test_example(client):\n    response = await client.post(\"/my_route/\")\n    assert response.status_code == 201\n

    Note: requests to nested URLs may exhibit the opposite behavior and require no trailing slash:

    async def test_nested_example(client):\n    response = await client.post(\"/my_route/filter/\")\n    assert response.status_code == 307\n\n    response = await client.post(\"/my_route/filter\")\n    assert response.status_code == 200\n

    Reference: \"HTTPX disabled redirect following by default\" in 0.22.0.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#pytestpytestunraisableexceptionwarning-or-resourcewarning","title":"pytest.PytestUnraisableExceptionWarning or ResourceWarning","text":"

    As you're working with one of the FlowRunner implementations, you may get an error like this one:

    E               pytest.PytestUnraisableExceptionWarning: Exception ignored in: <ssl.SSLSocket fd=-1, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0>\nE\nE               Traceback (most recent call last):\nE                 File \".../pytest_asyncio/plugin.py\", line 306, in setup\nE                   res = await func(**_add_kwargs(func, kwargs, event_loop, request))\nE               ResourceWarning: unclosed <ssl.SSLSocket fd=10, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 60605), raddr=('127.0.0.1', 6443)>\n\n.../_pytest/unraisableexception.py:78: PytestUnraisableExceptionWarning\n

    This error is saying that your test suite (or the prefect library code) opened a connection to something (like a Docker daemon or a Kubernetes cluster) and didn't close it.

    It may help to re-run the specific test with PYTHONTRACEMALLOC=25 pytest ... so that Python can display more of the stack trace where the connection was opened.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/upgrade-guide-agents-to-workers/","title":"Upgrade from Agents to Workers","text":"

    Upgrading from agents to workers significantly enhances the experience of deploying flows. It simplifies the specification of each flow's infrastructure and runtime environment.

    A worker is the fusion of an agent with an infrastructure block. Like agents, workers poll a work pool for flow runs that are scheduled to start. Like infrastructure blocks, workers are typed - they work with only one kind of infrastructure, and they specify the default configuration for jobs submitted to that infrastructure.

    Accordingly, workers are not a drop-in replacement for agents. Using workers requires deploying flows differently. In particular, deploying a flow with a worker does not involve specifying an infrastructure block. Instead, infrastructure configuration is specified on the work pool and passed to each worker that polls work from that pool.

    This guide provides an overview of the differences between agents and workers. It also describes how to upgrade from agents to workers in just a few quick steps.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#enhancements","title":"Enhancements","text":"","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#workers","title":"Workers","text":"
    • Improved visibility into the status of each worker, including when a worker was started and when it last polled.
    • Better handling of race conditions for high availability use cases.
    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#work-pools","title":"Work pools","text":"
    • Work pools allow greater customization and governance of infrastructure parameters for deployments via their base job template.
    • Prefect Cloud push work pools enable flow execution in your environment without needing to host a worker.
    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#improved-deployment-apis","title":"Improved deployment APIs","text":"
    • The new YAML-based and Python deployment APIs are more flexible and easier to use than block and agent-based deployments.
    • Both options allow you to deploy multiple flows with a single command.
    • Both options allow you to build Docker images for your flows to create portable execution environments.
    • The YAML-based API supports templating to enable dryer deployment definitions.
    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#whats-different","title":"What's different","text":"
    1. Deployment CLI and Python API:

      prefect deployment build <entrypoint>/prefect deployment apply --> prefect deploy

      Prefect will now automatically detect flows in your repo and provide a wizard \ud83e\uddd9 to guide you through setting required attributes for your deployments.

      Deployment.build_from_flow --> flow.deploy

    2. Configuring remote flow code storage:

      storage blocks --> pull action

      When using the YAML-based deployment API, you can configure a pull action in your prefect.yaml file to specify how to retrieve flow code for your deployments. You can use configuration from your existing storage blocks to define your pull action via templating.

      When using the Python deployment API, you can pass any storage block to the flow.deploy method to specify how to retrieve flow code for your deployment.

    3. Configuring flow run infrastructure:

      infrastructure blocks --> typed work pool

      Default infrastructure config is now set on the typed work pool, and can be overwritten by individual deployments.

    4. Managing multiple deployments:

      Create and/or update many deployments at once through a prefect.yaml file or use the deploy function.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#whats-similar","title":"What's similar","text":"
    • Storage blocks can be set as the pull action in a prefect.yaml file.
    • Infrastructure blocks have configuration fields similar to typed work pools.
    • Deployment-level infrastructure overrides operate in much the same way.

      infra_override -> job_variable

    • The process for starting an agent and starting a worker in your environment are virtually identical.

      prefect agent start --pool <work pool name> --> prefect worker start --pool <work pool name>

      Worker Helm chart

      If you host your agents in a Kubernetes cluster, you can use the Prefect worker Helm chart to host workers in your cluster.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#upgrade-guide","title":"Upgrade guide","text":"

    If you have existing deployments that use infrastructure blocks, you can quickly upgrade them to be compatible with workers by following these steps:

    1. Create a work pool

      This new work pool will replace your infrastructure block.

      You can use the .publish_as_work_pool method on any infrastructure block to create a work pool with the same configuration.

      For example, if you have a KubernetesJob infrastructure block named 'my-k8s-job', you can create a work pool with the same configuration with this script:

      from prefect.infrastructure import KubernetesJob\n\nKubernetesJob.load(\"my-k8s-job\").publish_as_work_pool()\n

      Running this script will create a work pool named 'my-k8s-job' with the same configuration as your infrastructure block.

      Serving flows

      If you are using a Process infrastructure block and a LocalFilesystem storage block (or aren't using an infrastructure and storage block at all), you can use flow.serve to create a deployment without needing to specify a work pool name or start a worker.

      This is a quick way to create a deployment for a flow and is a great way to manage your deployments if you don't need dynamic infrastructure creation or configuration offered by workers.

      Check out our Docker guide for how to build a served flow into a Docker image and host it in your environment.

    2. Start a worker

      This worker will replace your agent and poll your new work pool for flow runs to execute.

      prefect worker start -p <work pool name>\n

    3. Deploy your flows to the new work pool

      To deploy your flows to the new work pool, you can use flow.deploy for a Pythonic deployment experience or prefect deploy for a YAML-based deployment experience.

      If you currently use Deployment.build_from_flow, we recommend using flow.deploy.

      If you currently use prefect deployment build and prefect deployment apply, we recommend using prefect deploy.

      .deployprefect deploy

      If you have a Python script that uses Deployment.build_from_flow, you can replace it with flow.deploy.

      Most arguments to Deployment.build_from_flow can be translated directly to flow.deploy, but here are some changes that you may need to make:

      • Replace infrastructure with work_pool_name.
        • If you've used the .publish_as_work_pool method on your infrastructure block, use the name of the created work pool.
      • Replace infra_overrides with job_variables.
      • Replace storage with a call to flow.from_source.
        • flow.from_source will load your flow from a remote storage location and make it deployable. Your existing storage block can be passed to the source argument of flow.from_source.

      Below are some examples of how to translate Deployment.build_from_flow to flow.deploy.

      Always run prefect deploy commands from the root level of your repo!

      With agents, you might have had multiple deployment.yaml files, but under worker deployment patterns, each repo will have a single prefect.yaml file located at the root of the repo that contains deployment configuration for all flows in that repo.

      To set up a new prefect.yaml file for your deployments, run the following command from the root level of your repo:

      perfect deploy\n

      This will start a wizard that will guide you through setting up your first deployment.

      For step 4, select y on the last prompt to save the configuration for the deployment.

      Saving the configuration for your deployment will result in a prefect.yaml file populated with your first deployment. You can use this YAML file to edit and define multiple deployments for this repo.

      You can add more deployments to the deployments list in your prefect.yaml file and/or by continuing to use the deployment creation wizard.

    For more information, check out our in-depth guide for deploying flows to work pools.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#deploying-without-any-blocks","title":"Deploying without any blocks","text":"

    If you aren't using any blocks:

        from prefect import flow\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a Python script!\")\n\n    if __name__ == \"__main__\":\n        Deployment.build_from_flow(\n            my_flow,\n            name=\"my-deployment\",\n            parameters=dict(name=\"Marvin\"),\n        )\n

    You can replace Deployment.build_from_flow with flow.serve :

        from prefect import flow\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a Python script!\")\n\n    if __name__ == \"__main__\":\n        my_flow.serve(\n            name=\"my-deployment\",\n            parameters=dict(name=\"Marvin\"),\n        )\n

    This will start a process that will serve your flow and execute any flow runs that are scheduled to start.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#deploying-using-a-storage-block","title":"Deploying using a storage block","text":"

    If you currently use a storage block to load your flow code but no infrastructure block:

        from prefect import flow\n    from prefect.storage import GitHub\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a GitHub repo!\")\n\n    if __name__ == \"__main__\":\n        Deployment.build_from_flow(\n            my_flow,\n            name=\"my-deployment\",\n            storage=GitHub.load(\"demo-repo\"),\n            parameters=dict(name=\"Marvin\"),\n        )\n

    you can use flow.from_source to load your flow from the same location and flow.serve to create a deployment:

        from prefect import flow\n    from prefect.storage import GitHub\n\n    if __name__ == \"__main__\":\n        flow.from_source(\n            source=GitHub.load(\"demo-repo\"),\n            entrypoint=\"example.py:my_flow\"\n        ).serve(\n            name=\"my-deployment\",\n            parameters=dict(name=\"Marvin\"),\n        )\n

    This will allow you to execute scheduled flow runs without starting a worker. Additionally, the process serving your flow will regularly check for updates to your flow code and automatically update the flow if it detects any changes to the code.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#deploying-using-an-infrastructure-and-storage-block","title":"Deploying using an infrastructure and storage block","text":"

    For the code below, we'll need to create a work pool from our infrastructure block and pass it to flow.deploy as the work_pool_name argument. We'll also need to pass our storage block to flow.from_source as the source argument.

        from prefect import flow\n    from prefect.deployments import Deployment\n    from prefect.filesystems import GitHub\n    from prefect.infrastructure.kubernetes import KubernetesJob\n\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a GitHub repo!\")\n\n\n    if __name__ == \"__main__\":\n        Deployment.build_from_flow(\n            my_flow,\n            name=\"my-deployment\",\n            storage=GitHub.load(\"demo-repo\"),\n            entrypoint=\"example.py:my_flow\",\n            infrastructure=KubernetesJob.load(\"my-k8s-job\"),\n            infra_overrides=dict(pull_policy=\"Never\"),\n            parameters=dict(name=\"Marvin\"),\n        )\n

    The equivalent deployment code using flow.deploy would look like this:

        from prefect import flow\n    from prefect.storage import GitHub\n\n    if __name__ == \"__main__\":\n        flow.from_source(\n            source=GitHub.load(\"demo-repo\"),\n            entrypoint=\"example.py:my_flow\"\n        ).deploy(\n            name=\"my-deployment\",\n            work_pool_name=\"my-k8s-job\",\n            job_variables=dict(pull_policy=\"Never\"),\n            parameters=dict(name=\"Marvin\"),\n        )\n

    Note that when using flow.from_source(...).deploy(...), the flow you're deploying does not need to be available locally before running your script.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#deploying-via-a-docker-image","title":"Deploying via a Docker image","text":"

    If you currently bake your flow code into a Docker image before deploying, you can use the image argument of flow.deploy to build a Docker image as part of your deployment process:

        from prefect import flow\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a Docker image!\")\n\n\n    if __name__ == \"__main__\":\n        my_flow.deploy(\n            name=\"my-deployment\",\n            image=\"my-repo/my-image:latest\",\n            work_pool_name=\"my-k8s-job\",\n            job_variables=dict(pull_policy=\"Never\"),\n            parameters=dict(name=\"Marvin\"),\n        )\n

    You can skip a flow.from_source call when building an image with flow.deploy. Prefect will keep track of the flow's source code location in the image and load it from that location when the flow is executed.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/using-the-client/","title":"Using the Prefect Orchestration Client","text":"","tags":["client","API","filters","orchestration"]},{"location":"guides/using-the-client/#overview","title":"Overview","text":"

    In the API reference for the PrefectClient, you can find a bunch of useful client methods that make it simpler to do things like:

    • reschedule late flow runs
    • get the last N completed flow runs from my workspace

    The PrefectClient is an async context manager, so you can use it like this:

    from prefect import get_client\n\nasync with get_client() as client:\n    response = await client.hello()\n    print(response.json()) # \ud83d\udc4b\n

    ","tags":["client","API","filters","orchestration"]},{"location":"guides/using-the-client/#examples","title":"Examples","text":"","tags":["client","API","filters","orchestration"]},{"location":"guides/using-the-client/#rescheduling-late-flow-runs","title":"Rescheduling late flow runs","text":"

    Sometimes, you may need to bulk reschedule flow runs that are late - for example, if you've accidentally scheduled many flow runs of a deployment to an inactive work pool.

    To do this, we can delete late flow runs and create new ones in a Scheduled state with a delay.

    This example reschedules the last 3 late flow runs of a deployment named healthcheck-storage-test to run 6 hours later than their original expected start time. It also deletes any remaining late flow runs of that deployment.

    import asyncio\nfrom datetime import datetime, timedelta, timezone\nfrom typing import Optional\n\nfrom prefect import get_client\nfrom prefect.client.schemas.filters import (\n    DeploymentFilter, FlowRunFilter\n)\nfrom prefect.client.schemas.objects import FlowRun\nfrom prefect.client.schemas.sorting import FlowRunSort\nfrom prefect.states import Scheduled\n\nasync def reschedule_late_flow_runs(\n    deployment_name: str,\n    delay: timedelta,\n    most_recent_n: int,\n    delete_remaining: bool = True,\n    states: Optional[list[str]] = None\n) -> list[FlowRun]:\n    if not states:\n        states = [\"Late\"]\n\n    async with get_client() as client:\n        flow_runs = await client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=dict(name=dict(any_=states)),\n                expected_start_time=dict(\n                    before_=datetime.now(timezone.utc)\n                ),\n            ),\n            deployment_filter=DeploymentFilter(\n                name={'like_': deployment_name}\n            ),\n            sort=FlowRunSort.START_TIME_DESC,\n            limit=most_recent_n if not delete_remaining else None\n        )\n\n        if not flow_runs:\n            print(f\"No flow runs found in states: {states!r}\")\n            return []\n\n        rescheduled_flow_runs = []\n        for i, run in enumerate(flow_runs):\n            await client.delete_flow_run(flow_run_id=run.id)\n            if i < most_recent_n:\n                new_run = await client.create_flow_run_from_deployment(\n                    deployment_id=run.deployment_id,\n                    state=Scheduled(\n                        scheduled_time=run.expected_start_time + delay\n                    ),\n                )\n                rescheduled_flow_runs.append(new_run)\n\n        return rescheduled_flow_runs\n\nif __name__ == \"__main__\":\n    rescheduled_flow_runs = asyncio.run(\n        reschedule_late_flow_runs(\n            deployment_name=\"healthcheck-storage-test\",\n            delay=timedelta(hours=6),\n            most_recent_n=3,\n        )\n    )\n\n    print(f\"Rescheduled {len(rescheduled_flow_runs)} flow runs\")\n\n    assert all(\n        run.state.is_scheduled() for run in rescheduled_flow_runs\n    )\n    assert all(\n        run.expected_start_time > datetime.now(timezone.utc)\n        for run in rescheduled_flow_runs\n    )\n
    ","tags":["client","API","filters","orchestration"]},{"location":"guides/using-the-client/#get-the-last-n-completed-flow-runs-from-my-workspace","title":"Get the last N completed flow runs from my workspace","text":"

    To get the last N completed flow runs from our workspace, we can make use of read_flow_runs and prefect.client.schemas.

    This example gets the last three completed flow runs from our workspace:

    import asyncio\nfrom typing import Optional\n\nfrom prefect import get_client\nfrom prefect.client.schemas.filters import FlowRunFilter\nfrom prefect.client.schemas.objects import FlowRun\nfrom prefect.client.schemas.sorting import FlowRunSort\n\nasync def get_most_recent_flow_runs(\n    n: int = 3,\n    states: Optional[list[str]] = None\n) -> list[FlowRun]:\n    if not states:\n        states = [\"COMPLETED\"]\n\n    async with get_client() as client:\n        return await client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state={'type': {'any_': states}}\n            ),\n            sort=FlowRunSort.END_TIME_DESC,\n            limit=n,\n        )\n\nif __name__ == \"__main__\":\n    last_3_flow_runs: list[FlowRun] = asyncio.run(\n        get_most_recent_flow_runs()\n    )\n    print(last_3_flow_runs)\n\n    assert all(\n        run.state.is_completed() for run in last_3_flow_runs\n    )\n    assert (\n        end_times := [run.end_time for run in last_3_flow_runs]\n    ) == sorted(end_times, reverse=True)\n

    Instead of the last three from the whole workspace, you could also use the DeploymentFilter like the previous example to get the last three completed flow runs of a specific deployment.

    There are other ways to filter objects like flow runs

    See the filters API reference for more ways to filter flow runs and other objects in your Prefect ecosystem.

    ","tags":["client","API","filters","orchestration"]},{"location":"guides/variables/","title":"Variables","text":"

    Variables enable you to store and reuse non-sensitive bits of data, such as configuration information. Variables are named, mutable string values, much like environment variables. Variables are scoped to a Prefect server instance or a single workspace in Prefect Cloud.

    Variables can be created or modified at any time, but are intended for values with infrequent writes and frequent reads. Variable values may be cached for quicker retrieval.

    While variable values are most commonly loaded during flow runtime, they can be loaded in other contexts, at any time, such that they can be used to pass configuration information to Prefect configuration files, such as deployment steps.

    Variables are not Encrypted

    Using variables to store sensitive information, such as credentials, is not recommended. Instead, use Secret blocks to store and access sensitive information.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#managing-variables","title":"Managing variables","text":"

    You can create, read, edit and delete variables via the Prefect UI, API, and CLI. Names must adhere to traditional variable naming conventions:

    • Have no more than 255 characters.
    • Only contain lowercase alphanumeric characters ([a-z], [0-9]) or underscores (_). Spaces are not allowed.
    • be unique.

    Values must:

    • have less than or equal to 5000 characters.

    Optionally, you can add tags to the variable.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#via-the-prefect-ui","title":"Via the Prefect UI","text":"

    You can see all the variables in your Prefect server instance or Prefect Cloud workspace on the Variables page of the Prefect UI. Both the name and value of all variables are visible to anyone with access to the server or workspace.

    To create a new variable, select the + button next to the header of the Variables page. Enter the name and value of the variable.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#via-the-rest-api","title":"Via the REST API","text":"

    Variables can be created and deleted via the REST API. You can also set and get variables via the API with either the variable name or ID. See the REST reference for more information.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#via-the-cli","title":"Via the CLI","text":"

    You can list, inspect, and delete variables via the command line interface with the prefect variable ls, prefect variable inspect <name>, and prefect variable delete <name> commands, respectively.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#accessing-variables","title":"Accessing variables","text":"

    In addition to the UI and API, variables can be referenced in code and in certain Prefect configuration files.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#in-python-code","title":"In Python code","text":"

    You can access any variable via the Python SDK via the .get() method. If you attempt to reference a variable that does not exist, the method will return None.

    from prefect import variables\n\n# from a synchronous context\nanswer = variables.get('the_answer')\nprint(answer)\n# 42\n\n# from an asynchronous context\nanswer = await variables.get('the_answer')\nprint(answer)\n# 42\n\n# without a default value\nanswer = variables.get('not_the_answer')\nprint(answer)\n# None\n\n# with a default value\nanswer = variables.get('not_the_answer', default='42')\nprint(answer)\n# 42\n
    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#in-prefectyaml-deployment-steps","title":"In prefect.yaml deployment steps","text":"

    In .yaml files, variables are denoted by quotes and double curly brackets, like so: \"{{ prefect.variables.my_variable }}\". You can use variables to templatize deployment steps by referencing them in the prefect.yaml file used to create deployments. For example, you could pass a variable in to specify a branch for a git repo in a deployment pull step:

    pull:\n- prefect.deployments.steps.git_clone:\n    repository: https://github.com/PrefectHQ/hello-projects.git\n    branch: \"{{ prefect.variables.deployment_branch }}\"\n

    The deployment_branch variable will be evaluated at runtime for the deployed flow, allowing changes to be made to variables used in a pull action without updating a deployment directly.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/webhooks/","title":"Webhooks","text":"

    Use webhooks in your Prefect Cloud workspace to receive, observe, and react to events from other systems in your ecosystem. Each webhook exposes a unique URL endpoint to receive events from other systems and transforms them into Prefect events for use in automations.

    Webhooks are defined by two essential components: a unique URL and a template which translates incoming web requests to a Prefect event.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#configuring-webhooks","title":"Configuring webhooks","text":"","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#via-the-prefect-cloud-api","title":"Via the Prefect Cloud API","text":"

    Webhooks are managed via the Webhooks API endpoints. This is a Prefect Cloud-only feature. You authenticate API calls using the standard authentication methods you use with Prefect Cloud.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#via-prefect-cloud","title":"Via Prefect Cloud","text":"

    Webhooks can be created and managed from the Prefect Cloud UI.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#via-the-prefect-cli","title":"Via the Prefect CLI","text":"

    Webhooks can be managed and interacted with via the prefect cloud webhook command group.

    prefect cloud webhook --help\n

    You can create your first webhook by invoking create:

    prefect cloud webhook create your-webhook-name \\\n    --description \"Receives webhooks from your system\" \\\n    --template '{ \"event\": \"your.event.name\", \"resource\": { \"prefect.resource.id\": \"your.resource.id\" } }'\n

    Note the template string, which is discussed in greater detail down below

    You can retrieve details for a specific webhook by ID using get, or optionally query all webhooks in your workspace via ls:

    # get webhook by ID\nprefect cloud webhook get <webhook-id>\n\n# list all configured webhooks in your workspace\nprefect cloud webhook ls\n

    If you ever need to disable an existing webhook without deleting it, use toggle:

    prefect cloud webhook toggle <webhook-id>\nWebhook is now disabled\n\nprefect cloud webhook toggle <webhook-id>\nWebhook is now enabled\n

    If you are concerned that your webhook endpoint may have been compromised, use rotate to generate a new, random endpoint

    prefect cloud webhook rotate <webhook-url-slug>\n
    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#webhook-endpoints","title":"Webhook endpoints","text":"

    The webhook endpoints have randomly generated opaque URLs that do not divulge any information about your Prefect Cloud workspace. They are rooted at https://api.prefect.cloud/hooks/. For example: https://api.prefect.cloud/hooks/AERylZ_uewzpDx-8fcweHQ. Prefect Cloud assigns this URL when you create a webhook; it cannot be set via the API. You may rotate your webhook URL at any time without losing the associated configuration.

    All webhooks may accept requests via the most common HTTP methods:

    • GET, HEAD, and DELETE may be used for webhooks that define a static event template, or a template that does not depend on the body of the HTTP request. The headers of the request will be available for templates.
    • POST, PUT, and PATCH may be used when the webhook request will include a body. See How HTTP request components are handled for more details on how the body is parsed.

    Prefect Cloud webhooks are deliberately quiet to the outside world, and will only return a 204 No Content response when they are successful, and a 400 Bad Request error when there is any error interpreting the request. For more visibility when your webhooks fail, see the Troubleshooting section below.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#webhook-templates","title":"Webhook templates","text":"

    The purpose of a webhook is to accept an HTTP request from another system and produce a Prefect event from it. You may find that you often have little influence or control over the format of those requests, so Prefect's webhook system gives you full control over how you turn those notifications from other systems into meaningful events in your Prefect Cloud workspace. The template you define for each webhook will determine how individual components of the incoming HTTP request become the event name and resource labels of the resulting Prefect event.

    As with the templates available in Prefect Cloud Automation for defining notifications and other parameters, you will write templates in Jinja2. All of the built-in Jinja2 blocks and filters are available, as well as the filters from the jinja2-humanize-extensions package.

    Your goal when defining your event template is to produce a valid JSON object that defines (at minimum) the event name and the resource[\"prefect.resource.id\"], which are required of all events. The simplest template is one in which these are statically defined.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#static-webhook-events","title":"Static webhook events","text":"

    Let's see a static webhook template example. Say you want to configure a webhook that will notify Prefect when your recommendations machine learning model has been updated, so you can then send a Slack notification to your team and run a few subsequent deployments. Those models are produced on a daily schedule by another team that is using cron for scheduling. They aren't able to use Prefect for their flows (yet!), but they are happy to add a curl to the end of their daily script to notify you. Because this webhook will only be used for a single event from a single resource, your template can be entirely static:

    {\n    \"event\": \"model.refreshed\",\n    \"resource\": {\n        \"prefect.resource.id\": \"product.models.recommendations\",\n        \"prefect.resource.name\": \"Recommendations [Products]\",\n        \"producing-team\": \"Data Science\"\n    }\n}\n

    Make sure to produce valid JSON

    The output of your template, when rendered, should be a valid string that can be parsed, for example, with json.loads.

    A webhook with this template may be invoked via any of the HTTP methods, including a GET request with no body, so the team you are integrating with can include this line at the end of their daily script:

    curl https://api.prefect.cloud/hooks/AERylZ_uewzpDx-8fcweHQ\n

    Each time the script hits the webhook, the webhook will produce a single Prefect event with that name and resource in your workspace.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#event-fields-that-prefect-cloud-populates-for-you","title":"Event fields that Prefect Cloud populates for you","text":"

    You may notice that you only had to provide the event and resource definition, which is not a completely fleshed out event. Prefect Cloud will set default values for any missing fields, such as occurred and id, so you don't need to set them in your template. Additionally, Prefect Cloud will add the webhook itself as a related resource on all of the events it produces.

    If your template does not produce a payload field, the payload will default to a standard set of debugging information, including the HTTP method, headers, and body.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#dynamic-webhook-events","title":"Dynamic webhook events","text":"

    Now let's say that after a few days you and the Data Science team are getting a lot of value from the automations you have set up with the static webhook. You've agreed to upgrade this webhook to handle all of the various models that the team produces. It's time to add some dynamic information to your webhook template.

    Your colleagues on the team have adjusted their daily cron scripts to POST a small body that includes the ID and name of the model that was updated:

    curl \\\n    -d \"model=recommendations\" \\\n    -d \"friendly_name=Recommendations%20[Products]\" \\\n    -X POST https://api.prefect.cloud/hooks/AERylZ_uewzpDx-8fcweHQ\n

    This script will send a POST request and the body will include a traditional URL-encoded form with two fields describing the model that was updated: model and friendly_name. Here's the webhook code that uses Jinja to receive these values in your template and produce different events for the different models:

    {\n    \"event\": \"model.refreshed\",\n    \"resource\": {\n        \"prefect.resource.id\": \"product.models.{{ body.model }}\",\n        \"prefect.resource.name\": \"{{ body.friendly_name }}\",\n        \"producing-team\": \"Data Science\"\n    }\n}\n

    All subsequent POST requests will produce events with those variable resource IDs and names. The other statically-defined parts, such as event or the producing-team label you included earlier will still be used.

    Use Jinja2's default filter to handle missing values

    Jinja2 has a helpful default filter that can compensate for missing values in the request. In this example, you may want to use the model's ID in place of the friendly name when the friendly name is not provided: {{ body.friendly_name|default(body.model) }}.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#how-http-request-components-are-handled","title":"How HTTP request components are handled","text":"

    The Jinja2 template context includes the three parts of the incoming HTTP request:

    • method is the uppercased string of the HTTP method, like GET or POST.
    • headers is a case-insensitive dictionary of the HTTP headers included with the request. To prevent accidental disclosures, the Authorization header is removed.
    • body represents the body that was posted to the webhook, with a best-effort approach to parse it into an object you can access.

    HTTP headers are available without any alteration as a dict-like object, but you may access them with header names in any case. For example, these template expressions all return the value of the Content-Length header:

    {{ headers['Content-Length'] }}\n\n{{ headers['content-length'] }}\n\n{{ headers['CoNtEnt-LeNgTh'] }}\n

    The HTTP request body goes through some light preprocessing to make it more useful in templates. If the Content-Type of the request is application/json, the body will be parsed as a JSON object and made available to the webhook templates. If the Content-Type is application/x-www-form-urlencoded (as in our example above), the body is parsed into a flat dict-like object of key-value pairs. Jinja2 supports both index and attribute access to the fields of these objects, so the following two expressions are equivalent:

    {{ body['friendly_name'] }}\n\n{{ body.friendly_name }}\n

    Only for Python identifiers

    Jinja2's syntax only allows attribute-like access if the key is a valid Python identifier, so body.friendly-name will not work. Use body['friendly-name'] in those cases.

    You may not have much control over the client invoking your webhook, but would still like for bodies that look like JSON to be parsed as such. Prefect Cloud will attempt to parse any other content type (like text/plain) as if it were JSON first. In any case where the body cannot be transformed into JSON, it will be made available to your templates as a Python str.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#accepting-prefect-events-directly","title":"Accepting Prefect events directly","text":"

    In cases where you have more control over the client, your webhook can accept Prefect events directly with a simple pass-through template:

    {{ body|tojson }}\n

    This template accepts the incoming body (assuming it was in JSON format) and just passes it through unmodified. This allows a POST of a partial Prefect event as in this example:

    POST /hooks/AERylZ_uewzpDx-8fcweHQ HTTP/1.1\nHost: api.prefect.cloud\nContent-Type: application/json\nContent-Length: 228\n\n{\n    \"event\": \"model.refreshed\",\n    \"resource\": {\n        \"prefect.resource.id\": \"product.models.recommendations\",\n        \"prefect.resource.name\": \"Recommendations [Products]\",\n        \"producing-team\": \"Data Science\"\n    }\n}\n

    The resulting event will be filled out with the default values for occurred, id, and other fields as described above.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#accepting-cloudevents","title":"Accepting CloudEvents","text":"

    The Cloud Native Computing Foundation has standardized CloudEvents for use by systems to exchange event information in a common format. These events are supported by major cloud providers and a growing number of cloud-native systems. Prefect Cloud can interpret a webhook containing a CloudEvent natively with the following template:

    {{ body|from_cloud_event(headers) }}\n

    The resulting event will use the CloudEvent's subject as the resource (or the source if no subject is available). The CloudEvent's data attribute will become the Prefect event's payload['data'], and the other CloudEvent metadata will be at payload['cloudevents']. If you would like to handle CloudEvents in a more specific way tailored to your use case, use a dynamic template to interpret the incoming body.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#troubleshooting","title":"Troubleshooting","text":"

    The initial configuration of your webhook may require some trial and error as you get the sender and your receiving webhook speaking a compatible language. While you are in this phase, you may find the Event Feed in the UI to be indispensable for seeing the events as they are happening.

    When Prefect Cloud encounters an error during receipt of a webhook, it will produce a prefect-cloud.webhook.failed event in your workspace. This event will include critical information about the HTTP method, headers, and body it received, as well as what the template rendered. Keep an eye out for these events when something goes wrong.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/deployment/aci/","title":"Run an Agent with Azure Container Instances","text":"

    Microsoft Azure Container Instances (ACI) provides a convenient and simple service for quickly spinning up a Docker container that can host a Prefect Agent and execute flow runs.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#prerequisites","title":"Prerequisites","text":"

    To follow this quickstart, you'll need the following:

    • A Prefect Cloud account
    • A Prefect Cloud API key (Prefect Cloud Pro and Enterprise tier accounts can use a service account API key)
    • A Microsoft Azure account
    • Azure CLI installed and authenticated
    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#create-a-resource-group","title":"Create a resource group","text":"

    Like most Azure resources, ACI applications must live in a resource group. If you don\u2019t already have a resource group you\u2019d like to use, create a new one by running the az group create command. For example, this example creates a resource group called prefect-agents in the eastus region:

    az group create --name prefect-agents --location eastus\n

    Feel free to change the group name or location to match your use case. You can also run az account list-locations -o table to see all available resource group locations for your account.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#create-the-container-instance","title":"Create the container instance","text":"

    Prefect provides pre-configured Docker images you can use to quickly stand up a container instance. These Docker images include Python and Prefect. For example, the image prefecthq/prefect:2-python3.10 includes the latest release version of Prefect and Python 3.10.

    To create the container instance, use the az container create command. This example shows the syntax, but you'll need to provide the correct values for [ACCOUNT-ID],[WORKSPACE-ID], [API-KEY], and any dependencies you need to pip install on the instance. These options are discussed below.

    az container create \\\n--resource-group prefect-agents \\\n--name prefect-agent-example \\\n--image prefecthq/prefect:2-python3.10 \\\n--secure-environment-variables PREFECT_API_URL='https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]' PREFECT_API_KEY='[API-KEY]' \\\n--command-line \"/bin/bash -c 'pip install adlfs s3fs requests pandas; prefect agent start -p default-agent-pool -q test'\"\n

    When the container instance is running, go to Prefect Cloud and select the Work Pools page. Select default-agent-pool, then select the Queues tab to see work queues configured on this work pool. When the container instance is running and the agent has started, the test work queue displays \"Healthy\" status. This work queue and agent are ready to execute deployments configured to run on the test queue.

    Agents and queues

    The agent running in this container instance can now pick up and execute flow runs for any deployment configured to use the test queue on the default-agent-pool work pool.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#container-create-options","title":"Container create options","text":"

    Let's break down the details of the az container create command used here.

    The az container create command creates a new ACI container.

    --resource-group prefect-agents tells Azure which resource group the new container is created in. Here, the examples uses the prefect-agents resource group created earlier.

    --name prefect-agent-example determines the container name you will see in the Azure Portal. You can set any name you\u2019d like here to suit your use case, but container instance names must be unique in your resource group.

    --image prefecthq/prefect:2-python3.10 tells ACI which Docker images to run. The script above pulls a public Prefect image from Docker Hub. You can also build custom images and push them to a public container registry so ACI can access them. Or you can push your image to a private Azure Container Registry and use it to create a container instance.

    --secure-environment-variables sets environment variables that are only visible from inside the container. They do not show up when viewing the container\u2019s metadata. You'll populate these environment variables with a few pieces of information to configure the execution environment of the container instance so it can communicate with your Prefect Cloud workspace:

    • A Prefect Cloud [PREFECT_API_KEY]/concepts/settings/#prefect_api_key) value specifying the API key used to authenticate with your Prefect Cloud workspace. (Pro and Enterprise tier accounts can use a service account API key.)
    • The PREFECT_API_URL value specifying the API endpoint of your Prefect Cloud workspace.

    --command-line lets you override the container\u2019s normal entry point and run a command instead. The script above uses this section to install the adlfs pip package so it can read flow code from Azure Blob Storage, along with s3fs, pandas, and requests. It then runs the Prefect agent, in this case using the default work pool and a test work queue. If you want to use a different work pool or queue, make sure to change these values appropriately.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#create-a-deployment","title":"Create a deployment","text":"

    Following the example of the Flow deployments tutorial, let's create a deployment that can be executed by the agent on this container instance.

    In an environment where you have installed Prefect, create a new folder called health_test, and within it create a new file called health_flow.py containing the following code.

    import prefect\nfrom prefect import task, flow\nfrom prefect import get_run_logger\n\n\n@task\ndef say_hi():\n    logger = get_run_logger()\n    logger.info(\"Hello from the Health Check Flow! \ud83d\udc4b\")\n\n\n@task\ndef log_platform_info():\n    import platform\n    import sys\n    from prefect.server.api.server import SERVER_API_VERSION\n\n    logger = get_run_logger()\n    logger.info(\"Host's network name = %s\", platform.node())\n    logger.info(\"Python version = %s\", platform.python_version())\n    logger.info(\"Platform information (instance type) = %s \", platform.platform())\n    logger.info(\"OS/Arch = %s/%s\", sys.platform, platform.machine())\n    logger.info(\"Prefect Version = %s \ud83d\ude80\", prefect.__version__)\n    logger.info(\"Prefect API Version = %s\", SERVER_API_VERSION)\n\n\n@flow(name=\"Health Check Flow\")\ndef health_check_flow():\n    hi = say_hi()\n    log_platform_info(wait_for=[hi])\n

    Now create a deployment for this flow script, making sure that it's configured to use the test queue on the default-agent-pool work pool.

    prefect deployment build --infra process --storage-block azure/flowsville/health_test --name health-test --pool default-agent-pool --work-queue test --apply health_flow.py:health_check_flow\n

    Once created, any flow runs for this deployment will be picked up by the agent running on this container instance.

    Infrastructure and storage

    This Prefect deployment example was built using the Process infrastructure type and Azure Blob Storage.

    You might wonder why your deployment needs process infrastructure rather than DockerContainer infrastructure when you are deploying a Docker image to ACI.

    A Prefect deployment\u2019s infrastructure type describes how you want Prefect agents to run flows for the deployment. With DockerContainer infrastructure, the agent will try to use Docker to spin up a new container for each flow run. Since you\u2019ll be starting your own container on ACI, you don\u2019t need Prefect to do it for you. Specifying process infrastructure on the deployment tells Prefect you want to agent to run flows by starting a process in your ACI container.

    You can use any storage type as long as you've configured a block for it before creating the deployment.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#cleaning-up","title":"Cleaning up","text":"

    Note that ACI instances may incur usage charges while running, but must be running for the agent to pick up and execute flow runs.

    To stop a container, use the az container stop command:

    az container stop --resource-group prefect-agents --name prefect-agent-example\n

    To delete a container, use the az container delete command:

    az container delete --resource-group prefect-agents --name prefect-agent-example\n
    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/daemonize/","title":"Daemonize Processes for Prefect Deployments","text":"

    When running workflow applications, it can be helpful to create long-running processes that run at startup and are robust to failure. In this guide you'll learn how to set up a systemd service to create long-running Prefect processes that poll for scheduled flow runs.

    A systemd service is ideal for running a long-lived process on a Linux VM or physical Linux server. We will leverage systemd and see how to automatically start a Prefect worker or long-lived serve process when Linux starts. This approach provides resilience by automatically restarting the process if it crashes.

    In this guide we will:

    • Create a Linux user
    • Install and configure Prefect
    • Set up a systemd service for the Prefect worker or .serve process
    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#prerequisites","title":"Prerequisites","text":"
    • An environment with a linux operating system with systemd and Python 3.8 or later.
    • A superuser account (you can run sudo commands).
    • A Prefect Cloud account, or a local instance of a Prefect server running on your network.
    • If daemonizing a worker, you'll need a Prefect deployment with a work pool your worker can connect to.

    If using an AWS t2-micro EC2 instance with an AWS Linux image, you can install Python and pip with sudo yum install -y python3 python3-pip.

    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#step-1-add-a-user","title":"Step 1: Add a user","text":"

    Create a user account on your linux system for the Prefect process. While you can run a worker or serve process as root, it's good security practice to avoid doing so unless you are sure you need to.

    In a terminal, run:

    sudo useradd -m prefect\nsudo passwd prefect\n

    When prompted, enter a password for the prefect account.

    Next, log in to the prefect account by running:

    sudo su prefect\n
    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#step-2-install-prefect","title":"Step 2: Install Prefect","text":"

    Run:

    pip3 install prefect\n

    This guide assumes you are installing Prefect globally, not in a virtual environment. If running a systemd service in a virtual environment, you'll just need to change the ExecPath. For example, if using venv, change the ExecPath to target the prefect application in the bin subdirectory of your virtual environment.

    Next, set up your environment so that the Prefect client will know which server to connect to.

    If connecting to Prefect Cloud, follow the instructions to obtain an API key and then run the following:

    prefect cloud login -k YOUR_API_KEY\n

    When prompted, choose the Prefect workspace you'd like to log in to.

    If connecting to a self-hosted Prefect server instance instead of Prefect Cloud, run the following and substitute the IP address of your server:

    prefect config set PREFECT_API_URL=http://your-prefect-server-IP:4200\n

    Finally, run the exit command to sign out of the prefect Linux account. This command switches you back to your sudo-enabled account so you will can run the commands in the next section.

    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#step-3-set-up-a-systemd-service","title":"Step 3: Set up a systemd service","text":"Prefect worker.serve

    Move into the /etc/systemd/system folder and open a file for editing. We use the Vim text editor below.

    cd /etc/systemd/system\nsudo vim my-prefect-service.service\n
    my-prefect-service.service
    [Unit]\nDescription=Prefect worker\n\n[Service]\nUser=prefect\nWorkingDirectory=/home\nExecStart=prefect worker start --pool YOUR_WORK_POOL_NAME\nRestart=always\n\n[Install]\nWantedBy=multi-user.target\n

    Make sure you substitute your own work pool name.

    Copy your flow entrypoint Python file and any other files needed for your flow to run into the /home directory (or the directory of your choice).

    Here's a basic example flow:

    my_file.py
    from prefect import flow\n\n\n@flow(log_prints=True)\ndef say_hi():\n    print(\"Hello!\")\n\nif __name__==\"__main__\":\n    say_hi.serve(name=\"Greeting from daemonized .serve\")\n

    If you want to make changes to your flow code without restarting your process, you can push your code to git-based cloud storage (GitHub, BitBucket, GitLab) and use flow.from_source().serve(), as in the example below.

    my_remote_flow_code_file.py
    if __name__ == \"__main__\":\n  flow.from_source(\n      source=\"https://github.com/org/repo.git\",\n      entrypoint=\"path/to/my_remote_flow_code_file.py:say_hi\",\n  ).serve(name=\"deployment-with-github-storage\")\n

    Make sure you substitute your own flow code entrypoint path.

    Note that if you change the flow entrypoint parameters, you will need to restart the process.

    Move into the /etc/systemd/system folder and open a file for editing. We use the Vim text editor below.

    cd /etc/systemd/system\nsudo vim my-prefect-service.service\n
    my-prefect-service.service
    [Unit]\nDescription=Prefect serve \n\n[Service]\nUser=prefect\nWorkingDirectory=/home\nExecStart=python3 my_file.py\nRestart=always\n\n[Install]\nWantedBy=multi-user.target\n

    To save the file and exit Vim hit the escape key, type :wq!, then press the return key.

    Next, run sudo systemctl daemon-reload to make systemd aware of your new service.

    Then, run sudo systemctl enable my-prefect-service to enable the service. This command will ensure it runs when your system boots.

    Next, run sudo systemctl start my-prefect-service to start the service.

    Run your deployment from UI and check out the logs on the Flow Runs page.

    You can see if your daemonized Prefect worker or serve process is running and see the Prefect logs with systemctl status my-prefect-service.

    That's it! You now have a systemd service that starts when your system boots, and will restart if it ever crashes.

    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#next-steps","title":"Next steps","text":"

    If you want to set up a long-lived process on a Windows machine the pattern is similar. Instead of systemd, you can use NSSM.

    Check out other Prefect guides to see what else you can do with Prefect!

    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/","title":"Developing a New Worker Type","text":"

    Advanced Topic

    This tutorial is for users who want to extend the Prefect framework and completing this successfully will require deep knowledge of Prefect concepts. For standard use cases, we recommend using one of the available workers instead.

    Prefect workers are responsible for setting up execution infrastructure and starting flow runs on that infrastructure.

    A list of available workers can be found here. What if you want to execute your flow runs on infrastructure that doesn't have an available worker type? This tutorial will walk you through creating a custom worker that can run your flows on your chosen infrastructure.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#worker-configuration","title":"Worker configuration","text":"

    When setting up an execution environment for a flow run, a worker receives configuration for the infrastructure it is designed to work with. Examples of configuration values include memory allocation, CPU allocation, credentials, image name, etc. The worker then uses this configuration to create the execution environment and start the flow run.

    How are the configuration values populated?

    The work pool that a worker polls for flow runs has a base job template associated with it. The template is the contract for how configuration values populate for each flow run.

    The keys in the job_configuration section of this base job template match the worker's configuration class attributes. The values in the job_configuration section of the base job template are used to populate the attributes of the worker's configuration class.

    The work pool creator gets to decide how they want to populate the values in the job_configuration section of the base job template. The values can be hard-coded, templated using placeholders, or a mix of these two approaches. Because you, as the worker developer, don't know how the work pool creator will populate the values, you should set sensible defaults for your configuration class attributes as a matter of best practice.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#implementing-a-basejobconfiguration-subclass","title":"Implementing a BaseJobConfiguration subclass","text":"

    A worker developer defines their worker's configuration to function with a class extending BaseJobConfiguration.

    BaseJobConfiguration has attributes that are common to all workers:

    Attribute Description name The name to assign to the created execution environment. env Environment variables to set in the created execution environment. labels The labels assigned to the created execution environment for metadata purposes. command The command to use when starting a flow run.

    Prefect sets values for each attribute before giving the configuration to the worker. If you want to customize the values of these attributes, use the prepare_for_flow_run method.

    Here's an example prepare_for_flow_run method that adds a label to the execution environment:

    def prepare_for_flow_run(\n    self, flow_run, deployment = None, flow = None,\n):  \n    super().prepare_for_flow_run(flow_run, deployment, flow)  \n    self.labels.append(\"my-custom-label\")\n

    A worker configuration class is a Pydantic model, so you can add additional attributes to your configuration class as Pydantic fields. For example, if you want to allow memory and CPU requests for your worker, you can do so like this:

    from pydantic import Field\nfrom prefect.workers.base import BaseJobConfiguration\n\nclass MyWorkerConfiguration(BaseJobConfiguration):\n    memory: int = Field(\n            default=1024,\n            description=\"Memory allocation for the execution environment.\"\n        )\n    cpu: int = Field(\n            default=500, \n            description=\"CPU allocation for the execution environment.\"\n        )\n

    This configuration class will populate the job_configuration section of the resulting base job template.

    For this example, the base job template would look like this:

    job_configuration:\n    name: \"{{ name }}\"\n    env: \"{{ env }}\"\n    labels: \"{{ labels }}\"\n    command: \"{{ command }}\"\n    memory: \"{{ memory }}\"\n    cpu: \"{{ cpu }}\"\nvariables:\n    type: object\n    properties:\n        name:\n          title: Name\n          description: Name given to infrastructure created by a worker.\n          type: string\n        env:\n          title: Environment Variables\n          description: Environment variables to set when starting a flow run.\n          type: object\n          additionalProperties:\n            type: string\n        labels:\n          title: Labels\n          description: Labels applied to infrastructure created by a worker.\n          type: object\n          additionalProperties:\n            type: string\n        command:\n          title: Command\n          description: The command to use when starting a flow run. In most cases,\n            this should be left blank and the command will be automatically generated\n            by the worker.\n          type: string\n        memory:\n            title: Memory\n            description: Memory allocation for the execution environment.\n            type: integer\n            default: 1024\n        cpu:\n            title: CPU\n            description: CPU allocation for the execution environment.\n            type: integer\n            default: 500\n

    This base job template defines what values can be provided by deployment creators on a per-deployment basis and how those provided values will be translated into the configuration values that the worker will use to create the execution environment.

    Notice that each attribute for the class was added in the job_configuration section with placeholders whose name matches the attribute name. The variables section was also populated with the OpenAPI schema for each attribute. If a configuration class is used without explicitly declaring any template variables, the template variables will be inferred from the configuration class attributes.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#customizing-configuration-attribute-templates","title":"Customizing Configuration Attribute Templates","text":"

    You can customize the template for each attribute for situations where the configuration values should use more sophisticated templating. For example, if you want to add units for the memory attribute, you can do so like this:

    from pydantic import Field\nfrom prefect.workers.base import BaseJobConfiguration\n\nclass MyWorkerConfiguration(BaseJobConfiguration):\n    memory: str = Field(\n            default=\"1024Mi\",\n            description=\"Memory allocation for the execution environment.\"\n            template=\"{{ memory_request }}Mi\"\n        )\n    cpu: str = Field(\n            default=\"500m\", \n            description=\"CPU allocation for the execution environment.\"\n            template=\"{{ cpu_request }}m\"\n        )\n

    Notice that we changed the type of each attribute to str to accommodate the units, and we added a new template attribute to each attribute. The template attribute is used to populate the job_configuration section of the resulting base job template.

    For this example, the job_configuration section of the resulting base job template would look like this:

    job_configuration:\n    name: \"{{ name }}\"\n    env: \"{{ env }}\"\n    labels: \"{{ labels }}\"\n    command: \"{{ command }}\"\n    memory: \"{{ memory_request }}Mi\"\n    cpu: \"{{ cpu_request }}m\"\n

    Note that to use custom templates, you will need to declare the template variables used in the template because the names of those variables can no longer be inferred from the configuration class attributes. We will cover how to declare the default variable schema in the Worker Template Variables section.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#rules-for-template-variable-interpolation","title":"Rules for template variable interpolation","text":"

    When defining a job configuration model, it's useful to understand how template variables are interpolated into the job configuration. The templating engine follows a few simple rules:

    1. If a template variable is the only value for a key in the job_configuration section, the key will be replaced with the value template variable.
    2. If a template variable is part of a string (i.e., there is text before or after the template variable), the value of the template variable will be interpolated into the string.
    3. If a template variable is the only value for a key in the job_configuration section and no value is provided for the template variable, the key will be removed from the job_configuration section.

    These rules allow worker developers and work pool maintainers to define template variables that can be complex types like dictionaries and lists. These rules also mean that worker developers should give reasonable default values to job configuration fields whenever possible because values are not guaranteed to be provided if template variables are unset.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#template-variable-usage-strategies","title":"Template variable usage strategies","text":"

    Template variables define the interface that deployment creators interact with to configure the execution environments of their deployments. The complexity of this interface can be controlled via the template variables that are defined for a base job template. This control allows work pool maintainers to find a point along the spectrum of flexibility and simplicity appropriate for their organization.

    There are two patterns that are represented in current worker implementations:

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#pass-through","title":"Pass-through","text":"

    In the pass-through pattern, template variables are passed through to the job configuration with little change. This pattern exposes complete control to deployment creators but also requires them to understand the details of the execution environment.

    This pattern is useful when the execution environment is simple, and the deployment creators are expected to have high technical knowledge.

    The Docker worker is an example of a worker that uses this pattern.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#infrastructure-as-code-templating","title":"Infrastructure as code templating","text":"

    Depending on the infrastructure they interact with, workers can sometimes employ a declarative infrastructure syntax (i.e., infrastructure as code) to create execution environments (e.g., a Kubernetes manifest or an ECS task definition).

    In the IaC pattern, it's often useful to use template variables to template portions of the declarative syntax which then can be used to generate the declarative syntax into a final form.

    This approach allows work pool creators to provide a simpler interface to deployment creators while also controlling which portions of infrastructure are configurable by deployment creators.

    The Kubernetes worker is an example of a worker that uses this pattern.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#configuring-credentials","title":"Configuring credentials","text":"

    When executing flow runs within cloud services, workers will often need credentials to authenticate with those services. For example, a worker that executes flow runs in AWS Fargate will need AWS credentials. As a worker developer, you can use blocks to accept credentials configuration from the user.

    For example, if you want to allow the user to configure AWS credentials, you can do so like this:

    from prefect_aws import AwsCredentials\n\nclass MyWorkerConfiguration(BaseJobConfiguration):\n    aws_credentials: AwsCredentials = Field(\n        default=None,\n        description=\"AWS credentials to use when creating AWS resources.\"\n    )\n

    Users can create and assign a block to the aws_credentials attribute in the UI and the worker will use these credentials when interacting with AWS resources.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#worker-template-variables","title":"Worker template variables","text":"

    Providing template variables for a base job template defines the fields that deployment creators can override per deployment. The work pool creator ultimately defines the template variables for a base job template, but the worker developer is able to define default template variables for the worker to make it easier to use.

    Default template variables for a worker are defined by implementing the BaseVariables class. Like the BaseJobConfiguration class, the BaseVariables class has attributes that are common to all workers:

    Attribute Description name The name to assign to the created execution environment. env Environment variables to set in the created execution environment. labels The labels assigned the created execution environment for metadata purposes. command The command to use when starting a flow run.

    Additional attributes can be added to the BaseVariables class to define additional template variables. For example, if you want to allow memory and CPU requests for your worker, you can do so like this:

    from pydantic import Field\nfrom prefect.workers.base import BaseVariables\n\nclass MyWorkerTemplateVariables(BaseVariables):\n    memory_request: int = Field(\n            default=1024,\n            description=\"Memory allocation for the execution environment.\"\n        )\n    cpu_request: int = Field(\n            default=500, \n            description=\"CPU allocation for the execution environment.\"\n        )\n

    When MyWorkerTemplateVariables is used in conjunction with MyWorkerConfiguration from the Customizing Configuration Attribute Templates section, the resulting base job template will look like this:

    job_configuration:\n    name: \"{{ name }}\"\n    env: \"{{ env }}\"\n    labels: \"{{ labels }}\"\n    command: \"{{ command }}\"\n    memory: \"{{ memory_request }}Mi\"\n    cpu: \"{{ cpu_request }}m\"\nvariables:\n    type: object\n    properties:\n        name:\n          title: Name\n          description: Name given to infrastructure created by a worker.\n          type: string\n        env:\n          title: Environment Variables\n          description: Environment variables to set when starting a flow run.\n          type: object\n          additionalProperties:\n            type: string\n        labels:\n          title: Labels\n          description: Labels applied to infrastructure created by a worker.\n          type: object\n          additionalProperties:\n            type: string\n        command:\n          title: Command\n          description: The command to use when starting a flow run. In most cases,\n            this should be left blank and the command will be automatically generated\n            by the worker.\n          type: string\n        memory_request:\n            title: Memory Request\n            description: Memory allocation for the execution environment.\n            type: integer\n            default: 1024\n        cpu_request:\n            title: CPU Request\n            description: CPU allocation for the execution environment.\n            type: integer\n            default: 500\n

    Note that template variable classes are never used directly. Instead, they are used to generate a schema that is used to populate the variables section of a base job template and validate the template variables provided by the user.

    We don't recommend using template variable classes within your worker implementation for validation purposes because the work pool creator ultimately defines the template variables. The configuration class should handle any necessary run-time validation.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#worker-implementation","title":"Worker implementation","text":"

    Workers set up execution environments using provided configuration. Workers also observe the execution environment as the flow run executes and report any crashes to the Prefect API.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#attributes","title":"Attributes","text":"

    To implement a worker, you must implement the BaseWorker class and provide it with the following attributes:

    Attribute Description Required type The type of the worker. Yes job_configuration The configuration class for the worker. Yes job_configuration_variables The template variables class for the worker. No _documentation_url Link to documentation for the worker. No _logo_url Link to a logo for the worker. No _description A description of the worker. No","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#methods","title":"Methods","text":"","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#run","title":"run","text":"

    In addition to the attributes above, you must also implement a run method. The run method is called for each flow run the worker receives for execution from the work pool.

    The run method has the following signature:

     async def run(\n        self, flow_run: FlowRun, configuration: BaseJobConfiguration, task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> BaseWorkerResult:\n        ...\n

    The run method is passed: the flow run to execute, the execution environment configuration for the flow run, and a task status object that allows the worker to track whether the flow run was submitted successfully.

    The run method must also return a BaseWorkerResult object. The BaseWorkerResult object returned contains information about the flow run execution. For the most part, you can implement the BaseWorkerResult with no modifications like so:

    from prefect.workers.base import BaseWorkerResult\n\nclass MyWorkerResult(BaseWorkerResult):\n    \"\"\"Result returned by the MyWorker.\"\"\"\n

    If you would like to return more information about a flow run, then additional attributes can be added to the BaseWorkerResult class.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#kill_infrastructure","title":"kill_infrastructure","text":"

    Workers must implement a kill_infrastructure method to support flow run cancellation. The kill_infrastructure method is called when a flow run is canceled and is passed an identifier for the infrastructure to tear down and the execution environment configuration for the flow run.

    The infrastructure_pid passed to the kill_infrastructure method is the same identifier used to mark a flow run execution as started in the run method. The infrastructure_pid must be a string, but it can take on any format you choose.

    The infrastructure_pid should contain enough information to uniquely identify the infrastructure created for a flow run when used with the job_configuration passed to the kill_infrastructure method. Examples of useful information include: the cluster name, the hostname, the process ID, the container ID, etc.

    If a worker cannot tear down infrastructure created for a flow run, the kill_infrastructure command should raise an InfrastructureNotFound or InfrastructureNotAvailable exception.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#worker-implementation-example","title":"Worker implementation example","text":"

    Below is an example of a worker implementation. This example is not intended to be a complete implementation but to illustrate the aforementioned concepts.

    from prefect.workers.base import BaseWorker, BaseWorkerResult, BaseJobConfiguration, BaseVariables\n\nclass MyWorkerConfiguration(BaseJobConfiguration):\n    memory: str = Field(\n            default=\"1024Mi\",\n            description=\"Memory allocation for the execution environment.\"\n            template=\"{{ memory_request }}Mi\"\n        )\n    cpu: str = Field(\n            default=\"500m\", \n            description=\"CPU allocation for the execution environment.\"\n            template=\"{{ cpu_request }}m\"\n        )\n\nclass MyWorkerTemplateVariables(BaseVariables):\n    memory_request: int = Field(\n            default=1024,\n            description=\"Memory allocation for the execution environment.\"\n        )\n    cpu_request: int = Field(\n            default=500, \n            description=\"CPU allocation for the execution environment.\"\n        )\n\nclass MyWorkerResult(BaseWorkerResult):\n    \"\"\"Result returned by the MyWorker.\"\"\"\n\nclass MyWorker(BaseWorker):\n    type = \"my-worker\"\n    job_configuration = MyWorkerConfiguration\n    job_configuration_variables = MyWorkerTemplateVariables\n    _documentation_url = \"https://example.com/docs\"\n    _logo_url = \"https://example.com/logo\"\n    _description = \"My worker description.\"\n\n    async def run(\n        self, flow_run: FlowRun, configuration: BaseJobConfiguration, task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> BaseWorkerResult:\n        # Create the execution environment and start execution\n        job = await self._create_and_start_job(configuration)\n\n        if task_status:\n            # Use a unique ID to mark the run as started. This ID is later used to tear down infrastructure\n            # if the flow run is cancelled.\n            task_status.started(job.id) \n\n        # Monitor the execution\n        job_status = await self._watch_job(job, configuration)\n\n        exit_code = job_status.exit_code if job_status else -1 # Get result of execution for reporting\n        return MyWorkerResult(\n            status_code=exit_code,\n            identifier=job.id,\n        )\n\n    async def kill_infrastructure(self, infrastructure_pid: str, configuration: BaseJobConfiguration) -> None:\n        # Tear down the execution environment\n        await self._kill_job(infrastructure_pid, configuration)\n

    Most of the execution logic is omitted from the example above, but it shows that the typical order of operations in the run method is: 1. Create the execution environment and start the flow run execution 2. Mark the flow run as started via the passed task_status object 3. Monitor the execution 4. Get the execution's final status from the infrastructure and return a BaseWorkerResult object

    To see other examples of worker implementations, see the ProcessWorker and KubernetesWorker implementations.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#integrating-with-the-prefect-cli","title":"Integrating with the Prefect CLI","text":"

    Workers can be started via the Prefect CLI by providing the --type option to the prefect worker start CLI command. To make your worker type available via the CLI, it must be available at import time.

    If your worker is in a package, you can add an entry point to your setup file in the following format:

    entry_points={\n    \"prefect.collections\": [\n        \"my_package_name = my_worker_module\",\n    ]\n},\n

    Prefect will discover this entry point and load your work module in the specified module. The entry point will allow the worker to be available via the CLI.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/kubernetes/","title":"Running Flows with Kubernetes","text":"

    This guide will walk you through running your flows on Kubernetes. Though much of the guide is general to any Kubernetes cluster, there are differences between the managed Kubernetes offerings between cloud providers, especially when it comes to container registries and access management. We'll focus on Amazon Elastic Kubernetes Service (EKS).

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#prerequisites","title":"Prerequisites","text":"

    Before we begin, there are a few pre-requisites:

    1. A Prefect Cloud account
    2. A cloud provider (AWS, GCP, or Azure) account
    3. Install Python and Prefect
    4. Install Helm
    5. Install the Kubernetes CLI (kubectl)

    Administrator Access

    Though not strictly necessary, you may want to ensure you have admin access, both in Prefect Cloud and in your cloud provider. Admin access is only necessary during the initial setup and can be downgraded after.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-cluster","title":"Create a cluster","text":"

    Let's start by creating a new cluster. If you already have one, skip ahead to the next section.

    AWSGCPAzure

    One easy way to get set up with a cluster in EKS is with eksctl. Node pools can be backed by either EC2 instances or FARGATE. Let's choose FARGATE so there's less to manage. The following command takes around 15 minutes and must not be interrupted:

    # Replace the cluster name with your own value\neksctl create cluster --fargate --name <CLUSTER-NAME>\n\n# Authenticate to the cluster.\naws eks update-kubeconfig --name <CLUSTER-NAME>\n

    You can get a GKE cluster up and running with a few commands using the gcloud CLI. We'll build a bare-bones cluster that is accessible over the open internet - this should not be used in a production environment. To deploy the cluster, your project must have a VPC network configured.

    First, authenticate to GCP by setting the following configuration options.

    # Authenticate to gcloud\ngcloud auth login\n\n# Specify the project & zone to deploy the cluster to\n# Replace the project name with your GCP project name\ngcloud config set project <GCP-PROJECT-NAME>\ngcloud config set compute/zone <AVAILABILITY-ZONE>\n

    Next, deploy the cluster - this command will take ~15 minutes to complete. Once the cluster has been created, authenticate to the cluster.

    # Create cluster\n# Replace the cluster name with your own value\ngcloud container clusters create <CLUSTER-NAME> --num-nodes=1 \\\n--machine-type=n1-standard-2\n\n# Authenticate to the cluster\ngcloud container clusters <CLUSTER-NAME> --region <AVAILABILITY-ZONE>\n

    GCP Gotchas

    • You'll need to enable the default service account in the IAM console, or specify a different service account with the appropriate permissions to be used.
    ERROR: (gcloud.container.clusters.create) ResponseError: code=400, message=Service account \"000000000000-compute@developer.gserviceaccount.com\" is disabled.\n
    • Organization policy blocks creation of external (public) IPs. You can override this policy (if you have the appropriate permissions) under the Organizational Policy page within IAM.
    creation failed: Constraint constraints/compute.vmExternalIpAccess violated for project 000000000000. Add instance projects/<GCP-PROJECT-NAME>/zones/us-east1-b/instances/gke-gke-guide-1-default-pool-c369c84d-wcfl to the constraint to use external IP with it.\"\n

    You can quickly create an AKS cluster using the Azure CLI, or use the Cloud Shell directly from the Azure portal shell.azure.com.

    First, authenticate to Azure if not already done.

      az login\n

    Next, deploy the cluster - this command will take ~4 minutes to complete. Once the cluster has been created, authenticate to the cluster.

      # Create a Resource Group at the desired location, e.g. westus\n  az group create --name <RESOURCE-GROUP-NAME> --location <LOCATION>\n\n  # Create a kubernetes cluster with default kubernetes version, default SKU load balancer (Standard) and default vm set type (VirtualMachineScaleSets)\n  az aks create --resource-group <RESOURCE-GROUP-NAME> --name <CLUSTER-NAME>\n\n  # Configure kubectl to connect to your Kubernetes cluster\n  az aks get-credentials --resource-group <RESOURCE-GROUP-NAME> --name <CLUSTER-NAME>\n\n  # Verify the connection by listing the cluster nodes\n  kubectl get nodes\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-container-registry","title":"Create a container registry","text":"

    Besides a cluster, the other critical resource we'll need is a container registry. A registry is not strictly required, but in most cases you'll want to use custom images and/or have more control over where images are stored. If you already have a registry, skip ahead to the next section.

    AWSGCPAzure

    Let's create a registry using the AWS CLI and authenticate the docker daemon to said registry:

    # Replace the image name with your own value\naws ecr create-repository --repository-name <IMAGE-NAME>\n\n# Login to ECR\n# Replace the region and account ID with your own values\naws ecr get-login-password --region <REGION> | docker login \\\n  --username AWS --password-stdin <AWS_ACCOUNT_ID>.dkr.ecr.<REGION>.amazonaws.com\n

    Let's create a registry using the gcloud CLI and authenticate the docker daemon to said registry:

    # Create artifact registry repository to host your custom image\n# Replace the repository name with your own value; it can be the \n# same name as your image\ngcloud artifacts repositories create <REPOSITORY-NAME> \\\n--repository-format=docker --location=us\n\n# Authenticate to artifact registry\ngcloud auth configure-docker us-docker.pkg.dev\n

    Let's create a registry using the Azure CLI and authenticate the docker daemon to said registry:

    # Name must be a lower-case alphanumeric\n# Tier SKU can easily be updated later, e.g. az acr update --name <REPOSITORY-NAME> --sku Standard\naz acr create --resource-group <RESOURCE-GROUP-NAME> \\\n  --name <REPOSITORY-NAME> \\\n  --sku Basic\n\n# Attach ACR to AKS cluster\n# You need Owner, Account Administrator, or Co-Administrator role on your Azure subscription as per Azure docs\naz aks update --resource-group <RESOURCE-GROUP-NAME> --name <CLUSTER-NAME> --attach-acr <REPOSITORY-NAME>\n\n# You can verify AKS can now reach ACR\naz aks check-acr --resource-group RESOURCE-GROUP-NAME> --name <CLUSTER-NAME> --acr <REPOSITORY-NAME>.azurecr.io\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-kubernetes-work-pool","title":"Create a Kubernetes work pool","text":"

    Work pools allow you to manage deployment infrastructure. We'll configure the default values for our Kubernetes base job template. Note that these values can be overridden by individual deployments.

    Let's switch to the Prefect Cloud UI, where we'll create a new Kubernetes work pool (alternatively, you could use the Prefect CLI to create a work pool).

    1. Click on the Work Pools tab on the left sidebar
    2. Click the + button at the top of the page
    3. Select Kubernetes as the work pool type
    4. Click Next to configure the work pool settings

    Let's look at a few popular configuration options.

    Environment Variables

    Add environment variables to set when starting a flow run. So long as you are using a Prefect-maintained image and haven't overwritten the image's entrypoint, you can specify Python packages to install at runtime with {\"EXTRA_PIP_PACKAGES\":\"my_package\"}. For example {\"EXTRA_PIP_PACKAGES\":\"pandas==1.2.3\"} will install pandas version 1.2.3. Alternatively, you can specify package installation in a custom Dockerfile, which can allow you to take advantage of image caching. As we'll see below, Prefect can help us create a Dockerfile with our flow code and the packages specified in a requirements.txt file baked in.

    Namespace

    Set the Kubernetes namespace to create jobs within, such as prefect. By default, set to default.

    Image

    Specify the Docker container image for created jobs. If not set, the latest Prefect 2 image will be used (i.e. prefecthq/prefect:2-latest). Note that you can override this on each deployment through job_variables.

    Image Pull Policy

    Select from the dropdown options to specify when to pull the image. When using the IfNotPresent policy, make sure to use unique image tags, as otherwise old images could get cached on your nodes.

    Finished Job TTL

    Number of seconds before finished jobs are automatically cleaned up by Kubernetes' controller. You may want to set to 60 so that completed flow runs are cleaned up after a minute.

    Pod Watch Timeout Seconds

    Number of seconds for pod creation to complete before timing out. Consider setting to 300, especially if using a serverless type node pool, as these tend to have longer startup times.

    Kubernetes Cluster Config

    You can configure the Kubernetes cluster to use for job creation by specifying a KubernetesClusterConfig block. Generally you should leave the cluster config blank as the worker should be provisioned with appropriate access and permissions. Typically this setting is used when a worker is deployed to a cluster that is different from the cluster where flow runs are executed.

    Advanced Settings

    Want to modify the default base job template to add other fields or delete existing fields?

    Select the Advanced tab and edit the JSON representation of the base job template.

    For example, to set a CPU request, add the following section under variables:

    \"cpu_request\": {\n  \"title\": \"CPU Request\",\n  \"description\": \"The CPU allocation to request for this pod.\",\n  \"default\": \"default\",\n  \"type\": \"string\"\n},\n

    Next add the following to the first containers item under job_configuration:

    ...\n\"containers\": [\n  {\n    ...,\n    \"resources\": {\n      \"requests\": {\n        \"cpu\": \"{{ cpu_request }}\"\n      }\n    }\n  }\n],\n...\n

    Running deployments with this work pool will now request the specified CPU.

    After configuring the work pool settings, move to the next screen.

    Give the work pool a name and save.

    Our new Kubernetes work pool should now appear in the list of work pools.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-prefect-cloud-api-key","title":"Create a Prefect Cloud API key","text":"

    While in the Prefect Cloud UI, create a Prefect Cloud API key if you don't already have one. Click on your profile avatar picture, then click your name to go to your profile settings, click API Keys and hit the plus button to create a new API key here. Make sure to store it safely along with your other passwords, ideally via a password manager.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#deploy-a-worker-using-helm","title":"Deploy a worker using Helm","text":"

    With our cluster and work pool created, it's time to deploy a worker, which will set up Kubernetes infrastructure to run our flows. The best way to deploy a worker is using the Prefect Helm Chart.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#add-the-prefect-helm-repository","title":"Add the Prefect Helm repository","text":"

    Add the Prefect Helm repository to your Helm client:

    helm repo add prefect https://prefecthq.github.io/prefect-helm\nhelm repo update\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-namespace","title":"Create a namespace","text":"

    Create a new namespace in your Kubernetes cluster to deploy the Prefect worker:

    kubectl create namespace prefect\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-kubernetes-secret-for-the-prefect-api-key","title":"Create a Kubernetes secret for the Prefect API key","text":"
    kubectl create secret generic prefect-api-key \\\n--namespace=prefect --from-literal=key=your-prefect-cloud-api-key\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#configure-helm-chart-values","title":"Configure Helm chart values","text":"

    Create a values.yaml file to customize the Prefect worker configuration. Add the following contents to the file:

    worker:\n  cloudApiConfig:\n    accountId: <target account ID>\n    workspaceId: <target workspace ID>\n  config:\n    workPool: <target work pool name>\n

    These settings will ensure that the worker connects to the proper account, workspace, and work pool.

    View your Account ID and Workspace ID in your browser URL when logged into Prefect Cloud. For example: https://app.prefect.cloud/account/abc-my-account-id-is-here/workspaces/123-my-workspace-id-is-here.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-helm-release","title":"Create a Helm release","text":"

    Let's install the Prefect worker using the Helm chart with your custom values.yaml file:

    helm install prefect-worker prefect/prefect-worker \\\n  --namespace=prefect \\\n  -f values.yaml\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#verify-deployment","title":"Verify deployment","text":"

    Check the status of your Prefect worker deployment:

    kubectl get pods -n prefect\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#define-a-flow","title":"Define a flow","text":"

    Let's start simple with a flow that just logs a message. In a directory named flows, create a file named hello.py with the following contents:

    from prefect import flow, get_run_logger, tags\n\n@flow\ndef hello(name: str = \"Marvin\"):\n    logger = get_run_logger()\n    logger.info(f\"Hello, {name}!\")\n\nif __name__ == \"__main__\":\n    with tags(\"local\"):\n        hello()\n

    Run the flow locally with python hello.py to verify that it works. Note that we use the tags context manager to tag the flow run as local. This step is not required, but does add some helpful metadata.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#define-a-prefect-deployment","title":"Define a Prefect deployment","text":"

    Prefect has two recommended options for creating a deployment with dynamic infrastructure. You can define a deployment in a Python script using the flow.deploy mechanics or in a prefect.yaml definition file. The prefect.yaml file currently allows for more customization in terms of push and pull steps. Kubernetes objects are defined in YAML, so we expect many teams using Kubernetes work pools to create their deployments with YAML as well. To learn about the Python deployment creation method with flow.deploy refer to the Workers & Work Pools tutorial page.

    The prefect.yaml file is used by the prefect deploy command to deploy our flows. As a part of that process it will also build and push our image. Create a new file named prefect.yaml with the following contents:

    # Generic metadata about this project\nname: flows\nprefect-version: 2.13.8\n\n# build section allows you to manage and build docker images\nbuild:\n- prefect_docker.deployments.steps.build_docker_image:\n    id: build-image\n    requires: prefect-docker>=0.4.0\n    image_name: \"{{ $PREFECT_IMAGE_NAME }}\"\n    tag: latest\n    dockerfile: auto\n    platform: \"linux/amd64\"\n\n# push section allows you to manage if and how this project is uploaded to remote locations\npush:\n- prefect_docker.deployments.steps.push_docker_image:\n    requires: prefect-docker>=0.4.0\n    image_name: \"{{ build-image.image_name }}\"\n    tag: \"{{ build-image.tag }}\"\n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect.deployments.steps.set_working_directory:\n    directory: /opt/prefect/flows\n\n# the definitions section allows you to define reusable components for your deployments\ndefinitions:\n  tags: &common_tags\n    - \"eks\"\n  work_pool: &common_work_pool\n    name: \"kubernetes\"\n    job_variables:\n      image: \"{{ build-image.image }}\"\n\n# the deployments section allows you to provide configuration for deploying flows\ndeployments:\n- name: \"default\"\n  tags: *common_tags\n  schedule: null\n  entrypoint: \"flows/hello.py:hello\"\n  work_pool: *common_work_pool\n\n- name: \"arthur\"\n  tags: *common_tags\n  schedule: null\n  entrypoint: \"flows/hello.py:hello\"\n  parameters:\n    name: \"Arthur\"\n  work_pool: *common_work_pool\n

    We define two deployments of the hello flow: default and arthur. Note that by specifying dockerfile: auto, Prefect will automatically create a dockerfile that installs any requirements.txt and copies over the current directory. You can pass a custom Dockerfile instead with dockerfile: Dockerfile or dockerfile: path/to/Dockerfile. Also note that we are specifically building for the linux/amd64 platform. This specification is often necessary when images are built on Macs with M series chips but run on cloud provider instances.

    Deployment specific build, push, and pull

    The build, push, and pull steps can be overridden for each deployment. This allows for more custom behavior, such as specifying a different image for each deployment.

    Let's make sure we define our requirements in a requirements.txt file:

    prefect>=2.13.8\nprefect-docker>=0.4.0\nprefect-kubernetes>=0.3.1\n

    The directory should now look something like this:

    .\n\u251c\u2500\u2500 prefect.yaml\n\u2514\u2500\u2500 flows\n    \u251c\u2500\u2500 requirements.txt\n    \u2514\u2500\u2500 hello.py\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#tag-images-with-a-git-sha","title":"Tag images with a Git SHA","text":"

    If your code is stored in a GitHub repository, it's good practice to tag your images with the Git SHA of the code used to build it. This can be done in the prefect.yaml file with a few minor modifications, and isn't yet an option with the Python deployment creation method. Let's use the run_shell_script command to grab the SHA and pass it to the tag parameter of build_docker_image:

    build:\n- prefect.deployments.steps.run_shell_script:\n    id: get-commit-hash\n    script: git rev-parse --short HEAD\n    stream_output: false\n- prefect_docker.deployments.steps.build_docker_image:\n    id: build-image\n    requires: prefect-docker>=0.4.0\n    image_name: \"{{ $PREFECT_IMAGE_NAME }}\"\n    tag: \"{{ get-commit-hash.stdout }}\"\n    dockerfile: auto\n    platform: \"linux/amd64\"\n

    Let's also set the SHA as a tag for easy identification in the UI:

    definitions:\n  tags: &common_tags\n    - \"eks\"\n    - \"{{ get-commit-hash.stdout }}\"\n  work_pool: &common_work_pool\n    name: \"kubernetes\"\n    job_variables:\n      image: \"{{ build-image.image }}\"\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#authenticate-to-prefect","title":"Authenticate to Prefect","text":"

    Before we deploy the flows to Prefect, we will need to authenticate via the Prefect CLI. We will also need to ensure that all of our flow's dependencies are present at deploy time.

    This example uses a virtual environment to ensure consistency across environments.

    # Create a virtualenv & activate it\nvirtualenv prefect-demo\nsource prefect-demo/bin/activate\n\n# Install dependencies of your flow\nprefect-demo/bin/pip install -r requirements.txt\n\n# Authenticate to Prefect & select the appropriate \n# workspace to deploy your flows to\nprefect-demo/bin/prefect cloud login\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#deploy-the-flows","title":"Deploy the flows","text":"

    Now we're ready to deploy our flows which will build our images. The image name determines which registry it will end up in. We have configured our prefect.yaml file to get the image name from the PREFECT_IMAGE_NAME environment variable, so let's set that first:

    AWSGCPAzure
    export PREFECT_IMAGE_NAME=<AWS_ACCOUNT_ID>.dkr.ecr.<REGION>.amazonaws.com/<IMAGE-NAME>\n
    export PREFECT_IMAGE_NAME=us-docker.pkg.dev/<GCP-PROJECT-NAME>/<REPOSITORY-NAME>/<IMAGE-NAME>\n
    export PREFECT_IMAGE_NAME=<REPOSITORY-NAME>.azurecr.io/<IMAGE-NAME>\n

    To deploy your flows, ensure your Docker daemon is running first. Deploy all the flows with prefect deploy --all or deploy them individually by name: prefect deploy -n hello/default or prefect deploy -n hello/arthur.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#run-the-flows","title":"Run the flows","text":"

    Once the deployments are successfully created, we can run them from the UI or the CLI:

    prefect deployment run hello/default\nprefect deployment run hello/arthur\n

    Congratulations! You just ran two deployments in Kubernetes. Head over to the UI to check their status!

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/push-work-pools/","title":"Push Work to Serverless Computing Infrastructure","text":"

    Push work pools are a special type of work pool that allows Prefect Cloud to submit flow runs for execution to serverless computing infrastructure without running a worker. Push work pools currently support execution in GCP Cloud Run Jobs, Azure Container Instances, and AWS ECS Tasks.

    In this guide you will:

    • Create a push work pool that sends work to Google Cloud Run, Amazon Elastic Container Service (AWS ECS), or Azure Container Instances (ACI)
    • Deploy a flow to that work pool
    • Execute a flow without having to run a worker or agent process to poll for flow runs

    You can automatically provision infrastructure and create your push work pool using the prefect work-pool create CLI command with the --provision-infra flag. This approach greatly simplifies the setup process.

    Let's explore automatic infrastructure provisioning for push work pools first, and then we'll cover how to manually set up your push work pool.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#automatic-infrastructure-provisioning","title":"Automatic infrastructure provisioning","text":"

    With Perfect Cloud you can provision infrastructure for use with an AWS ECS, Google Cloud Run, ACI push work pool. Push work pools in Prefect Cloud simplify the setup and management of the infrastructure necessary to run your flows. However, setting up infrastructure on your cloud provider can still be a time-consuming process. Prefect can dramatically simplify this process by automatically provisioning the necessary infrastructure for you.

    We'll use the prefect work-pool create CLI command with the --provision-infra flag to automatically provision your serverless cloud resources and set up your Prefect workspace to use a new push pool.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#prerequisites","title":"Prerequisites","text":"

    To use automatic infrastructure provisioning, you'll need to have the relevant cloud CLI library installed and to have authenticated with your cloud provider.

    AWS ECSAzure Container InstancesGoogle Cloud Run

    Install the AWS CLI, authenticate with your AWS account, and set a default region.

    If you already have the AWS CLI installed, be sure to update to the latest version.

    You will need the following permissions in your authenticated AWS account:

    IAM Permissions:

    • iam:CreatePolicy
    • iam:GetPolicy
    • iam:ListPolicies
    • iam:CreateUser
    • iam:GetUser
    • iam:AttachUserPolicy
    • iam:CreateRole
    • iam:GetRole
    • iam:AttachRolePolicy
    • iam:ListRoles
    • iam:PassRole

    Amazon ECS Permissions:

    • ecs:CreateCluster
    • ecs:DescribeClusters

    Amazon EC2 Permissions:

    • ec2:CreateVpc
    • ec2:DescribeVpcs
    • ec2:CreateInternetGateway
    • ec2:AttachInternetGateway
    • ec2:CreateRouteTable
    • ec2:CreateRoute
    • ec2:CreateSecurityGroup
    • ec2:DescribeSubnets
    • ec2:CreateSubnet
    • ec2:DescribeAvailabilityZones
    • ec2:AuthorizeSecurityGroupIngress
    • ec2:AuthorizeSecurityGroupEgress

    Amazon ECR Permissions:

    • ecr:CreateRepository
    • ecr:DescribeRepositories
    • ecr:GetAuthorizationToken

    If you want to use AWS managed policies, you can use the following:

    • AmazonECS_FullAccess
    • AmazonEC2FullAccess
    • IAMFullAccess
    • AmazonEC2ContainerRegistryFullAccess

    Note that the above policies will give you all the permissions needed, but are more permissive than necessary.

    Docker is also required to build and push images to your registry. You can install Docker here.

    Install the Azure CLI and authenticate with your Azure account.

    If you already have the Azure CLI installed, be sure to update to the latest version with az upgrade.

    You will also need the following roles in your Azure subscription:

    • Contributor
    • User Access Administrator
    • Application Administrator
    • Managed Identity Operator
    • Azure Container Registry Contributor

    Docker is also required to build and push images to your registry. You can install Docker here.

    Install the gcloud CLI and authenticate with your GCP project.

    If you already have the gcloud CLI installed, be sure to update to the latest version with gcloud components update.

    You will also need the following permissions in your GCP project:

    • resourcemanager.projects.list
    • serviceusage.services.enable
    • iam.serviceAccounts.create
    • iam.serviceAccountKeys.create
    • resourcemanager.projects.setIamPolicy
    • artifactregistry.repositories.create

    Docker is also required to build and push images to your registry. You can install Docker here.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#automatically-creating-a-new-push-work-pool-and-provisioning-infrastructure","title":"Automatically creating a new push work pool and provisioning infrastructure","text":"

    Here's the command to create a new push work pool and configure the necessary infrastructure.

    AWS ECSAzure Container InstancesGoogle Cloud Run
    prefect work-pool create --type ecs:push --provision-infra my-ecs-pool\n

    Using the --provision-infra flag will automatically set up your default AWS account to be ready to execute flows via ECS tasks. In your AWS account, this command will create a new IAM user, IAM policy, ECS cluster that uses AWS Fargate, VPC, and ECR repository if they don't already exist. In your Prefect workspace, this command will create an AWSCredentials block for storing the generated credentials.

    Here's an abbreviated example output from running the command:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-ecs-pool will require:                                          \u2502\n\u2502                                                                                                                   \u2502\n\u2502          - Creating an IAM user for managing ECS tasks: prefect-ecs-user                                          \u2502\n\u2502          - Creating and attaching an IAM policy for managing ECS tasks: prefect-ecs-policy                        \u2502\n\u2502          - Storing generated AWS credentials in a block                                                           \u2502\n\u2502          - Creating an ECS cluster for running Prefect flows: prefect-ecs-cluster                                 \u2502\n\u2502          - Creating a VPC with CIDR 172.31.0.0/16 for running ECS tasks: prefect-ecs-vpc                          \u2502\n\u2502          - Creating an ECR repository for storing Prefect images: prefect-flows                                   \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]: y\nProvisioning IAM user\nCreating IAM policy\nGenerating AWS credentials\nCreating AWS credentials block\nProvisioning ECS cluster\nProvisioning VPC\nCreating internet gateway\nSetting up subnets\nSetting up security group\nProvisioning ECR repository\nAuthenticating with ECR\nSetting default Docker build namespace\nProvisioning Infrastructure \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned!\nCreated work pool 'my-ecs-pool'!\n

    Default Docker build namespace

    After infrastructure provisioning completes, you will be logged into your new ECR repository and the default Docker build namespace will be set to the URL of the registry.

    While the default namespace is set, you will not need to provide the registry URL when building images as part of your deployment process.

    To take advantage of this, you can write your deploy scripts like this:

    example_deploy_script.py
    from prefect import flow\nfrom prefect.deployments import DeploymentImage\n\n@flow(log_prints=True)            \ndef my_flow(name: str = \"world\"):                          \n    print(f\"Hello {name}! I'm a flow running in a ECS task!\") \n\n\nif __name__ == \"__main__\":\n    my_flow.deploy(\n        name=\"my-deployment\", \n        work_pool_name=\"my-work-pool\",\n        image=DeploymentImage(                                                 \n            name=\"my-repository:latest\",\n            platform=\"linux/amd64\",\n        )                                                                      \n    )       \n

    This will build an image with the tag <ecr-registry-url>/my-image:latest and push it to the registry.

    Your image name will need to match the name of the repository created with your work pool. You can create new repositories in the ECR console.

    prefect work-pool create --type azure-container-instance:push --provision-infra my-aci-pool\n

    Using the --provision-infra flag will automatically set up your default Azure account to be ready to execute flows via Azure Container Instances. In your Azure account, this command will create a resource group, app registration, service account with necessary permission, generate a secret for the app registration, and create an Azure Container Registry, if they don't already exist. In your Prefect workspace, this command will create an AzureContainerInstanceCredentials block for storing the client secret value from the generated secret.

    Here's an abbreviated example output from running the command:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-aci-work-pool will require:                     \u2502\n\u2502                                                                                           \u2502\n\u2502     Updates in subscription Azure subscription 1                                          \u2502\n\u2502                                                                                           \u2502\n\u2502         - Create a resource group in location eastus                                      \u2502\n\u2502         - Create an app registration in Azure AD prefect-aci-push-pool-app                \u2502\n\u2502         - Create/use a service principal for app registration                             \u2502\n\u2502         - Generate a secret for app registration                                          \u2502\n\u2502         - Create an Azure Container Registry with prefix prefect                          \u2502\n\u2502         - Create an identity prefect-acr-identity to allow access to the created registry \u2502\n\u2502         - Assign Contributor role to service account                                      \u2502\n\u2502         - Create an ACR registry for image hosting                                        \u2502\n\u2502         - Create an identity for Azure Container Instance to allow access to the registry \u2502\n\u2502                                                                                           \u2502\n\u2502     Updates in Prefect workspace                                                          \u2502\n\u2502                                                                                           \u2502\n\u2502         - Create Azure Container Instance credentials block aci-push-pool-credentials     \u2502\n\u2502                                                                                           \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]:     \nCreating resource group\nCreating app registration\nGenerating secret for app registration\nCreating ACI credentials block\nACI credentials block 'aci-push-pool-credentials' created in Prefect Cloud\nAssigning Contributor role to service account\nCreating Azure Container Registry\nCreating identity\nProvisioning infrastructure... \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned for 'my-aci-work-pool' work pool!\nCreated work pool 'my-aci-work-pool'!\n

    Default Docker build namespace

    After infrastructure provisioning completes, you will be logged into your new Azure Container Registry and the default Docker build namespace will be set to the URL of the registry.

    While the default namespace is set, any images you build without specifying a registry or username/organization will be pushed to the registry.

    To take advantage of this functionality, you can write your deploy scripts like this:

    example_deploy_script.py
    from prefect import flow                                                       \nfrom prefect.deployments import DeploymentImage                                \n\n\n@flow(log_prints=True)                                                         \ndef my_flow(name: str = \"world\"):                                              \n    print(f\"Hello {name}! I'm a flow running on an Azure Container Instance!\") \n\n\nif __name__ == \"__main__\":                                                     \n    my_flow.deploy(                                                            \n        name=\"my-deployment\",\n        work_pool_name=\"my-work-pool\",                                                \n        image=DeploymentImage(                                                 \n            name=\"my-image:latest\",                                            \n            platform=\"linux/amd64\",                                            \n        )                                                                      \n    )       \n

    This will build an image with the tag <acr-registry-url>/my-image:latest and push it to the registry.

    prefect work-pool create --type cloud-run:push --provision-infra my-cloud-run-pool \n

    Using the --provision-infra flag will allow you to select a GCP project to use for your work pool and automatically configure it to be ready to execute flows via Cloud Run. In your GCP project, this command will activate the Cloud Run API, create a service account, and create a key for the service account, if they don't already exist. In your Prefect workspace, this command will create a GCPCredentials block for storing the service account key.

    Here's an abbreviated example output from running the command:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-cloud-run-pool will require:                           \u2502\n\u2502                                                                                                          \u2502\n\u2502     Updates in GCP project central-kit-405415 in region us-central1                                      \u2502\n\u2502                                                                                                          \u2502\n\u2502         - Activate the Cloud Run API for your project                                                    \u2502\n\u2502         - Activate the Artifact Registry API for your project                                            \u2502\n\u2502         - Create an Artifact Registry repository named prefect-images                                    \u2502\n\u2502         - Create a service account for managing Cloud Run jobs: prefect-cloud-run                        \u2502\n\u2502             - Service account will be granted the following roles:                                       \u2502\n\u2502                 - Service Account User                                                                   \u2502\n\u2502                 - Cloud Run Developer                                                                    \u2502\n\u2502         - Create a key for service account prefect-cloud-run                                             \u2502\n\u2502                                                                                                          \u2502\n\u2502     Updates in Prefect workspace                                                                         \u2502\n\u2502                                                                                                          \u2502\n\u2502         - Create GCP credentials block my--pool-push-pool-credentials to store the service account key   \u2502\n\u2502                                                                                                          \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]: y\nActivating Cloud Run API\nActivating Artifact Registry API\nCreating Artifact Registry repository\nConfiguring authentication to Artifact Registry\nSetting default Docker build namespace\nCreating service account\nAssigning roles to service account\nCreating service account key\nCreating GCP credentials block\nProvisioning Infrastructure \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned!\nCreated work pool 'my-cloud-run-pool'!\n

    Default Docker build namespace

    After infrastructure provisioning completes, you will be logged into your new Artifact Registry repository and the default Docker build namespace will be set to the URL of the repository.

    While the default namespace is set, any images you build without specifying a registry or username/organization will be pushed to the repository.

    To take advantage of this functionality, you can write your deploy scripts like this:

    example_deploy_script.py
    from prefect import flow                                                       \nfrom prefect.deployments import DeploymentImage                                \n\n\n@flow(log_prints=True)\ndef my_flow(name: str = \"world\"):\n    print(f\"Hello {name}! I'm a flow running on Cloud Run!\")\n\n\nif __name__ == \"__main__\":                                                     \n    my_flow.deploy(                                                            \n        name=\"my-deployment\",\n        work_pool_name=\"above-ground\",\n        image=DeploymentImage(\n            name=\"my-image:latest\",\n            platform=\"linux/amd64\",\n        )\n    )\n

    This will build an image with the tag <region>-docker.pkg.dev/<project>/<repository-name>/my-image:latest and push it to the repository.

    That's it! You're ready to create and schedule deployments that use your new push work pool. Reminder that no worker is needed to run flows with a push work pool.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#using-existing-resources-with-automatic-infrastructure-provisioning","title":"Using existing resources with automatic infrastructure provisioning","text":"

    If you already have the necessary infrastructure set up, Prefect will detect that upon work pool creation and the infrastructure provisioning for that resource will be skipped.

    For example, here's how prefect work-pool create my-work-pool --provision-infra looks when existing Azure resources are detected:

    Proceed with infrastructure provisioning? [y/n]: y\nCreating resource group\nResource group 'prefect-aci-push-pool-rg' already exists in location 'eastus'.\nCreating app registration\nApp registration 'prefect-aci-push-pool-app' already exists.\nGenerating secret for app registration\nProvisioning infrastructure\nACI credentials block 'bb-push-pool-credentials' created\nAssigning Contributor role to service account...\nService principal with object ID '4be6fed7-...' already has the 'Contributor' role assigned in \n'/subscriptions/.../'\nCreating Azure Container Instance\nContainer instance 'prefect-aci-push-pool-container' already exists.\nCreating Azure Container Instance credentials block\nProvisioning infrastructure... \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned!\nCreated work pool 'my-work-pool'!\n
    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#provisioning-infrastructure-for-an-existing-push-work-pool","title":"Provisioning infrastructure for an existing push work pool","text":"

    If you already have a push work pool set up, but haven't configured the necessary infrastructure, you can use the provision-infra sub-command to provision the infrastructure for that work pool. For example, you can run the following command if you have a work pool named \"my-work-pool\".

    prefect work-pool provision-infra my-work-pool\n

    Prefect will create the necessary infrastructure for the my-work-pool work pool and provide you with a summary of the changes to be made:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-work-pool will require:                                      \u2502\n\u2502                                                                                                                \u2502\n\u2502     Updates in subscription Azure subscription 1                                                               \u2502\n\u2502                                                                                                                \u2502\n\u2502         - Create a resource group in location eastus                                                           \u2502\n\u2502         - Create an app registration in Azure AD prefect-aci-push-pool-app                                     \u2502\n\u2502         - Create/use a service principal for app registration                                                  \u2502\n\u2502         - Generate a secret for app registration                                                               \u2502\n\u2502         - Assign Contributor role to service account                                                           \u2502\n\u2502         - Create Azure Container Instance 'aci-push-pool-container' in resource group prefect-aci-push-pool-rg \u2502\n\u2502                                                                                                                \u2502\n\u2502     Updates in Prefect workspace                                                                               \u2502\n\u2502                                                                                                                \u2502\n\u2502         - Create Azure Container Instance credentials block aci-push-pool-credentials                          \u2502\n\u2502                                                                                                                \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]: y\n

    This command can speed up your infrastructure setup process.

    As with the examples above, you will need to have the related cloud CLI library installed and be authenticated with your cloud provider.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#manual-infrastructure-provisioning","title":"Manual infrastructure provisioning","text":"

    If you prefer to set up your infrastructure manually, don't include the --provision-infra flag in the CLI command. In the examples below, we'll create a push work pool via the Prefect Cloud UI.

    AWS ECSAzure Container InstancesGoogle Cloud Run

    To push work to ECS, AWS credentials are required.

    Create a user and attach the AmazonECS_FullAccess permissions.

    From that user's page create credentials and store them somewhere safe for use in the next section.

    To push work to Azure, an Azure subscription, resource group and tenant secret are required.

    Create Subscription and Resource Group

    1. In the Azure portal, create a subscription.
    2. Create a resource group within your subscription.

    Create App Registration

    1. In the Azure portal, create an app registration.
    2. In the app registration, create a client secret. Copy the value and store it somewhere safe.

    Add App Registration to Resource Group

    1. Navigate to the resource group you created earlier.
    2. Choose the \"Access control (IAM)\" blade in the left-hand side menu. Click \"+ Add\" button at the top, then \"Add role assignment\".
    3. Go to the \"Privileged administrator roles\" tab, click on \"Contributor\", then click \"Next\" at the bottom of the page.
    4. Click on \"+ Select members\". Type the name of the app registration (otherwise it may not autopopulate) and click to add it. Then hit \"Select\" and click \"Next\". The default permissions associated with a role like \"Contributor\" might not always be sufficient for all operations related to Azure Container Instances (ACI). The specific permissions required can depend on the operations you need to perform (like creating, running, and deleting ACI container groups) and your organization's security policies. In some cases, additional permissions or custom roles might be necessary.
    5. Click \"Review + assign\" to finish.

    A GCP service account and an API Key are required, to push work to Cloud Run.

    Create a service account by navigating to the service accounts page and clicking Create. Name and describe your service account, and click continue to configure permissions.

    The service account must have two roles at a minimum, Cloud Run Developer, and Service Account User.

    Once the Service account is created, navigate to its Keys page to add an API key. Create a JSON type key, download it, and store it somewhere safe for use in the next section.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#work-pool-configuration","title":"Work pool configuration","text":"

    Our push work pool will store information about what type of infrastructure our flow will run on, what default values to provide to compute jobs, and other important execution environment parameters. Because our push work pool needs to integrate securely with your serverless infrastructure, we need to start by storing our credentials in Prefect Cloud, which we'll do by making a block.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#creating-a-credentials-block","title":"Creating a Credentials block","text":"AWS ECSAzure Container InstancesGoogle Cloud Run

    Navigate to the blocks page, click create new block, and select AWS Credentials for the type.

    For use in a push work pool, this block must have the region and cluster name filled out, in addition to access key and access key secret.

    Provide any other optional information and create your block.

    Navigate to the blocks page and click the \"+\" at the top to create a new block. Find the Azure Container Instance Credentials block and click \"Add +\".

    Locate the client ID and tenant ID on your app registration and use the client secret you saved earlier. Be sure to use the value of the secret, not the secret ID!

    Provide any other optional information and click \"Create\".

    Navigate to the blocks page, click create new block, and select GCP Credentials for the type.

    For use in a push work pool, this block must have the contents of the JSON key stored in the Service Account Info field, as such:

    Provide any other optional information and create your block.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#creating-a-push-work-pool","title":"Creating a push work pool","text":"

    Now navigate to the work pools page. Click Create to start configuring your push work pool by selecting a push option in the infrastructure type step.

    AWS ECSAzure Container InstancesGoogle Cloud Run

    Each step has several optional fields that are detailed in the work pools documentation. Select the block you created under the AWS Credentials field. This will allow Prefect Cloud to securely interact with your ECS cluster.

    Fill in the subscription ID and resource group name from the resource group you created. Add the Azure Container Instance Credentials block you created in the step above.

    Each step has several optional fields that are detailed in the work pools documentation. Select the block you created under the GCP Credentials field. This will allow Prefect Cloud to securely interact with your GCP project.

    Create your pool and you are ready to deploy flows to your Push work pool.

    Push work pool concurrency

    Push work pools do not have a concurrency setting. If you would like to control concurrency at the flow level, you can use global concurrency limits.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#deployment","title":"Deployment","text":"

    Deployment details are described in the deployments concept section. Your deployment needs to be configured to send flow runs to our push work pool. For example, if you create a deployment through the interactive command line experience, choose the work pool you just created. If you are deploying an existing prefect.yaml file, the deployment would contain:

      work_pool:\n    name: my-push-pool\n

    Deploying your flow to the my-push-pool work pool will ensure that runs that are ready for execution will be submitted immediately, without the need for a worker to poll for them.

    Serverless infrastructure may require a certain image architecture

    Note that serverless infrastructure may assume a certain Docker image architecture; for example, Google Cloud Run will fail to run images built with linux/arm64 architecture. If using Prefect to build your image, you can change the image architecture through the platform keyword (e.g., platform=\"linux/amd64\").

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#putting-it-all-together","title":"Putting it all together","text":"

    With your deployment created, navigate to its detail page and create a new flow run. You'll see the flow start running without ever having to poll the work pool, because Prefect Cloud securely connected to your serverless infrastructure, created a job, ran the job, and began reporting on its execution.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/serverless-workers/","title":"Run Deployments on Serverless Infrastructure with Prefect Workers","text":"

    Prefect provides work pools for workers to run flows on the serverless platforms of major cloud providers. The following options are available:

    • AWS Elastic Container Service (ECS)
    • Azure Container Instances (ACI)
    • Google Cloud Run
    • Google Cloud Run V2
    • Google Vertex AI

    • Create a work pool that sends work to your chosen serverless infrastructure
    • Deploy a flow to that work pool
    • Start a worker in your serverless cloud provider that will poll its matched work pool for scheduled runs
    • Schedule a deployment run that a worker will pick up from the work pool and run on your serverless infrastructure

    Push work pools don't require a worker

    Options for push work pool versions of AWS ECS, Azure Container Instances, and Google Cloud Run that do not require a worker are available with Prefect Cloud. These push work pool options require connection configuration information to be stored on Prefect Cloud. Read more in the Serverless Push Work Pool Guide.

    This is a brief overview of the options to run workflows on serverless infrastructure. For in-depth guides, see the Prefect integration libraries:

    • AWS ECS guide in the prefect-aws docs
    • Azure Container Instances guide (forthcoming)
    • Google Cloud Run guide in the prefect-gcp docs.
    • For Google Vertex AI, follow the Cloud Run guide, substituting Google Vertex AI where Google Cloud Run is mentioned.

    Choosing between Google Cloud Run and Google Vertex AI

    Google Vertex AI is well-suited for machine learning model training applications in which GPUs or TPUs and high resource levels are desired.

    ","tags":["work pools","deployments","Cloud Run","GCP","Vertex AI","AWS ECS","Azure Container Instances","ACI"],"boost":2},{"location":"guides/deployment/serverless-workers/#steps","title":"Steps","text":"
    1. Make sure you have an user or service account on your chosen cloud provider with the necessary permissions to run serverless jobs
    2. Create the appropriate serverless work pool that uses a worker in the Prefect UI
    3. Create a deployment that references the work pool
    4. Start a worker in your chose serverless cloud provider infrastructure
    5. Run the deployment
    ","tags":["work pools","deployments","Cloud Run","GCP","Vertex AI","AWS ECS","Azure Container Instances","ACI"],"boost":2},{"location":"guides/deployment/serverless-workers/#next-steps","title":"Next steps","text":"

    Options for push versions on AWS ECS, Azure Container Instances, and Google Cloud Run work pools that do not require a worker are available with Prefect Cloud. Read more in the Serverless Push Work Pool Guide.

    Learn more about workers and work pools in the Prefect concept documentation.

    ","tags":["work pools","deployments","Cloud Run","GCP","Vertex AI","AWS ECS","Azure Container Instances","ACI"],"boost":2},{"location":"guides/deployment/storage-guide/","title":"Where to Store Your Flow Code","text":"

    When a flow runs, the execution environment needs access to its code. Flow code is not stored in a Prefect server database instance or Prefect Cloud. When deploying a flow, you have several flow code storage options.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#option-1-local-storage","title":"Option 1: Local storage","text":"

    Local flow code storage is often used with a Local Subprocess work pool for initial experimentation.

    To create a deployment with local storage and a Local Subprocess work pool, do the following:

    1. Run prefect deploy from the root of the directory containing your flow code.
    2. Select that you want to create a new deployment, select the flow code entrypoint, and name your deployment.
    3. Select a process work pool.

    You are then shown the location that your flow code will be fetched from when a flow is run. For example:

    Your Prefect workers will attempt to load your flow from: \n/my-path/my-flow-file.py. To see more options for managing your flow's code, run:\n\n    $ prefect init\n

    When deploying a flow to production, you most likely want code to run with infrastructure-specific configuration. The flow code storage options shown below are recommended for production deployments.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#option-2-git-based-storage","title":"Option 2: Git-based storage","text":"

    Git-based version control platforms are popular locations for code storage. They provide redundancy, version control, and easier collaboration.

    GitHub is the most popular cloud-based repository hosting provider. GitLab and Bitbucket are other popular options. Prefect supports each of these platforms.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#creating-a-deployment-with-git-based-storage","title":"Creating a deployment with git-based storage","text":"

    Run prefect deploy from within a git repository and create a new deployment. You will see a series of prompts. Select that you want to create a new deployment, select the flow code entrypoint, and name your deployment.

    Prefect detects that you are in a git repository and asks if you want to store your flow code in a git repository. Select \"y\" and you will be prompted to confirm the URL of your git repository and the branch name, as in the example below:

    ? Your Prefect workers will need access to this flow's code in order to run it. \nWould you like your workers to pull your flow code from its remote repository when running this flow? [y/n] (y): \n? Is https://github.com/my_username/my_repo.git the correct URL to pull your flow code from? [y/n] (y): \n? Is main the correct branch to pull your flow code from? [y/n] (y): \n? Is this a private repository? [y/n]: y\n

    In this example, the git repository is hosted on GitHub. If you are using Bitbucket or GitLab, the URL will match your provider. If the repository is public, enter \"n\" and you are on your way.

    If the repository is private, you can enter a token to access your private repository. This token will be saved in an encrypted Prefect Secret block.

    ? Please enter a token that can be used to access your private repository. This token will be saved as a Secret block via the Prefect API: \"123_abc_this_is_my_token\"\n

    Verify that you have a new Secret block in your active workspace named in the format \"deployment-my-deployment-my-flow-name-repo-token\".

    Creating access tokens differs for each provider.

    GitHubBitbucketGitLab

    We recommend using HTTPS with fine-grained Personal Access Tokens so that you can limit access by repository. See the GitHub docs for Personal Access Tokens (PATs).

    Under Your Profile->Developer Settings->Personal access tokens->Fine-grained token choose Generate New Token and fill in the required fields. Under Repository access choose Only select repositories and grant the token permissions for Contents.

    We recommend using HTTPS with Repository, Project, or Workspace Access Tokens.

    You can create a Repository Access Token with Scopes->Repositories->Read.

    Bitbucket requires you prepend the token string with x-token-auth: So the full string looks like x-token-auth:abc_123_this_is_my_token.

    We recommend using HTTPS with Project Access Tokens.

    In your repository in the GitLab UI, select Settings->Repository->Project Access Tokens and check read_repository under Select scopes.

    If you want to configure a Secret block ahead of time for use when deploying a prefect.yaml file, create the block via code or the Prefect UI and reference it like this:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://bitbucket.org/org/my-private-repo.git\n        access_token: \"{{ prefect.blocks.secret.my-block-name }}\"\n

    Alternatively, you can create a Credentials block ahead of time and reference it in the prefect.yaml pull step.

    GitHubBitbucketGitLab
    1. Install the Prefect-Github library with pip install -U prefect-github
    2. Register the blocks in that library to make them available on the server with prefect block register -m prefect_github.
    3. Create a GitHub Credentials block via code or the Prefect UI and reference it as shown above.
    4. In addition to the block name, most users will need to fill in the GitHub Username and GitHub Personal Access Token fields.
    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/discdiver/my-private-repo.git\n        credentials: \"{{ prefect.blocks.github-credentials.my-block-name }}\"\n
    1. Install the relevant library with pip install -U prefect-bitbucket
    2. Register the blocks in that library with prefect block register -m prefect_bitbucket
    3. Create a Bitbucket credentials block via code or the Prefect UI and reference it as shown above.
    4. In addition to the block name, most users will need to fill in the Bitbucket Username and Bitbucket Personal Access Token fields.
    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://bitbucket.org/org/my-private-repo.git\n        credentials: \"{{ prefect.blocks.bitbucket-credentials.my-block-name }}\"\n
    1. Install the relevant library with pip install -U prefect-gitlab
    2. Register the blocks in that library with prefect block register -m prefect_gitlab
    3. Create a GitLab credentials block via code or the Prefect UI and reference it as shown above.
    4. In addition to the block name, most users will need to fill in the GitLab Username and GitLab Personal Access Token fields.
    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://gitlab.com/org/my-private-repo.git\n        credentials: \"{{ prefect.blocks.gitlab-credentials.my-block-name }}\"\n

    Push your code

    When you make a change to your code, Prefect does not push your code to your git-based version control platform. You need to push your code manually or as part of your CI/CD pipeline. This design decision is an intentional one to avoid confusion about the git history and push process.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#option-3-docker-based-storage","title":"Option 3: Docker-based storage","text":"

    Another popular way to store your flow code is to include it in a Docker image. The following work pools use Docker containers, so the flow code can be directly baked into the image:

    • Docker
    • Kubernetes
    • Serverless cloud-based options
      • AWS Elastic Container Service
      • Azure Container Instances
      • Google Cloud Run
    • Push-based serverless cloud-based options (no worker required)

      • AWS Elastic Container Service - Push
      • Azure Container Instances - Push
      • Google Cloud Run - Push
    • Run prefect init in the root of your repository and choose docker for the project name and answer the prompts to create a prefect.yaml file with a build step that will create a Docker image with the flow code built in. See the Workers and Work Pools page of the tutorial for more info.

    • Run prefect deploy to create a deployment.
    • Upon deployment run the worker will pull the Docker image and spin up a container.
    • The flow code baked into the image will run inside the container.

    CI/CD may not require push or pull steps

    You don't need push or pull steps in the prefect.yaml file if using CI/CD to build a Docker image outside of Prefect. Instead, the work pool can reference the image directly.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#option-4-cloud-provider-storage","title":"Option 4: Cloud-provider storage","text":"

    You can store your code in an AWS S3 bucket, Azure Blob Storage container, or GCP GCS bucket and specify the destination directly in the push and pull steps of your prefect.yaml file.

    To create a templated prefect.yaml file run prefect init and select the recipe for the applicable cloud-provider storage. Below are the recipe options and the relevant portions of the prefect.yaml file.

    AWS S3 bucketAzure Blob Storage containerGCP GCS bucket

    Choose s3 as the recipe and enter the bucket name when prompted.

    # push section allows you to manage if and how this project is uploaded to remote locations\npush:\n- prefect_aws.deployments.steps.push_to_s3:\n    id: push_code\n    requires: prefect-aws>=0.3.4\n    bucket: my-bucket\n    folder: my-folder\n    credentials: \"{{ prefect.blocks.aws-credentials.my-credentials-block }}\" # if private\n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect_aws.deployments.steps.pull_from_s3:\n    id: pull_code\n    requires: prefect-aws>=0.3.4\n    bucket: '{{ push_code.bucket }}'\n    folder: '{{ push_code.folder }}'\n    credentials: \"{{ prefect.blocks.aws-credentials.my-credentials-block }}\" # if private \n

    If the bucket requires authentication to access it, you can do the following:

    1. Install the Prefect-AWS library with pip install -U prefect-aws
    2. Register the blocks in Prefect-AWS with prefect block register -m prefect_aws
    3. Create a user with a role with read and write permissions to access the bucket. If using the UI, create an access key pair with IAM->Users->Security credentials->Access keys->Create access key. Choose Use case->Other and then copy the Access key and Secret access key values.
    4. Create an AWS Credentials block via code or the Prefect UI. In addition to the block name, most users will fill in the AWS Access Key ID and AWS Access Key Secret fields.
    5. Reference the block as shown in the push and pull steps above.

    Choose azure as the recipe and enter the container name when prompted.

    # push section allows you to manage if and how this project is uploaded to remote locations\npush:\n- prefect_azure.deployments.steps.push_to_azure_blob_storage:\n    id: push_code\n    requires: prefect-azure>=0.2.8\n    container: my-prefect-azure-container\n    folder: my-folder\n    credentials: \"{{ prefect.blocks.azure-blob-storage-credentials.my-credentials-block }}\" # if private\n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect_azure.deployments.steps.pull_from_azure_blob_storage:\n    id: pull_code\n    requires: prefect-azure>=0.2.8\n    container: '{{ push_code.container }}'\n    folder: '{{ push_code.folder }}'\n    credentials: \"{{ prefect.blocks.azure-blob-storage-credentials.my-credentials-block }}\" # if private\n

    If the blob requires authentication to access it, you can do the following:

    1. Install the Prefect-Azure library with pip install -U prefect-azure
    2. Register the blocks in Prefect-Azure with prefect block register -m prefect_azure
    3. Create an access key for a role with sufficient (read and write) permissions to access the blob. A connection string that will contain all needed information can be created in the UI under Storage Account->Access keys.
    4. Create an Azure Blob Storage Credentials block via code or the Prefect UI. Enter a name for the block and paste the connection string into the Connection String field.
    5. Reference the block as shown in the push and pull steps above.

    Choose `gcs`` as the recipe and enter the bucket name when prompted.

    # push section allows you to manage if and how this project is uploaded to remote locations\npush:\n- prefect_gcp.deployment.steps.push_to_gcs:\n    id: push_code\n    requires: prefect-gcp>=0.4.3\n    bucket: my-bucket\n    folder: my-folder\n    credentials: \"{{ prefect.blocks.gcp-credentials.my-credentials-block }}\" # if private \n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect_gcp.deployment.steps.pull_from_gcs:\n    id: pull_code\n    requires: prefect-gcp>=0.4.3\n    bucket: '{{ push_code.bucket }}'\n    folder: '{{ pull_code.folder }}'\n    credentials: \"{{ prefect.blocks.gcp-credentials.my-credentials-block }}\" # if private \n

    If the bucket requires authentication to access it, you can do the following:

    1. Install the Prefect-GCP library with pip install -U prefect-gcp
    2. Register the blocks in Prefect-GCP with prefect block register -m prefect_gcp
    3. Create a service account in GCP for a role with read and write permissions to access the bucket contents. If using the GCP console, go to IAM & Admin->Service accounts->Create service account. After choosing a role with the required permissions, see your service account and click on the three dot menu in the Actions column. Select Manage Keys->ADD KEY->Create new key->JSON. Download the JSON file.
    4. Create a GCP Credentials block via code or the Prefect UI. Enter a name for the block and paste the entire contents of the JSON key file into the Service Account Info field.
    5. Reference the block as shown in the push and pull steps above.

    Another option for authentication is for the worker to have access to the the storage location at runtime via SSH keys.

    Alternatively, you can inject environment variables into your deployment like this example that uses an environment variable named CUSTOM_FOLDER:

     push:\n    - prefect_gcp.deployment.steps.push_to_gcs:\n        id: push_code\n        requires: prefect-gcp>=0.4.3\n        bucket: my-bucket\n        folder: '{{ $CUSTOM_FOLDER }}'\n
    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#including-and-excluding-files-from-storage","title":"Including and excluding files from storage","text":"

    By default, Prefect uploads all files in the current folder to the configured storage location when you create a deployment.

    When using a git repository, Docker image, or cloud-provider storage location, you may want to exclude certain files or directories. - If you are familiar with git you are likely familiar with the .gitignore file. - If you are familiar with Docker you are likely familiar with the .dockerignore file. - For cloud-provider storage the .prefectignore file serves the same purpose and follows a similar syntax as those files. So an entry of *.pyc will exclude all .pyc files from upload.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#other-code-storage-creation-methods","title":"Other code storage creation methods","text":"

    In earlier versions of Prefect storage blocks were the recommended way to store flow code. Storage blocks are still supported, but not recommended.

    As shown above, repositories can be referenced directly through interactive prompts with prefect deploy or in a prefect.yaml. When authentication is needed, Secret or Credential blocks can be referenced, and in some cases created automatically through interactive deployment creation prompts.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#next-steps","title":"Next steps","text":"

    You've seen options for where to store your flow code.

    We recommend using Docker-based storage or git-based storage for your production deployments.

    Check out more guides to reach your goals with Prefect.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"integrations/","title":"Integrations","text":"

    Prefect integrations are organized into collections of pre-built tasks, flows, blocks and more that are installable as PyPI packages.

    Airbyte

    Maintained by Prefect

    Alert

    Maintained by Khuyen Tran

    AWS

    Maintained by Prefect

    Azure

    Maintained by Prefect

    Bitbucket

    Maintained by Prefect

    Census

    Maintained by Prefect

    CubeJS

    Maintained by Alessandro Lollo

    Dask

    Maintained by Prefect

    Databricks

    Maintained by Prefect

    dbt

    Maintained by Prefect

    Docker

    Maintained by Prefect

    Earthdata

    Maintained by Giorgio Basile

    Email

    Maintained by Prefect

    Firebolt

    Maintained by Prefect

    Fivetran

    Maintained by Fivetran

    Fugue

    Maintained by The Fugue Development Team

    GCP

    Maintained by Prefect

    GitHub

    Maintained by Prefect

    GitLab

    Maintained by Prefect

    Google Sheets

    Maintained by Stefano Cascavilla

    Great Expectations

    Maintained by Prefect

    HashiCorp Vault

    Maintained by Pavel Chekin

    Hex

    Maintained by Prefect

    Hightouch

    Maintained by Prefect

    Jupyter

    Maintained by Prefect

    Kubernetes

    Maintained by Prefect

    KV

    Maintained by Michael Adkins

    MetricFlow

    Maintained by Alessandro Lollo

    Monday

    Maintained by Prefect

    MonteCarlo

    Maintained by Prefect

    OpenAI

    Maintained by Prefect

    OpenMetadata

    Maintained by Prefect

    Planetary Computer

    Maintained by Giorgio Basile

    Ray

    Maintained by Prefect

    Shell

    Maintained by Prefect

    Sifflet

    Maintained by Sifflet and Alessandro Lollo

    Slack

    Maintained by Prefect

    Snowflake

    Maintained by Prefect

    Soda Core

    Maintained by Soda and Alessandro Lollo

    Spark on Kubernetes

    Maintained by Manoj Babu Katragadda

    SQLAlchemy

    Maintained by Prefect

    Stitch

    Maintained by Alessandro Lollo

    Transform

    Maintained by Alessandro Lollo

    Twitter

    Maintained by Prefect

    ","tags":["tasks","flows","blocks","collections","task library","integrations","Airbyte","Alert","AWS","Azure","Bitbucket","Census","CubeJS","Dask","Databricks","dbt","Docker","Earthdata","Email","Firebolt","Fivetran","Fugue","GCP","GitHub","GitLab","Google Sheets","Great Expectations","HashiCorp Vault","Hex","Hightouch","Jupyter","Kubernetes","KV","MetricFlow","Monday","MonteCarlo","OpenAI","OpenMetadata","Planetary Computer","Ray","Shell","Sifflet","Slack","Snowflake","Soda Core","Spark on Kubernetes","SQLAlchemy","Stitch","Transform","Twitter"],"boost":2},{"location":"integrations/contribute/","title":"Contribute","text":"

    We welcome contributors! You can help contribute blocks and integrations by following these steps.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#contributing-blocks","title":"Contributing Blocks","text":"

    Building your own custom block is simple!

    1. Subclass from Block.
    2. Add a description alongside an Attributes and Example section in the docstring.
    3. Set a _logo_url to point to a relevant image.
    4. Create the pydantic.Fields of the block with a type annotation, default or default_factory, and a short description about the field.
    5. Define the methods of the block.

    For example, this is how the Secret block is implemented:

    from pydantic import Field, SecretStr\nfrom prefect.blocks.core import Block\n\nclass Secret(Block):\n    \"\"\"\n    A block that represents a secret value. The value stored in this block will be obfuscated when\n    this block is logged or shown in the UI.\n\n    Attributes:\n        value: A string value that should be kept secret.\n\n    Example:\n        ```python\n        from prefect.blocks.system import Secret\n        secret_block = Secret.load(\"BLOCK_NAME\")\n\n        # Access the stored secret\n        secret_block.get()\n        ```\n    \"\"\"\n\n    _logo_url = \"https://example.com/logo.png\"\n\n    value: SecretStr = Field(\n        default=..., description=\"A string value that should be kept secret.\"\n    )  # ... indicates it's a required field\n\n    def get(self):\n        return self.value.get_secret_value()\n

    To view in Prefect Cloud or the Prefect server UI, register the block.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#contributing-integrations","title":"Contributing Integrations","text":"

    Anyone can create and share a Prefect Integration and we encourage anyone interested in creating an integration to do so!

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#generate-a-project","title":"Generate a project","text":"

    To help you get started with your integration, we've created a template that gives the tools you need to create and publish your integration.

    Use the Prefect Integration template to get started creating an integration with a bootstrapped project!

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#list-a-project-in-the-integrations-catalog","title":"List a project in the Integrations Catalog","text":"

    To list your integration in the Prefect Integrations Catalog, submit a PR to the Prefect repository adding a file to the docs/integrations/catalog directory with details about your integration. Please use TEMPLATE.yaml in that folder as a guide.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#contribute-fixes-or-enhancements-to-integrations","title":"Contribute fixes or enhancements to Integrations","text":"

    If you'd like to help contribute to fix an issue or add a feature to any of our Integrations, please propose changes through a pull request from a fork of the repository.

    1. Fork the repository
    2. Clone the forked repository
    3. Install the repository and its dependencies:
      pip install -e \".[dev]\"\n
    4. Make desired changes
    5. Add tests
    6. Insert an entry to the Integration's CHANGELOG.md
    7. Install pre-commit to perform quality checks prior to commit:
      pre-commit install\n
    8. git commit, git push, and create a pull request
    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/usage/","title":"Using Integrations","text":"","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#installing-an-integration","title":"Installing an Integration","text":"

    Install the Integration via pip.

    For example, to use prefect-aws:

    pip install prefect-aws\n
    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#registering-blocks-from-an-integration","title":"Registering Blocks from an Integration","text":"

    Once the Prefect Integration is installed, register the blocks within the integration to view them in the Prefect Cloud UI:

    For example, to register the blocks available in prefect-aws:

    prefect block register -m prefect_aws\n

    Updating blocks from an integrations

    If you install an updated Prefect integration that adds fields to a block type, you will need to re-register that block type.

    Loading a block in code

    To use the load method on a Block, you must already have a block document saved either through code or through the Prefect UI.

    Learn more about Blocks here!

    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#using-tasks-and-flows-from-an-integration","title":"Using Tasks and Flows from an Integration","text":"

    Integrations also contain pre-built tasks and flows that can be imported and called within your code.

    As an example, to read a secret from AWS Secrets Manager with the read_secret task:

    from prefect import flow\nfrom prefect_aws import AwsCredentials\nfrom prefect_aws.secrets_manager import read_secret\n\n@flow\ndef connect_to_database():\n    aws_credentials = AwsCredentials.load(\"MY_BLOCK_NAME\")\n    secret_value = read_secret(\n        secret_name=\"db_password\",\n        aws_credentials=aws_credentials\n    )\n\n    # Use secret_value to connect to a database\n
    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#customizing-tasks-and-flows-from-an-integration","title":"Customizing Tasks and Flows from an Integration","text":"

    To customize the settings of a task or flow pre-configured in a collection, use with_options:

    from prefect import flow\nfrom prefect_dbt.cloud import DbtCloudCredentials\nfrom prefect_dbt.cloud.jobs import trigger_dbt_cloud_job_run_and_wait_for_completion\n\ncustom_run_dbt_cloud_job = trigger_dbt_cloud_job_run_and_wait_for_completion.with_options(\n    name=\"Run My DBT Cloud Job\",\n    retries=2,\n    retry_delay_seconds=10\n)\n\n@flow\ndef run_dbt_job_flow():\n    run_result = custom_run_dbt_cloud_job(\n        dbt_cloud_credentials=DbtCloudCredentials.load(\"my-dbt-cloud-credentials\"),\n        job_id=1\n    )\n\nrun_dbt_job_flow()\n
    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#recipes-and-tutorials","title":"Recipes and Tutorials","text":"

    To learn more about how to use Integrations, check out Prefect recipes on GitHub. These recipes provide examples of how Integrations can be used in various scenarios.

    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"recipes/recipes/","title":"Prefect Recipes","text":"

    Prefect recipes are common, extensible examples for setting up Prefect in your execution environment with ready-made ingredients such as Dockerfiles, Terraform files, and GitHub Actions.

    Recipes are useful when you are looking for tutorials on how to deploy a worker, use event-driven flows, set up unit testing, and more.

    The following are Prefect recipes specific to Prefect 2. You can find a full repository of recipes at https://github.com/PrefectHQ/prefect-recipes and additional recipes at Prefect Discourse.

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#recipe-catalog","title":"Recipe catalog","text":"Agent on Azure with Kubernetes

    Configure Prefect on Azure with Kubernetes, running a Prefect agent to execute deployment flow runs.

    Maintained by Prefect

    This recipe uses:

    Agent on ECS Fargate with AWS CLI

    Run a Prefect 2 agent on ECS Fargate using the AWS CLI.

    Maintained by Prefect

    This recipe uses:

    Agent on ECS Fargate with Terraform

    Run a Prefect 2 agent on ECS Fargate using Terraform.

    Maintained by Prefect

    This recipe uses:

    Agent on an Azure VM

    Set up an Azure VM and run a Prefect agent.

    Maintained by Prefect

    This recipe uses:

    Flow Deployment with GitHub Actions

    Deploy a Prefect flow with storage and infrastructure blocks, update and push Docker image to container registry.

    Maintained by Prefect

    This recipe uses:

    Flow Deployment with GitHub Storage and Docker Infrastructure

    Create a deployment with GitHub as a storage and Docker Container as an infrastructure

    Maintained by Prefect

    This recipe uses:

    Prefect server on an AKS Cluster

    Deploy a Prefect server to an Azure Kubernetes Service (AKS) Cluster with Azure Blob Storage.

    Maintained by Prefect

    This recipe uses:

    Serverless Prefect with AWS Chalice

    Execute Prefect flows in an AWS Lambda function managed by Chalice.

    Maintained by Prefect

    This recipe uses:

    Serverless Workflows with ECSTask Blocks

    Deploy a Prefect agent to AWS ECS Fargate using GitHub Actions and ECSTask infrastructure blocks.

    Maintained by Prefect

    This recipe uses:

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#contributing-recipes","title":"Contributing recipes","text":"

    We're always looking for new recipe contributions! See the Prefect Recipes repository for details on how you can add your Prefect recipe, share best practices with fellow Prefect users, and earn some swag.

    Prefect recipes provide a vital cookbook where users can find helpful code examples and, when appropriate, common steps for specific Prefect use cases.

    We love recipes from anyone who has example code that another Prefect user can benefit from (e.g. a Prefect flow that loads data into Snowflake).

    Have a blog post, Discourse article, or tutorial you\u2019d like to share as a recipe? All submissions are welcome. Clone the prefect-recipes repo, create a branch, add a link to your recipe to the README, and submit a PR. Have more questions? Read on.

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#what-is-a-recipe","title":"What is a recipe?","text":"

    A Prefect recipe is like a cookbook recipe: it tells you what you need \u2014 the ingredients \u2014 and some basic steps, but assumes you can put the pieces together. Think of the Hello Fresh meal experience, but for dataflows.

    A tutorial, on the other hand, is Julia Child holding your hand through the entire cooking process: explaining each ingredient and procedure, demonstrating best practices, pointing out potential problems, and generally making sure you can\u2019t stray from the happy path to a delicious meal.

    We love Julia, and we love tutorials. But we don\u2019t expect that a Prefect recipe should handhold users through every step and possible contingency of a solution. A recipe can start from an expectation of more expertise and problem-solving ability on the part of the reader.

    To see an example of a high quality recipe, check out Serverless with AWS Chalice. This recipe includes all of the elements we like to see.

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#steps-to-add-your-recipe","title":"Steps to add your recipe","text":"

    Here\u2019s our guide to creating a recipe:

    # Clone the repository\ngit clone git@github.com:PrefectHQ/prefect-recipes.git\ncd prefect-recipes\n\n# Create and checkout a new branch\n\ngit checkout -b new_recipe_branch_name\n
    1. Add your recipe. Your code may simply be a copy/paste of a single Python file or an entire folder. Unsure of where to add your file or folder? Just add under the flows-advanced/ folder. A Prefect Recipes maintainer will help you find the best place for your recipe. Just want to direct others to a project you made, whether it be a repo or a blogpost? Simply link to it in the Prefect Recipes README!
    2. (Optional) Write a README.
    3. Include a dependencies file, if applicable.
    4. Push your code and make a PR to the repository.

    That\u2019s it!

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#what-makes-a-good-recipe","title":"What makes a good recipe?","text":"

    Every recipe is useful, as other Prefect users can adapt the recipe to their needs. Particularly good ones help a Prefect user bake a great dataflow solution! Take a look at the prefect-recipes repo to see some examples.

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#what-are-the-common-ingredients-of-a-good-recipe","title":"What are the common ingredients of a good recipe?","text":"
    • Easy to understand: Can a user easily follow your recipe? Would a README or code comments help? A simple explanation providing context on how to use the example code is useful, but not required. A good README can set a recipe apart, so we have some additional suggestions for README files below.
    • Code and more: Sometimes a use case is best represented in Python code or shell scripts. Sometimes a configuration file is the most important artifact \u2014 think of a Dockerfile or Terraform file for configuring infrastructure.
    • All-inclusive: Share as much code as you can. Even boilerplate code like Dockerfiles or Terraform or Helm files are useful. Just don\u2019t share company secrets or IP.
    • Specific: Don't worry about generalizing your code, aside from removing anything internal/secret! Other users will extrapolate their own unique solutions from your example.
    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#what-are-some-tips-for-a-good-recipe-readme","title":"What are some tips for a good recipe README?","text":"

    A thoughtful README can take a recipe from good to great. Here are some best practices that we\u2019ve found make for a great recipe README:

    • Provide a brief explanation of what your recipe demonstrates. This helps users determine quickly whether the recipe is relevant to their needs or answers their questions.
    • List which files are included and what each is meant to do. Each explanation can contain only a few words.
    • Describe any dependencies and prerequisites (in addition to any dependencies you include in a requirements file). This includes both libraries or modules and any services your recipes depends on.
    • If steps are involved or there\u2019s an order to do things, a simple list of steps is helpful.
    • Bonus: troubleshooting steps you encountered to get here or tips where other users might get tripped up.
    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#next-steps","title":"Next steps","text":"

    We hope you\u2019ll feel comfortable sharing your Prefect solutions as recipes in the prefect-recipes repo. Collaboration and knowledge sharing are defining attributes of our Prefect Community!

    Have questions about sharing or using recipes? Reach out on our active Prefect Slack Community!

    Happy engineering!

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"tutorial/","title":"Tutorial Overview","text":"

    This tutorial provides a guided walk-through of Prefect core concepts and instructions on how to use them.

    By the end of this tutorial you will have:

    1. Created a flow
    2. Added tasks to it
    3. Deployed and run the flow locally
    4. Created a work pool and run the flow remotely

    These four topics will get most users to their first production deployment.

    Advanced users that need more governance and control of their workflow infrastructure can go one step further by:

    1. Using a worker-based deployment

    If you're looking for examples of more advanced operations (like deploying on Kubernetes), check out Prefect's guides.

    ","tags":["tutorial","getting started","basics","tasks","flows","subflows","deployments","workers","work pools"],"boost":2},{"location":"tutorial/#prerequisites","title":"Prerequisites","text":"
    1. Before you start, make sure you have Python installed, then install Prefect: pip install -U prefect

    See the install guide for more detailed instructions, if needed.

    1. To use Prefect, you need to self-host a Prefect server or connect to Prefect Cloud.

    To get the most out of this tutorial, we recommend using Prefect Cloud. Sign up for a forever free Prefect Cloud account or accept your organization's invite to join their Prefect Cloud account.

    1. Create a new account or sign in at https://app.prefect.cloud/.
    2. Use the prefect cloud login CLI command to authenticate to Prefect Cloud from your environment.
    prefect cloud login\n

    Choose Log in with a web browser and click the Authorize button in the browser window that opens.

    As an alternative to using Prefect Cloud, you can self-host a Prefect server instance. If you choose this option, run prefect server start to start a local Prefect server instance.

    ","tags":["tutorial","getting started","basics","tasks","flows","subflows","deployments","workers","work pools"],"boost":2},{"location":"tutorial/#what-is-prefect","title":"What is Prefect?","text":"

    Prefect orchestrates workflows \u2014 it simplifies the creation, scheduling, and monitoring of complex data pipelines. With Prefect, you define workflows as Python code and let it handle the rest.

    Prefect also provides error handling, retry mechanisms, and a user-friendly dashboard for monitoring. It's the easiest way to transform any Python function into a unit of work that can be observed and orchestrated.

    Just bring your Python code, sprinkle in a few decorators, and go!

    ","tags":["tutorial","getting started","basics","tasks","flows","subflows","deployments","workers","work pools"],"boost":2},{"location":"tutorial/#first-steps-flows","title":"First steps: Flows","text":"

    Let's begin by learning how to create your first Prefect flow - click here to get started.

    ","tags":["tutorial","getting started","basics","tasks","flows","subflows","deployments","workers","work pools"],"boost":2},{"location":"tutorial/deployments/","title":"Deploying Flows","text":"

    Reminder to connect to Prefect Cloud or a self-hosted Prefect server instance

    Some features in this tutorial, such as scheduling, require you to be connected to a Prefect server. If using a self-hosted setup, run prefect server start to run both the webserver and UI. If using Prefect Cloud, make sure you have successfully authenticated your local environment.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#why-deployments","title":"Why deployments?","text":"

    Some of the most common reasons to use an orchestration tool such as Prefect are for scheduling and event-based triggering. Up to this point, we\u2019ve demonstrated running Prefect flows as scripts, but this means you have been the one triggering and managing flow runs. You can certainly continue to trigger your workflows in this way and use Prefect as a monitoring layer for other schedulers or systems, but you will miss out on many of the other benefits and features that Prefect offers.

    Deploying a flow exposes an API and UI so that you can:

    • trigger new runs, cancel active runs, pause scheduled runs, customize parameters, and more
    • remotely configure schedules and automation rules for your deployments
    • dynamically provision infrastructure using workers
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#what-is-a-deployment","title":"What is a deployment?","text":"

    Deploying a flow is the act of specifying where and how it will run. This information is encapsulated and sent to Prefect as a deployment that contains the crucial metadata needed for remote orchestration. Deployments elevate workflows from functions that you call manually to API-managed entities.

    Attributes of a deployment include (but are not limited to):

    • Flow entrypoint: path to your flow function
    • Schedule or Trigger: optional schedule or triggering rule for this deployment
    • Tags: optional text labels for organizing your deployments
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#create-a-deployment","title":"Create a deployment","text":"

    Using our get_repo_info flow from the previous sections, we can easily create a deployment for it by calling a single method on the flow object: flow.serve.

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.serve(name=\"my-first-deployment\")\n

    Running this script will do two things:

    • create a deployment called \"my-first-deployment\" for your flow in the Prefect API
    • stay running to listen for flow runs for this deployment; when a run is found, it will be asynchronously executed within a subprocess

    Deployments must be defined in static files

    Flows can be defined and run interactively, that is, within REPLs or Notebooks. Deployments, on the other hand, require that your flow definition be in a known file (which can be located on a remote filesystem in certain setups, as we'll see in the next section of the tutorial).

    Because this deployment has no schedule or triggering automation, you will need to use the UI or API to create runs for it. Let's use the CLI (in a separate terminal window) to create a run for this deployment:

    prefect deployment run 'get-repo-info/my-first-deployment'\n

    If you are watching either your terminal or your UI, you should see the newly created run execute successfully! Let's take this example further by adding a schedule and additional metadata.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#additional-options","title":"Additional options","text":"

    The serve method on flows exposes many options for the deployment. Let's use a few of these options now:

    • cron: a keyword that allows us to set a cron string schedule for the deployment; see schedules for more advanced scheduling options
    • tags: a keyword that allows us to tag this deployment and its runs for bookkeeping and filtering purposes
    • description: a keyword that allows us to document what this deployment does; by default the description is set from the docstring of the flow function, but we did not document our flow function
    • version: a keyword that allows us to track changes to our deployment; by default a hash of the file containing the flow is used; popular options include semver tags or git commit hashes

    Let's add these options to our deployment:

    if __name__ == \"__main__\":\n    get_repo_info.serve(\n        name=\"my-first-deployment\",\n        cron=\"* * * * *\",\n        tags=[\"testing\", \"tutorial\"],\n        description=\"Given a GitHub repository, logs repository statistics for that repo.\",\n        version=\"tutorial/deployments\",\n    )\n

    When you rerun this script, you will find an updated deployment in the UI that is actively scheduling work! Stop the script in the CLI using CTRL+C and your schedule will be automatically paused.

    .serve is a long-running process

    For remotely triggered or scheduled runs to be executed, your script with flow.serve must be actively running.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#running-multiple-deployments-at-once","title":"Running multiple deployments at once","text":"

    This method is useful for creating deployments for single flows, but what if we have two or more flows? This situation only requires a few additional method calls and imports to get up and running:

    multi_flow_deployment.py
    import time\nfrom prefect import flow, serve\n\n\n@flow\ndef slow_flow(sleep: int = 60):\n    \"Sleepy flow - sleeps the provided amount of time (in seconds).\"\n    time.sleep(sleep)\n\n\n@flow\ndef fast_flow():\n    \"Fastest flow this side of the Mississippi.\"\n    return\n\n\nif __name__ == \"__main__\":\n    slow_deploy = slow_flow.to_deployment(name=\"sleeper\", interval=45)\n    fast_deploy = fast_flow.to_deployment(name=\"fast\")\n    serve(slow_deploy, fast_deploy)\n

    A few observations are in order:

    • the flow.to_deployment interface exposes the exact same options as flow.serve; this method produces a deployment object
    • the deployments are only registered with the API once serve(...) is called
    • when serving multiple deployments, the only requirement is that they share a Python environment; they can be executed and scheduled independently of each other

    Spend some time experimenting with this setup. A few potential next steps for exploration include:

    • pausing and unpausing the schedule for the \"sleeper\" deployment
    • using the UI to submit ad-hoc runs for the \"sleeper\" deployment with different values for sleep
    • cancelling an active run for the \"sleeper\" deployment from the UI (good luck cancelling the \"fast\" one \ud83d\ude09)

    Hybrid execution option

    Another implication of Prefect's deployment interface is that you can choose to use our hybrid execution model. Whether you use Prefect Cloud or host a Prefect server instance yourself, you can run work flows in the environments best suited to their execution. This model allows you efficient use of your infrastructure resources while maintaining the privacy of your code and data. There is no ingress required. For more information read more about our hybrid model.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#next-steps","title":"Next steps","text":"

    Congratulations! You now have your first working deployment.

    Deploying flows through the serve method is a fast way to start scheduling flows with Prefect. However, if your team has more complex infrastructure requirements or you'd like to have Prefect manage flow execution, you can deploy flows to a work pool.

    Learn about work pools and how Prefect Cloud can handle infrastructure configuration for you in the next step of the tutorial.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/flows/","title":"Flows","text":"

    Prerequisites

    This tutorial assumes you have already installed Prefect and connected to Prefect Cloud or a self-hosted server instance. See the prerequisites section of the tutorial for more details.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#what-is-a-flow","title":"What is a flow?","text":"

    Flows are like functions. They can take inputs, perform work, and return an output. In fact, you can turn any function into a Prefect flow by adding the @flow decorator. When a function becomes a flow, its behavior changes, giving it the following advantages:

    • All runs of the flow have persistent state. Transitions between states are recorded, allowing for flow execution to be observed and acted upon.
    • Input arguments can be type validated as workflow parameters.
    • Retries can be performed on failure.
    • Timeouts can be enforced to prevent unintentional, long-running workflows.
    • Metadata about flow runs, such as run time and final state, is automatically tracked.
    • They can easily be elevated to a deployment, which exposes a remote API for interacting with it
    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#run-your-first-flow","title":"Run your first flow","text":"

    The simplest way to get started with Prefect is to annotate a Python function with the\u00a0@flow\u00a0decorator. The script below fetches statistics about the main Prefect repository. Let's turn it into a Prefect flow and run it:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow\ndef get_repo_info():\n    url = \"https://api.github.com/repos/PrefectHQ/prefect\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(\"PrefectHQ/prefect repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\nif __name__ == \"__main__\":\n    get_repo_info()\n

    Running this file will result in some interesting output:

    12:47:42.792 | INFO | prefect.engine - Created flow run 'ludicrous-warthog' for flow 'get-repo-info'\nPrefectHQ/prefect repository statistics \ud83e\udd13:\nStars \ud83c\udf20 : 12146\nForks \ud83c\udf74 : 1245\n12:47:45.008 | INFO | Flow run 'ludicrous-warthog' - Finished in state Completed()\n

    Flows can contain arbitrary Python

    As we can see above, flow definitions can contain arbitrary Python logic.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#parameters","title":"Parameters","text":"

    As with any Python function, you can pass arguments to a flow. The positional and keyword arguments defined on your flow function are called parameters. Prefect will automatically perform type conversion using any provided type hints. Let's make the repository a string parameter with a default value:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info(repo_name=\"PrefectHQ/marvin\")\n

    We can call our flow with varying values for the repo_name parameter (including \"bad\" values):

    python repo_info.py\n

    Try passing repo_name=\"missing-org/missing-repo\".

    You should see

    HTTPStatusError: Client error '404 Not Found' for url '<https://api.github.com/repos/missing-org/missing-repo>'\n

    Now navigate to your Prefect dashboard and compare the displays for these two runs.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#logging","title":"Logging","text":"

    Prefect enables you to log a variety of useful information about your flow and task runs, capturing information about your workflows for purposes such as monitoring, troubleshooting, and auditing. If we navigate to our dashboard and explore the runs we created above, we will notice that the repository statistics are not captured in the flow run logs. Let's fix that by adding some logging to our flow:

    repo_info.py
    import httpx\nfrom prefect import flow, get_run_logger\n\n\n@flow\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    logger = get_run_logger()\n    logger.info(\"%s repository statistics \ud83e\udd13:\", repo_name)\n    logger.info(f\"Stars \ud83c\udf20 : %d\", repo[\"stargazers_count\"])\n    logger.info(f\"Forks \ud83c\udf74 : %d\", repo[\"forks_count\"])\n

    Now the output looks more consistent and, more importantly, our statistics are stored in the Prefect backend and displayed in the UI for this flow run:

    12:47:42.792 | INFO    | prefect.engine - Created flow run 'ludicrous-warthog' for flow 'get-repo-info'\n12:47:43.016 | INFO    | Flow run 'ludicrous-warthog' - PrefectHQ/prefect repository statistics \ud83e\udd13:\n12:47:43.016 | INFO    | Flow run 'ludicrous-warthog' - Stars \ud83c\udf20 : 12146\n12:47:43.042 | INFO    | Flow run 'ludicrous-warthog' - Forks \ud83c\udf74 : 1245\n12:47:45.008 | INFO    | Flow run 'ludicrous-warthog' - Finished in state Completed()\n

    log_prints=True

    We could have achieved the exact same outcome by using Prefect's convenient log_prints keyword argument in the flow decorator:

    @flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    ...\n

    Logging vs Artifacts

    The example above is for educational purposes. In general, it is better to use Prefect artifacts for storing metrics and output. Logs are best for tracking progress and debugging errors.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#retries","title":"Retries","text":"

    So far our script works, but in the future unexpected errors may occur. For example the GitHub API may be temporarily unavailable or rate limited. Retries help make our flow more resilient. Let's add retry functionality to our example above:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(retries=3, retry_delay_seconds=5, log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\nif __name__ == \"__main__\":\n    get_repo_info()\n
    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#next-tasks","title":"Next: Tasks","text":"

    As you have seen, adding a flow decorator converts our Python function to a resilient and observable workflow. In the next section, you'll supercharge this flow by using tasks to break down the workflow's complexity and make it more performant and observable - click here to continue.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/tasks/","title":"Tasks","text":"","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#what-is-a-task","title":"What is a task?","text":"

    A task is any Python function decorated with a @task decorator called within a flow. You can think of a flow as a recipe for connecting a known sequence of tasks together. Tasks, and the dependencies between them, are displayed in the flow run graph, enabling you to break down a complex flow into something you can observe, understand and control at a more granular level. When a function becomes a task, it can be executed concurrently and its return value can be cached.

    Flows and tasks share some common features:

    • Both are defined easily using their respective decorator, which accepts settings for that flow / task (see all task settings / flow settings).
    • Each can be given a name, description and tags for organization and bookkeeping.
    • Both provide functionality for retries, timeouts, and other hooks to handle failure and completion events.

    Network calls (such as our GET requests to the GitHub API) are particularly useful as tasks because they take advantage of task features such as retries, caching, and concurrency.

    Tasks must be called from flows

    All tasks must be called from within a flow. Tasks may not call other tasks directly.

    When to use tasks

    Not all functions in a flow need be tasks. Use them only when their features are useful.

    Let's take our flow from before and move the request into a task:

    repo_info.py
    import httpx\nfrom prefect import flow, task\n\n\n@task\ndef get_url(url: str, params: dict = None):\n    response = httpx.get(url, params=params)\n    response.raise_for_status()\n    return response.json()\n\n\n@flow(retries=3, retry_delay_seconds=5, log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    repo_stats = get_url(url)\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo_stats['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo_stats['forks_count']}\")\n\nif __name__ == \"__main__\":\n    get_repo_info()\n

    Running the flow in your terminal will result in something like this:

    09:55:55.412 | INFO    | prefect.engine - Created flow run 'great-ammonite' for flow 'get-repo-info'\n09:55:55.499 | INFO    | Flow run 'great-ammonite' - Created task run 'get_url-0' for task 'get_url'\n09:55:55.500 | INFO    | Flow run 'great-ammonite' - Executing 'get_url-0' immediately...\n09:55:55.825 | INFO    | Task run 'get_url-0' - Finished in state Completed()\n09:55:55.827 | INFO    | Flow run 'great-ammonite' - PrefectHQ/prefect repository statistics \ud83e\udd13:\n09:55:55.827 | INFO    | Flow run 'great-ammonite' - Stars \ud83c\udf20 : 12157\n09:55:55.827 | INFO    | Flow run 'great-ammonite' - Forks \ud83c\udf74 : 1251\n09:55:55.849 | INFO    | Flow run 'great-ammonite' - Finished in state Completed('All states completed.')\n

    And you should now see this task run tracked in the UI as well.

    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#caching","title":"Caching","text":"

    Tasks support the ability to cache their return value. Caching allows you to efficiently reuse results of tasks that may be expensive to reproduce with every flow run, or reuse cached results if the inputs to a task have not changed.

    To enable caching, specify a cache_key_fn \u2014 a function that returns a cache key \u2014 on your task. You may optionally provide a cache_expiration timedelta indicating when the cache expires. You can define a task that is cached based on its inputs by using the Prefect task_input_hash. Let's add caching to our get_url task:

    import httpx\nfrom datetime import timedelta\nfrom prefect import flow, task, get_run_logger\nfrom prefect.tasks import task_input_hash\n\n\n@task(cache_key_fn=task_input_hash, \n      cache_expiration=timedelta(hours=1),\n      )\ndef get_url(url: str, params: dict = None):\n    response = httpx.get(url, params=params)\n    response.raise_for_status()\n    return response.json()\n

    You can test this caching behavior by using a personal repository as your workflow parameter - give it a star, or remove a star and see how the output of this task changes (or doesn't) by running your flow multiple times.

    Task results and caching

    Task results are cached in memory during a flow run and persisted to your home directory by default. Prefect Cloud only stores the cache key, not the data itself.

    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#concurrency","title":"Concurrency","text":"

    Tasks enable concurrency, allowing you to execute multiple tasks asynchronously. This concurrency can greatly enhance the efficiency and performance of your workflows. Let's expand our script to calculate the average open issues per user. This will require making more requests:

    repo_info.py
    import httpx\nfrom datetime import timedelta\nfrom prefect import flow, task\nfrom prefect.tasks import task_input_hash\n\n\n@task(cache_key_fn=task_input_hash, cache_expiration=timedelta(hours=1))\ndef get_url(url: str, params: dict = None):\n    response = httpx.get(url, params=params)\n    response.raise_for_status()\n    return response.json()\n\n\ndef get_open_issues(repo_name: str, open_issues_count: int, per_page: int = 100):\n    issues = []\n    pages = range(1, -(open_issues_count // -per_page) + 1)\n    for page in pages:\n        issues.append(\n            get_url(\n                f\"https://api.github.com/repos/{repo_name}/issues\",\n                params={\"page\": page, \"per_page\": per_page, \"state\": \"open\"},\n            )\n        )\n    return [i for p in issues for i in p]\n\n\n@flow(retries=3, retry_delay_seconds=5, log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    repo_stats = get_url(f\"https://api.github.com/repos/{repo_name}\")\n    issues = get_open_issues(repo_name, repo_stats[\"open_issues_count\"])\n    issues_per_user = len(issues) / len(set([i[\"user\"][\"id\"] for i in issues]))\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo_stats['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo_stats['forks_count']}\")\n    print(f\"Average open issues per user \ud83d\udc8c : {issues_per_user:.2f}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info()\n

    Now we're fetching the data we need, but the requests are happening sequentially. Tasks expose a submit method that changes the execution from sequential to concurrent. In our specific example, we also need to use the result method because we are unpacking a list of return values:

    def get_open_issues(repo_name: str, open_issues_count: int, per_page: int = 100):\n    issues = []\n    pages = range(1, -(open_issues_count // -per_page) + 1)\n    for page in pages:\n        issues.append(\n            get_url.submit(\n                f\"https://api.github.com/repos/{repo_name}/issues\",\n                params={\"page\": page, \"per_page\": per_page, \"state\": \"open\"},\n            )\n        )\n    return [i for p in issues for i in p.result()]\n

    The logs show that each task is running concurrently:

    12:45:28.241 | INFO    | prefect.engine - Created flow run 'intrepid-coua' for flow 'get-repo-info'\n12:45:28.311 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-0' for task 'get_url'\n12:45:28.312 | INFO    | Flow run 'intrepid-coua' - Executing 'get_url-0' immediately...\n12:45:28.543 | INFO    | Task run 'get_url-0' - Finished in state Completed()\n12:45:28.583 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-1' for task 'get_url'\n12:45:28.584 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-1' for execution.\n12:45:28.594 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-2' for task 'get_url'\n12:45:28.594 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-2' for execution.\n12:45:28.609 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-4' for task 'get_url'\n12:45:28.610 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-4' for execution.\n12:45:28.624 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-5' for task 'get_url'\n12:45:28.625 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-5' for execution.\n12:45:28.640 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-6' for task 'get_url'\n12:45:28.641 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-6' for execution.\n12:45:28.708 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-3' for task 'get_url'\n12:45:28.708 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-3' for execution.\n12:45:29.096 | INFO    | Task run 'get_url-6' - Finished in state Completed()\n12:45:29.565 | INFO    | Task run 'get_url-2' - Finished in state Completed()\n12:45:29.721 | INFO    | Task run 'get_url-5' - Finished in state Completed()\n12:45:29.749 | INFO    | Task run 'get_url-4' - Finished in state Completed()\n12:45:29.801 | INFO    | Task run 'get_url-3' - Finished in state Completed()\n12:45:29.817 | INFO    | Task run 'get_url-1' - Finished in state Completed()\n12:45:29.820 | INFO    | Flow run 'intrepid-coua' - PrefectHQ/prefect repository statistics \ud83e\udd13:\n12:45:29.820 | INFO    | Flow run 'intrepid-coua' - Stars \ud83c\udf20 : 12159\n12:45:29.821 | INFO    | Flow run 'intrepid-coua' - Forks \ud83c\udf74 : 1251\nAverage open issues per user \ud83d\udc8c : 2.27\n12:45:29.838 | INFO    | Flow run 'intrepid-coua' - Finished in state Completed('All states completed.')\n
    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#subflows","title":"Subflows","text":"

    Not only can you call tasks within a flow, but you can also call other flows! Child flows are called\u00a0subflows\u00a0and allow you to efficiently manage, track, and version common multi-task logic.

    Subflows are a great way to organize your workflows and offer more visibility within the UI.

    Let's add a flow decorator to our get_open_issues function:

    @flow\ndef get_open_issues(repo_name: str, open_issues_count: int, per_page: int = 100):\n    issues = []\n    pages = range(1, -(open_issues_count // -per_page) + 1)\n    for page in pages:\n        issues.append(\n            get_url.submit(\n                f\"https://api.github.com/repos/{repo_name}/issues\",\n                params={\"page\": page, \"per_page\": per_page, \"state\": \"open\"},\n            )\n        )\n    return [i for p in issues for i in p.result()]\n

    Whenever we run the parent flow, a new run will be generated for related functions within that as well. Not only is this run tracked as a subflow run of the main flow, but you can also inspect it independently in the UI!

    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#next-deployments","title":"Next: Deployments","text":"

    We now have a flow with tasks, subflows, retries, logging, caching, and concurrent execution. In the next section, we'll see how we can deploy this flow in order to run it on a schedule and/or external infrastructure - click here to learn how to create your first deployment.

    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/work-pools/","title":"Work Pools","text":"","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#why-work-pools","title":"Why work pools?","text":"

    Work pools are a bridge between the Prefect orchestration layer and infrastructure for flow runs that can be dynamically provisioned. To transition from persistent infrastructure to dynamic infrastructure, use flow.deploy instead of flow.serve.

    Choosing Between flow.deploy() and flow.serve()

    Earlier in the tutorial you used serve to deploy your flows. For many use cases, serve is sufficient to meet scheduling and orchestration needs. Work pools are optional. If infrastructure needs escalate, work pools can become a handy tool. The best part? You're not locked into one method. You can seamlessly combine approaches as needed.

    Deployment definition methods differ slightly for work pools

    When you use work-pool-based execution, you define deployments differently. Deployments for workers are configured with deploy, which requires additional configuration. A deployment created with serve cannot be used with a work pool.

    The primary reason to use work pools is for dynamic infrastructure provisioning and configuration. For example, you might have a workflow that has expensive infrastructure requirements and is run infrequently. In this case, you don't want an idle process running within that infrastructure.

    Other advantages to using work pools include:

    • You can configure default infrastructure configurations on your work pools that all jobs inherit and can override.
    • Platform teams can use work pools to expose opinionated (and enforced!) interfaces to the infrastructure that they oversee.
    • Work pools can be used to prioritize (or limit) flow runs through the use of work queues.

    Prefect provides several types of work pools. Prefect Cloud provides a Prefect Managed work pool option that is the simplest way to run workflows remotely. A cloud-provider account, such as AWS, is not required with a Prefect Managed work pool.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#set-up-a-work-pool","title":"Set up a work pool","text":"

    Prefect Cloud

    This tutorial uses Prefect Cloud to deploy flows to work pools. Managed execution and push work pools are available in Prefect Cloud only. If you are not using Prefect Cloud, please learn about work pools below and then proceed to the next tutorial that uses worker-based work pools.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#create-a-prefect-managed-work-pool","title":"Create a Prefect Managed work pool","text":"

    In your terminal, run the following command to set up a work pool named my-managed-pool of type prefect:managed.

    prefect work-pool create my-managed-pool --type prefect:managed \n

    Let\u2019s confirm that the work pool was successfully created by running the following command.

    prefect work-pool ls\n

    You should see your new my-managed-pool in the output list.

    Finally, let\u2019s double check that you can see this work pool in the UI.

    Navigate to the Work Pools tab and verify that you see my-managed-pool listed.

    Feel free to select Edit from the three-dot menu on right of the work pool card to view the details of your work pool.

    Work pools contain configuration that is used to provision infrastructure for flow runs. For example, you can specify additional Python packages or environment variables that should be set for all deployments that use this work pool. Note that individual deployments can override the work pool configuration.

    Now that you\u2019ve set up your work pool, we can deploy a flow to this work pool. Let's deploy your tutorial flow to my-managed-pool.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#create-the-deployment","title":"Create the deployment","text":"

    From our previous steps, we now have:

    1. A flow
    2. A work pool

    Let's update our repo_info.py file to create a deployment in Prefect Cloud.

    The updates that we need to make to repo_info.py are:

    1. Change flow.serve to flow.deploy.
    2. Tell flow.deploy which work pool to deploy to.

    Here's what the updated repo_info.py looks like:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.from_source(\n        source=\"https://github.com/discdiver/demos.git\", \n        entrypoint=\"repo_info.py:get_repo_info\"\n    ).deploy(\n        name=\"my-first-deployment\", \n        work_pool_name=\"my-managed-pool\", \n    )\n

    In the from_source method, we specify the source of our flow code.

    In the deploy method, we specify the name of our deployment and the name of the work pool that we created earlier.

    You can store your flow code in any of several types of remote storage. In this example, we use a GitHub repository, but you could use a Docker image, as you'll see in an upcoming section of the tutorial. Alternatively, you could store your flow code in cloud provider storage such as AWS S3, or within a different git-based cloud provider such as GitLab or Bitbucket.

    Note

    In the example above, we store our code in a GitHub repository. If you make changes to the flow code, you will need to push those changes to your own GitHub account and update the source argument of from_source to point to your repository.

    Run the script again and you should see a message in the CLI that your deployment was created with instructions for how to run it.

    Successfully created/updated all deployments!\n\n                       Deployments                       \n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                              \u2503 Status  \u2503 Details \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 get-repo-info/my-first-deployment  | applied \u2502         \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n\nTo schedule a run for this deployment, use the following command:\n\n        $ prefect deployment run 'get-repo-info/my-first-deployment'\n\n\nYou can also run your flow via the Prefect UI: https://app.prefect.cloud/account/\nabc/workspace/123/deployments/deployment/xyz\n

    Navigate to your Prefect Cloud UI and view your new deployment. Click the Run button to trigger a run of your deployment.

    Because this deployment was configured with a Prefect Managed work pool, Prefect Cloud will run your flow on your behalf.

    View the logs in the UI.

    Now that you've updated your script, you can run it to register your deployment on Prefect Cloud:

    python repo_info.py\n
    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#schedule-a-deployment-run","title":"Schedule a deployment run","text":"

    Now everything is set up for us to submit a flow-run to the work pool. Go ahead and run the deployment from the CLI or the UI.

    prefect deployment run 'get_repo_info/my-deployment'\n

    Prefect Managed work pools are a great way to get started with Prefect. See the Managed Execution guide for more details.

    Many users will find that they need more control over the infrastructure that their flows run on. Prefect Cloud's push work pools are a popular option in those cases.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#push-work-pools-with-automatic-infrastructure-provisioning","title":"Push work pools with automatic infrastructure provisioning","text":"

    Serverless push work pools scale infinitely and provide more configuration options than Prefect Managed work pools.

    Prefect provides push work pools for AWS ECS on Fargate, Azure Container Instances, and Google Cloud Run. You will need to have an account with sufficient permissions on the cloud provider that you want to use. We'll use GCP for this example.

    Setting up the cloud provider pieces for infrastructure can be tricky and time consuming. Fortunately, Prefect can automatically provision infrastructure for you and wire it all together to work with your push work pool.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#create-a-push-work-pool-with-automatic-infrastructure-provisioning","title":"Create a push work pool with automatic infrastructure provisioning","text":"

    In your terminal, run the following command to set up a push work pool.

    Install the gcloud CLI and authenticate with your GCP project.

    If you already have the gcloud CLI installed, be sure to update to the latest version with gcloud components update.

    You will need the following permissions in your GCP project:

    • resourcemanager.projects.list
    • serviceusage.services.enable
    • iam.serviceAccounts.create
    • iam.serviceAccountKeys.create
    • resourcemanager.projects.setIamPolicy
    • artifactregistry.repositories.create

    Docker is also required to build and push images to your registry. You can install Docker here.

    Run the following command to set up a work pool named my-cloud-run-pool of type cloud-run:push.

    prefect work-pool create --type cloud-run:push --provision-infra my-cloud-run-pool \n

    Using the --provision-infra flag will allow you to select a GCP project to use for your work pool and automatically configure it to be ready to execute flows via Cloud Run. In your GCP project, this command will activate the Cloud Run API, create a service account, and create a key for the service account, if they don't already exist. In your Prefect workspace, this command will create a GCPCredentials block for storing the service account key.

    Here's an abbreviated example output from running the command:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-cloud-run-pool will require:                           \u2502\n\u2502                                                                                                          \u2502\n\u2502     Updates in GCP project central-kit-405415 in region us-central1                                      \u2502\n\u2502                                                                                                          \u2502\n\u2502         - Activate the Cloud Run API for your project                                                    \u2502\n\u2502         - Activate the Artifact Registry API for your project                                            \u2502\n\u2502         - Create an Artifact Registry repository named prefect-images                                    \u2502\n\u2502         - Create a service account for managing Cloud Run jobs: prefect-cloud-run                        \u2502\n\u2502             - Service account will be granted the following roles:                                       \u2502\n\u2502                 - Service Account User                                                                   \u2502\n\u2502                 - Cloud Run Developer                                                                    \u2502\n\u2502         - Create a key for service account prefect-cloud-run                                             \u2502\n\u2502                                                                                                          \u2502\n\u2502     Updates in Prefect workspace                                                                         \u2502\n\u2502                                                                                                          \u2502\n\u2502         - Create GCP credentials block my--pool-push-pool-credentials to store the service account key   \u2502\n\u2502                                                                                                          \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]: y\nActivating Cloud Run API\nActivating Artifact Registry API\nCreating Artifact Registry repository\nConfiguring authentication to Artifact Registry\nSetting default Docker build namespace\nCreating service account\nAssigning roles to service account\nCreating service account key\nCreating GCP credentials block\nProvisioning Infrastructure \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned!\nCreated work pool 'my-cloud-run-pool'!\n

    After infrastructure provisioning completes, you will be logged into your new Artifact Registry repository and the default Docker build namespace will be set to the URL of the repository.

    While the default namespace is set, any images you build without specifying a registry or username/organization will be pushed to the repository.

    To take advantage of this functionality, you can write your deploy script like this:

    example_deploy_script.py
    from prefect import flow                                                       \nfrom prefect.deployments import DeploymentImage                                \n\n\n@flow(log_prints=True)\ndef my_flow(name: str = \"world\"):\n    print(f\"Hello {name}! I'm a flow running on Cloud Run!\")\n\n\nif __name__ == \"__main__\":                                                     \n    my_flow.deploy(                                                            \n        name=\"my-deployment\",\n        work_pool_name=\"above-ground\",\n        image=DeploymentImage(\n            name=\"my-image:latest\",\n            platform=\"linux/amd64\",\n        )\n    )\n

    Running this script will build a Docker image with the tag <region>-docker.pkg.dev/<project>/<repository-name>/my-image:latest and push it to your repository.

    Tip

    Make sure you have Docker running locally before running this script.

    Note that you only need to include an object of the DeploymentImage class with the argument platform=\"linux/amd64 if you're building your image on a machine with an ARM-based processor. Otherwise, you could just pass image=\"my-image:latest\" to deploy.

    See the Push Work Pool guide for more details and example commands for each cloud provider.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#next-step","title":"Next step","text":"

    Congratulations! You've learned how to deploy flows to work pools. If these work pool options meet all of your needs, we encourage you to go deeper with the concepts docs or explore our how-to guides to see examples of particular Prefect use cases.

    However, if you need more control over your infrastructure, want to run your workflows in Kubernetes, or are running a self-hosted Prefect server instance, we encourage you to see the next section of the tutorial. There you'll learn how to use work pools that rely on a worker and see how to customize Docker images for container-based infrastructure.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/workers/","title":"Workers","text":"","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#prerequisites","title":"Prerequisites","text":"

    Docker installed and running on your machine.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#why-workers","title":"Why workers","text":"

    In the previous section of the tutorial, you learned how work pools are a bridge between the Prefect orchestration layer and infrastructure for flow runs that can be dynamically provisioned. You saw how you can transition from persistent infrastructure to dynamic infrastructure by using flow.deploy instead of flow.serve.

    Work pools that rely on client-side workers take this a step further by enabling you to run work flows in your own Docker containers, Kubernetes clusters, and serverless environments such as AWS ECS, Azure Container Instances, and GCP Cloud Run.

    The architecture of a worker-based work pool deployment can be summarized with the following diagram:

    graph TD\n    subgraph your_infra[\"Your Execution Environment\"]\n        worker[\"Worker\"]\n    subgraph flow_run_infra[Flow Run Infra]\n     flow_run_a((\"Flow Run A\"))\n    end\n    subgraph flow_run_infra_2[Flow Run Infra]\n     flow_run_b((\"Flow Run B\"))\n    end      \n    end\n\n    subgraph api[\"Prefect API\"]\n    Deployment --> |assigned to| work_pool\n        work_pool([\"Work Pool\"])\n    end\n\n    worker --> |polls| work_pool\n    worker --> |creates| flow_run_infra\n    worker --> |creates| flow_run_infra_2

    Notice above that the worker is in charge of provisioning the flow run infrastructure. In context of this tutorial, that flow run infrastructure is an ephemeral Docker container to host each flow run. Different worker types create different types of flow run infrastructure.

    Now that we\u2019ve reviewed the concepts of a work pool and worker, let\u2019s create them so that you can deploy your tutorial flow, and execute it later using the Prefect API.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#set-up-a-worker-and-work-pool","title":"Set up a worker and work pool","text":"

    For this tutorial you will create a Docker type work pool via the CLI.

    Using the Docker work pool type means that all work sent to this work pool will run within a dedicated Docker container using a Docker client available to the worker.

    Other work pool types

    There are work pool types for serverless computing environments such as AWS ECS, Azure Container Instances, Google Cloud Run, and Vertex AI. Kubernetes is also a popular work pool type.

    These options are expanded upon in various How-to Guides.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#create-a-work-pool","title":"Create a work pool","text":"

    In your terminal, run the following command to set up a Docker type work pool.

    prefect work-pool create --type docker my-docker-pool\n

    Let\u2019s confirm that the work pool was successfully created by running the following command in the same terminal. You should see your new my-docker-pool in the output list.

    prefect work-pool ls\n

    Finally, let\u2019s double check that you can see this work pool in your Prefect UI.

    Navigate to the Work Pools tab and verify that you see my-docker-pool listed.

    When you click into my-docker-pool you should see a red status icon signifying that this work pool is not ready to submit work.

    To get the work pool ready to submit flow runs, you need to start a worker.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#start-a-worker","title":"Start a worker","text":"

    Workers are a lightweight polling process that kick off scheduled flow runs on a certain type of infrastructure (such as Docker). To start a worker on your laptop, open a new terminal and confirm that your virtual environment has prefect installed.

    Run the following command in this new terminal to start the worker:

    prefect worker start --pool my-docker-pool\n

    You should see the worker start. It's now polling the Prefect API to request any scheduled flow runs it should pick up and then submit for execution. You\u2019ll see your new worker listed in the UI under the Workers tab of the Work Pools page with a recent last polled date.

    You should also be able to see a Ready status indicator on your work pool - progress!

    You will need to keep this terminal session active in order for the worker to continue to pick up jobs. Since you are running this worker locally, the worker will terminate if you close the terminal. Therefore, in a production setting this worker should run as a daemonized or managed process.

    Now that you\u2019ve set up your work pool and worker, we have what we need to kick off and execute flow runs of flows deployed to this work pool. Let's deploy your tutorial flow to my-docker-pool.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#create-the-deployment","title":"Create the deployment","text":"

    From our previous steps, we now have:

    1. A flow
    2. A work pool
    3. A worker

    Now it\u2019s time to put it all together. We're going to update our repo_info.py file to build a Docker image and update our deployment so our worker can execute it.

    The updates that you need to make to repo_info.py are:

    1. Change flow.serve to flow.deploy.
    2. Tell flow.deploy which work pool to deploy to.
    3. Tell flow.deploy the name to use for the Docker image it builds.

    Here's what the updated repo_info.py looks like:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-first-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"my-first-deployment-image:tutorial\",\n        push=False\n    )\n

    Why the push=False?

    For this tutorial, your Docker worker is running on your machine, so we don't need to push the image built by flow.deploy to a registry. When your worker is running on a remote machine, you will need to push the image to a registry that the worker can access.

    Remove the push=False argument, include your registry name, and ensure you've authenticated with the Docker CLI to push the image to a registry.

    Now that you've updated your script, you can run it to deploy your flow to the work pool:

    python repo_info.py\n

    Prefect will build a custom Docker image containing your workflow code that the worker can use to dynamically spawn Docker containers whenever this workflow needs to run.

    What Dockerfile?

    In this example, Prefect generates a Dockerfile for you that will build an image based off of one of Prefect's published images. The generated Dockerfile will copy the current directory into the Docker image and install any dependencies listed in a requirements.txt file.

    If you want to use a custom Dockerfile, you can specify the path to the Dockerfile using the DeploymentImage class:

    repo_info.py
    import httpx\nfrom prefect import flow\nfrom prefect.deployments import DeploymentImage\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-first-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=DeploymentImage(\n            name=\"my-first-deployment-image\",\n            tag=\"tutorial\",\n            dockerfile=\"Dockerfile\"\n        ),\n        push=False\n    )\n
    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#modify-the-deployment","title":"Modify the deployment","text":"

    If you need to make updates to your deployment, you can do so by modifying your script and rerunning it. You'll need to make one update to specify a value for job_variables to ensure your Docker worker can successfully execute scheduled runs for this flow. See the example below.

    The job_variables section allows you to fine-tune the infrastructure settings for a specific deployment. These values override default values in the specified work pool's base job template.

    When testing images locally without pushing them to a registry (to avoid potential errors like docker.errors.NotFound), it's recommended to include an image_pull_policy job_variable set to Never. However, for production workflows, always consider pushing images to a remote registry for more reliability and accessibility.

    Here's how you can easily set the image_pull_policy to be Never for this tutorial deployment without affecting the default value set on your work pool:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-first-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        job_variables={\"image_pull_policy\": \"Never\"},\n        image=\"my-first-deployment-image:tutorial\",\n        push=False\n    )\n

    To register this update to your deployment's parameters with Prefect's API, run:

    python repo_info.py\n

    Now everything is set for us to submit a flow-run to the work pool:

    prefect deployment run 'get_repo_info/my-deployment'\n

    Common Pitfall

    • Store and run your deploy scripts at the root of your repo, otherwise the built Docker file may be missing files that it needs to execute!

    Did you know?

    A Prefect flow can have more than one deployment. This can be useful if you want your flow to run in different execution environments or have multiple schedules.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#next-steps","title":"Next steps","text":"
    • Go deeper with deployments and learn about configuring deployments in YAML with prefect.yaml.
    • Concepts contain deep dives into Prefect components.
    • Guides provide step-by-step recipes for common Prefect operations including:
    • Deploying flows on Kubernetes
    • Deploying flows in Docker
    • Deploying flows on serverless infrastructure
    • Daemonizing workers

    Happy building!

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2}]} \ No newline at end of file +{"config":{"lang":["en"],"separator":"[\\s\\-]+","pipeline":["stopWordFilter"],"fields":{"title":{"boost":1000.0},"text":{"boost":1.0},"tags":{"boost":1000000.0}}},"docs":[{"location":"","title":"Welcome to Prefect","text":"

    Prefect is a workflow orchestration tool empowering developers to build, observe, and react to data pipelines.

    It's the easiest way to transform any Python function into a unit of work that can be observed and orchestrated. Just bring your Python code, sprinkle in a few decorators, and go!

    With Prefect you gain:

    • scheduling
    • retries
    • logging
    • convenient async functionality
    • caching
    • notifications
    • observability
    • event-based orchestration

    ","tags":["getting started","quick start","overview"],"boost":2},{"location":"#new-to-prefect","title":"New to Prefect?","text":"

    Get up and running quickly with the quickstart guide.

    Want more hands on practice to productionize your workflows? Follow our tutorial.

    For deeper dives on common use cases, explore our guides.

    Take your understanding even further with Prefect's concepts and API reference.

    Join Prefect's vibrant community of over 26,000 engineers to learn with others and share your knowledge!

    Need help?

    Get your questions answered by a Prefect Product Advocate! Book a Meeting

    ","tags":["getting started","quick start","overview"],"boost":2},{"location":"faq/","title":"Frequently Asked Questions","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#prefect","title":"Prefect","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#how-is-prefect-licensed","title":"How is Prefect licensed?","text":"

    Prefect is licensed under the Apache 2.0 License, an OSI approved open-source license. If you have any questions about licensing, please contact us.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#is-the-prefect-v2-cloud-url-different-than-the-prefect-v1-cloud-url","title":"Is the Prefect v2 Cloud URL different than the Prefect v1 Cloud URL?","text":"

    Yes. Prefect Cloud for v2 is at app.prefect.cloud/ while Prefect Cloud for v1 is at cloud.prefect.io.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#the-prefect-orchestration-engine","title":"The Prefect Orchestration Engine","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#why-was-the-prefect-orchestration-engine-created","title":"Why was the Prefect orchestration engine created?","text":"

    The Prefect orchestration engine has three major objectives:

    • Embracing dynamic, DAG-free workflows
    • An extraordinary developer experience
    • Transparent and observable orchestration rules

    As Prefect has matured, so has the modern data stack. The on-demand, dynamic, highly scalable workflows that used to exist principally in the domain of data science and analytics are now prevalent throughout all of data engineering. Few companies have workflows that don\u2019t deal with streaming data, uncertain timing, runtime logic, complex dependencies, versioning, or custom scheduling.

    This means that the current generation of workflow managers are built around the wrong abstraction: the directed acyclic graph (DAG). DAGs are an increasingly arcane, constrained way of representing the dynamic, heterogeneous range of modern data and computation patterns.

    Furthermore, as workflows have become more complex, it has become even more important to focus on the developer experience of building, testing, and monitoring them. Faced with an explosion of available tools, it is more important than ever for development teams to seek orchestration tools that will be compatible with any code, tools, or services they may require in the future.

    And finally, this additional complexity means that providing clear and consistent insight into the behavior of the orchestration engine and any decisions it makes is critically important.

    The Prefect orchestration engine represents a unified solution to these three problems.

    The Prefect orchestration engine is capable of governing any code through a well-defined series of state transitions designed to maximize the user's understanding of what happened during execution. It's popular to describe \"workflows as code\" or \"orchestration as code,\" but the Prefect engine represents \"code as workflows\": rather than ask users to change how they work to meet the requirements of the orchestrator, we've defined an orchestrator that adapts to how our users work.

    To achieve this, we've leveraged the familiar tools of native Python: first class functions, type annotations, and async support. Users are free to implement as much \u2014 or as little \u2014 of the Prefect engine as is useful for their objectives.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#if-im-using-prefect-cloud-2-do-i-still-need-to-run-a-prefect-server-locally","title":"If I\u2019m using Prefect Cloud 2, do I still need to run a Prefect server locally?","text":"

    No, Prefect Cloud hosts an instance of the Prefect API for you. In fact, each workspace in Prefect Cloud corresponds directly to a single instance of the Prefect orchestration engine. See the Prefect Cloud Overview for more information.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#features","title":"Features","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#does-prefect-support-mapping","title":"Does Prefect support mapping?","text":"

    Yes! For more information, see the Task.map API reference

    @flow\ndef my_flow():\n\n    # map over a constant\n    for i in range(10):\n        my_mapped_task(i)\n\n    # map over a task's output\n    l = list_task()\n    for i in l.wait().result():\n        my_mapped_task_2(i)\n

    Note that when tasks are called on constant values, they cannot detect their upstream edges automatically. In this example, my_mapped_task_2 does not know that it is downstream from list_task(). Prefect will have convenience functions for detecting these associations, and Prefect's .map() operator will automatically track them.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-i-enforce-ordering-between-tasks-that-dont-share-data","title":"Can I enforce ordering between tasks that don't share data?","text":"

    Yes! For more information, see the Tasks section.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#does-prefect-support-proxies","title":"Does Prefect support proxies?","text":"

    Yes!

    Prefect supports communicating via proxies through the use of environment variables. You can read more about this in the Installation documentation and the article Using Prefect Cloud with proxies.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-i-run-prefect-flows-on-linux","title":"Can I run Prefect flows on Linux?","text":"

    Yes!

    See the Installation documentation and Linux installation notes for details on getting started with Prefect on Linux.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-i-run-prefect-flows-on-windows","title":"Can I run Prefect flows on Windows?","text":"

    Yes!

    See the Installation documentation and Windows installation notes for details on getting started with Prefect on Windows.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#what-external-requirements-does-prefect-have","title":"What external requirements does Prefect have?","text":"

    Prefect does not have any additional requirements besides those installed by pip install --pre prefect. The entire system, including the UI and services, can be run in a single process via prefect server start and does not require Docker.

    Prefect Cloud users do not need to worry about the Prefect database. Prefect Cloud uses PostgreSQL on GCP behind the scenes. To use PostgreSQL with a self-hosted Prefect server, users must provide the connection string for a running database via the PREFECT_API_DATABASE_CONNECTION_URL environment variable.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#what-databases-does-prefect-support","title":"What databases does Prefect support?","text":"

    A self-hosted Prefect server can work with SQLite and PostgreSQL. New Prefect installs default to a SQLite database hosted at ~/.prefect/prefect.db on Mac or Linux machines. SQLite and PostgreSQL are not installed by Prefect.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#how-do-i-choose-between-sqlite-and-postgres","title":"How do I choose between SQLite and Postgres?","text":"

    SQLite generally works well for getting started and exploring Prefect. We have tested it with up to hundreds of thousands of task runs. Many users may be able to stay on SQLite for some time. However, for production uses, Prefect Cloud or self-hosted PostgreSQL is highly recommended. Under write-heavy workloads, SQLite performance can begin to suffer. Users running many flows with high degrees of parallelism or concurrency should use PostgreSQL.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#relationship-with-other-prefect-products","title":"Relationship with other Prefect products","text":"","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-a-flow-written-with-prefect-1-be-orchestrated-with-prefect-2-and-vice-versa","title":"Can a flow written with Prefect 1 be orchestrated with Prefect 2 and vice versa?","text":"

    No. Flows written with the Prefect 1 client must be rewritten with the Prefect 2 client. For most flows, this should take just a few minutes. See our migration guide and our Upgrade to Prefect 2 post for more information.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"faq/#can-a-use-prefect-1-and-prefect-2-at-the-same-time-on-my-local-machine","title":"Can a use Prefect 1 and Prefect 2 at the same time on my local machine?","text":"

    Yes. Just use different virtual environments.

    ","tags":["FAQ","frequently asked questions","questions","license","databases"]},{"location":"api-ref/","title":"API Reference","text":"

    Prefect auto-generates reference documentation for the following components:

    • Prefect Python SDK: used to build, test, and execute workflows.
    • Prefect REST API: used by both workflow clients as well as the Prefect UI for orchestration and data retrieval
    • Prefect Cloud REST API documentation is available at https://app.prefect.cloud/api/docs.
    • The REST API documentation for a locally hosted open-source Prefect server is available in the Prefect REST API Reference.
    • Prefect Server SDK: used primarily by the server to work with workflow metadata and enforce orchestration logic. This is only used directly by Prefect developers and contributors.

    Self-hosted docs

    When self-hosting, you can access REST API documentation at the /docs endpoint of your PREFECT_API_URL - for example, if you ran prefect server start with no additional configuration you can find this reference at http://localhost:4200/docs.

    ","tags":["API","Prefect API","Prefect SDK","Prefect Cloud","REST API","development","orchestration"]},{"location":"api-ref/rest-api-reference/","title":"Prefect server REST API reference","text":"","tags":["REST API","Prefect server"]},{"location":"api-ref/prefect/agent/","title":"prefect.agent","text":"","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent","title":"prefect.agent","text":"

    The agent is responsible for checking for flow runs that are ready to run and starting their execution.

    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent","title":"PrefectAgent","text":"Source code in prefect/agent.py
    class PrefectAgent:\n    @experimental_parameter(\n        \"work_pool_name\", group=\"work_pools\", when=lambda y: y is not None\n    )\n    def __init__(\n        self,\n        work_queues: List[str] = None,\n        work_queue_prefix: Union[str, List[str]] = None,\n        work_pool_name: str = None,\n        prefetch_seconds: int = None,\n        default_infrastructure: Infrastructure = None,\n        default_infrastructure_document_id: UUID = None,\n        limit: Optional[int] = None,\n    ) -> None:\n        if default_infrastructure and default_infrastructure_document_id:\n            raise ValueError(\n                \"Provide only one of 'default_infrastructure' and\"\n                \" 'default_infrastructure_document_id'.\"\n            )\n\n        self.work_queues: Set[str] = set(work_queues) if work_queues else set()\n        self.work_pool_name = work_pool_name\n        self.prefetch_seconds = prefetch_seconds\n        self.submitting_flow_run_ids = set()\n        self.cancelling_flow_run_ids = set()\n        self.scheduled_task_scopes = set()\n        self.started = False\n        self.logger = get_logger(\"agent\")\n        self.task_group: Optional[anyio.abc.TaskGroup] = None\n        self.limit: Optional[int] = limit\n        self.limiter: Optional[anyio.CapacityLimiter] = None\n        self.client: Optional[PrefectClient] = None\n\n        if isinstance(work_queue_prefix, str):\n            work_queue_prefix = [work_queue_prefix]\n        self.work_queue_prefix = work_queue_prefix\n\n        self._work_queue_cache_expiration: pendulum.DateTime = None\n        self._work_queue_cache: List[WorkQueue] = []\n\n        if default_infrastructure:\n            self.default_infrastructure_document_id = (\n                default_infrastructure._block_document_id\n            )\n            self.default_infrastructure = default_infrastructure\n        elif default_infrastructure_document_id:\n            self.default_infrastructure_document_id = default_infrastructure_document_id\n            self.default_infrastructure = None\n        else:\n            self.default_infrastructure = Process()\n            self.default_infrastructure_document_id = None\n\n    async def update_matched_agent_work_queues(self):\n        if self.work_queue_prefix:\n            if self.work_pool_name:\n                matched_queues = await self.client.read_work_queues(\n                    work_pool_name=self.work_pool_name,\n                    work_queue_filter=WorkQueueFilter(\n                        name=WorkQueueFilterName(startswith_=self.work_queue_prefix)\n                    ),\n                )\n            else:\n                matched_queues = await self.client.match_work_queues(\n                    self.work_queue_prefix, work_pool_name=DEFAULT_AGENT_WORK_POOL_NAME\n                )\n\n            matched_queues = set(q.name for q in matched_queues)\n            if matched_queues != self.work_queues:\n                new_queues = matched_queues - self.work_queues\n                removed_queues = self.work_queues - matched_queues\n                if new_queues:\n                    self.logger.info(\n                        f\"Matched new work queues: {', '.join(new_queues)}\"\n                    )\n                if removed_queues:\n                    self.logger.info(\n                        f\"Work queues no longer matched: {', '.join(removed_queues)}\"\n                    )\n            self.work_queues = matched_queues\n\n    async def get_work_queues(self) -> AsyncIterator[WorkQueue]:\n        \"\"\"\n        Loads the work queue objects corresponding to the agent's target work\n        queues. If any of them don't exist, they are created.\n        \"\"\"\n\n        # if the queue cache has not expired, yield queues from the cache\n        now = pendulum.now(\"UTC\")\n        if (self._work_queue_cache_expiration or now) > now:\n            for queue in self._work_queue_cache:\n                yield queue\n            return\n\n        # otherwise clear the cache, set the expiration for 30 seconds, and\n        # reload the work queues\n        self._work_queue_cache.clear()\n        self._work_queue_cache_expiration = now.add(seconds=30)\n\n        await self.update_matched_agent_work_queues()\n\n        for name in self.work_queues:\n            try:\n                work_queue = await self.client.read_work_queue_by_name(\n                    work_pool_name=self.work_pool_name, name=name\n                )\n            except (ObjectNotFound, Exception):\n                work_queue = None\n\n            # if the work queue wasn't found and the agent is NOT polling\n            # for queues using a regex, try to create it\n            if work_queue is None and not self.work_queue_prefix:\n                try:\n                    work_queue = await self.client.create_work_queue(\n                        work_pool_name=self.work_pool_name, name=name\n                    )\n                except Exception:\n                    # if creating it raises an exception, it was probably just\n                    # created by some other agent; rather than entering a re-read\n                    # loop with new error handling, we log the exception and\n                    # continue.\n                    self.logger.exception(f\"Failed to create work queue {name!r}.\")\n                    continue\n                else:\n                    log_str = f\"Created work queue {name!r}\"\n                    if self.work_pool_name:\n                        log_str = (\n                            f\"Created work queue {name!r} in work pool\"\n                            f\" {self.work_pool_name!r}.\"\n                        )\n                    else:\n                        log_str = f\"Created work queue '{name}'.\"\n                    self.logger.info(log_str)\n\n            if work_queue is None:\n                self.logger.error(\n                    f\"Work queue '{name!r}' with prefix {self.work_queue_prefix} wasn't\"\n                    \" found\"\n                )\n            else:\n                self._work_queue_cache.append(work_queue)\n                yield work_queue\n\n    async def get_and_submit_flow_runs(self) -> List[FlowRun]:\n        \"\"\"\n        The principle method on agents. Queries for scheduled flow runs and submits\n        them for execution in parallel.\n        \"\"\"\n        if not self.started:\n            raise RuntimeError(\n                \"Agent is not started. Use `async with PrefectAgent()...`\"\n            )\n\n        self.logger.debug(\"Checking for scheduled flow runs...\")\n\n        before = pendulum.now(\"utc\").add(\n            seconds=self.prefetch_seconds or PREFECT_AGENT_PREFETCH_SECONDS.value()\n        )\n\n        submittable_runs: List[FlowRun] = []\n\n        if self.work_pool_name:\n            responses = await self.client.get_scheduled_flow_runs_for_work_pool(\n                work_pool_name=self.work_pool_name,\n                work_queue_names=[wq.name async for wq in self.get_work_queues()],\n                scheduled_before=before,\n            )\n            submittable_runs.extend([response.flow_run for response in responses])\n\n        else:\n            # load runs from each work queue\n            async for work_queue in self.get_work_queues():\n                # print a nice message if the work queue is paused\n                if work_queue.is_paused:\n                    self.logger.info(\n                        f\"Work queue {work_queue.name!r} ({work_queue.id}) is paused.\"\n                    )\n\n                else:\n                    try:\n                        queue_runs = await self.client.get_runs_in_work_queue(\n                            id=work_queue.id, limit=10, scheduled_before=before\n                        )\n                        submittable_runs.extend(queue_runs)\n                    except ObjectNotFound:\n                        self.logger.error(\n                            f\"Work queue {work_queue.name!r} ({work_queue.id}) not\"\n                            \" found.\"\n                        )\n                    except Exception as exc:\n                        self.logger.exception(exc)\n\n            submittable_runs.sort(key=lambda run: run.next_scheduled_start_time)\n\n        for flow_run in submittable_runs:\n            # don't resubmit a run\n            if flow_run.id in self.submitting_flow_run_ids:\n                continue\n\n            try:\n                if self.limiter:\n                    self.limiter.acquire_on_behalf_of_nowait(flow_run.id)\n            except anyio.WouldBlock:\n                self.logger.info(\n                    f\"Flow run limit reached; {self.limiter.borrowed_tokens} flow runs\"\n                    \" in progress.\"\n                )\n                break\n            else:\n                self.logger.info(f\"Submitting flow run '{flow_run.id}'\")\n                self.submitting_flow_run_ids.add(flow_run.id)\n                self.task_group.start_soon(\n                    self.submit_run,\n                    flow_run,\n                )\n\n        return list(\n            filter(lambda run: run.id in self.submitting_flow_run_ids, submittable_runs)\n        )\n\n    async def check_for_cancelled_flow_runs(self):\n        if not self.started:\n            raise RuntimeError(\n                \"Agent is not started. Use `async with PrefectAgent()...`\"\n            )\n\n        self.logger.debug(\"Checking for cancelled flow runs...\")\n\n        work_queue_filter = (\n            WorkQueueFilter(name=WorkQueueFilterName(any_=list(self.work_queues)))\n            if self.work_queues\n            else None\n        )\n\n        work_pool_filter = (\n            WorkPoolFilter(name=WorkPoolFilterName(any_=[self.work_pool_name]))\n            if self.work_pool_name\n            else WorkPoolFilter(name=WorkPoolFilterName(any_=[\"default-agent-pool\"]))\n        )\n        named_cancelling_flow_runs = await self.client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLED]),\n                    name=FlowRunFilterStateName(any_=[\"Cancelling\"]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(not_any_=list(self.cancelling_flow_run_ids)),\n            ),\n            work_pool_filter=work_pool_filter,\n            work_queue_filter=work_queue_filter,\n        )\n\n        typed_cancelling_flow_runs = await self.client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLING]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(not_any_=list(self.cancelling_flow_run_ids)),\n            ),\n            work_pool_filter=work_pool_filter,\n            work_queue_filter=work_queue_filter,\n        )\n\n        cancelling_flow_runs = named_cancelling_flow_runs + typed_cancelling_flow_runs\n\n        if cancelling_flow_runs:\n            self.logger.info(\n                f\"Found {len(cancelling_flow_runs)} flow runs awaiting cancellation.\"\n            )\n\n        for flow_run in cancelling_flow_runs:\n            self.cancelling_flow_run_ids.add(flow_run.id)\n            self.task_group.start_soon(self.cancel_run, flow_run)\n\n        return cancelling_flow_runs\n\n    async def cancel_run(self, flow_run: FlowRun) -> None:\n        \"\"\"\n        Cancel a flow run by killing its infrastructure\n        \"\"\"\n        if not flow_run.infrastructure_pid:\n            self.logger.error(\n                f\"Flow run '{flow_run.id}' does not have an infrastructure pid\"\n                \" attached. Cancellation cannot be guaranteed.\"\n            )\n            await self._mark_flow_run_as_cancelled(\n                flow_run,\n                state_updates={\n                    \"message\": (\n                        \"This flow run is missing infrastructure tracking information\"\n                        \" and cancellation cannot be guaranteed.\"\n                    )\n                },\n            )\n            return\n\n        try:\n            infrastructure = await self.get_infrastructure(flow_run)\n            if infrastructure.is_using_a_runner:\n                self.logger.info(\n                    f\"Skipping cancellation because flow run {str(flow_run.id)!r} is\"\n                    \" using enhanced cancellation. A dedicated runner will handle\"\n                    \" cancellation.\"\n                )\n                return\n        except Exception:\n            self.logger.exception(\n                f\"Failed to get infrastructure for flow run '{flow_run.id}'. \"\n                \"Flow run cannot be cancelled.\"\n            )\n            # Note: We leave this flow run in the cancelling set because it cannot be\n            #       cancelled and this will prevent additional attempts.\n            return\n\n        if not hasattr(infrastructure, \"kill\"):\n            self.logger.error(\n                f\"Flow run '{flow_run.id}' infrastructure {infrastructure.type!r} \"\n                \"does not support killing created infrastructure. \"\n                \"Cancellation cannot be guaranteed.\"\n            )\n            return\n\n        self.logger.info(\n            f\"Killing {infrastructure.type} {flow_run.infrastructure_pid} for flow run \"\n            f\"'{flow_run.id}'...\"\n        )\n        try:\n            await infrastructure.kill(flow_run.infrastructure_pid)\n        except InfrastructureNotFound as exc:\n            self.logger.warning(f\"{exc} Marking flow run as cancelled.\")\n            await self._mark_flow_run_as_cancelled(flow_run)\n        except InfrastructureNotAvailable as exc:\n            self.logger.warning(f\"{exc} Flow run cannot be cancelled by this agent.\")\n        except Exception:\n            self.logger.exception(\n                \"Encountered exception while killing infrastructure for flow run \"\n                f\"'{flow_run.id}'. Flow run may not be cancelled.\"\n            )\n            # We will try again on generic exceptions\n            self.cancelling_flow_run_ids.remove(flow_run.id)\n            return\n        else:\n            await self._mark_flow_run_as_cancelled(flow_run)\n            self.logger.info(f\"Cancelled flow run '{flow_run.id}'!\")\n\n    async def _mark_flow_run_as_cancelled(\n        self, flow_run: FlowRun, state_updates: Optional[dict] = None\n    ) -> None:\n        state_updates = state_updates or {}\n        state_updates.setdefault(\"name\", \"Cancelled\")\n        state_updates.setdefault(\"type\", StateType.CANCELLED)\n        state = flow_run.state.copy(update=state_updates)\n\n        await self.client.set_flow_run_state(flow_run.id, state, force=True)\n\n        # Do not remove the flow run from the cancelling set immediately because\n        # the API caches responses for the `read_flow_runs` and we do not want to\n        # duplicate cancellations.\n        await self._schedule_task(\n            60 * 10, self.cancelling_flow_run_ids.remove, flow_run.id\n        )\n\n    async def get_infrastructure(self, flow_run: FlowRun) -> Infrastructure:\n        deployment = await self.client.read_deployment(flow_run.deployment_id)\n\n        flow = await self.client.read_flow(deployment.flow_id)\n\n        # overrides only apply when configuring known infra blocks\n        if not deployment.infrastructure_document_id:\n            if self.default_infrastructure:\n                infra_block = self.default_infrastructure\n            else:\n                infra_document = await self.client.read_block_document(\n                    self.default_infrastructure_document_id\n                )\n                infra_block = Block._from_block_document(infra_document)\n\n            # Add flow run metadata to the infrastructure\n            prepared_infrastructure = infra_block.prepare_for_flow_run(\n                flow_run, deployment=deployment, flow=flow\n            )\n            return prepared_infrastructure\n\n        ## get infra\n        infra_document = await self.client.read_block_document(\n            deployment.infrastructure_document_id\n        )\n\n        # this piece of logic applies any overrides that may have been set on the\n        # deployment; overrides are defined as dot.delimited paths on possibly nested\n        # attributes of the infrastructure block\n        doc_dict = infra_document.dict()\n        infra_dict = doc_dict.get(\"data\", {})\n        for override, value in (deployment.infra_overrides or {}).items():\n            nested_fields = override.split(\".\")\n            data = infra_dict\n            for field in nested_fields[:-1]:\n                data = data[field]\n\n            # once we reach the end, set the value\n            data[nested_fields[-1]] = value\n\n        # reconstruct the infra block\n        doc_dict[\"data\"] = infra_dict\n        infra_document = BlockDocument(**doc_dict)\n        infrastructure_block = Block._from_block_document(infra_document)\n\n        # TODO: Here the agent may update the infrastructure with agent-level settings\n\n        # Add flow run metadata to the infrastructure\n        prepared_infrastructure = infrastructure_block.prepare_for_flow_run(\n            flow_run, deployment=deployment, flow=flow\n        )\n\n        return prepared_infrastructure\n\n    async def submit_run(self, flow_run: FlowRun) -> None:\n        \"\"\"\n        Submit a flow run to the infrastructure\n        \"\"\"\n        ready_to_submit = await self._propose_pending_state(flow_run)\n\n        if ready_to_submit:\n            try:\n                infrastructure = await self.get_infrastructure(flow_run)\n            except Exception as exc:\n                self.logger.exception(\n                    f\"Failed to get infrastructure for flow run '{flow_run.id}'.\"\n                )\n                await self._propose_failed_state(flow_run, exc)\n                if self.limiter:\n                    self.limiter.release_on_behalf_of(flow_run.id)\n            else:\n                # Wait for submission to be completed. Note that the submission function\n                # may continue to run in the background after this exits.\n                readiness_result = await self.task_group.start(\n                    self._submit_run_and_capture_errors, flow_run, infrastructure\n                )\n\n                if readiness_result and not isinstance(readiness_result, Exception):\n                    try:\n                        await self.client.update_flow_run(\n                            flow_run_id=flow_run.id,\n                            infrastructure_pid=str(readiness_result),\n                        )\n                    except Exception:\n                        self.logger.exception(\n                            \"An error occurred while setting the `infrastructure_pid`\"\n                            f\" on flow run {flow_run.id!r}. The flow run will not be\"\n                            \" cancellable.\"\n                        )\n\n                self.logger.info(f\"Completed submission of flow run '{flow_run.id}'\")\n\n        else:\n            # If the run is not ready to submit, release the concurrency slot\n            if self.limiter:\n                self.limiter.release_on_behalf_of(flow_run.id)\n\n        self.submitting_flow_run_ids.remove(flow_run.id)\n\n    async def _submit_run_and_capture_errors(\n        self,\n        flow_run: FlowRun,\n        infrastructure: Infrastructure,\n        task_status: anyio.abc.TaskStatus = None,\n    ) -> Union[InfrastructureResult, Exception]:\n        # Note: There is not a clear way to determine if task_status.started() has been\n        #       called without peeking at the internal `_future`. Ideally we could just\n        #       check if the flow run id has been removed from `submitting_flow_run_ids`\n        #       but it is not so simple to guarantee that this coroutine yields back\n        #       to `submit_run` to execute that line when exceptions are raised during\n        #       submission.\n        try:\n            result = await infrastructure.run(task_status=task_status)\n        except Exception as exc:\n            if not task_status._future.done():\n                # This flow run was being submitted and did not start successfully\n                self.logger.exception(\n                    f\"Failed to submit flow run '{flow_run.id}' to infrastructure.\"\n                )\n                # Mark the task as started to prevent agent crash\n                task_status.started(exc)\n                await self._propose_crashed_state(\n                    flow_run, \"Flow run could not be submitted to infrastructure\"\n                )\n            else:\n                self.logger.exception(\n                    f\"An error occurred while monitoring flow run '{flow_run.id}'. \"\n                    \"The flow run will not be marked as failed, but an issue may have \"\n                    \"occurred.\"\n                )\n            return exc\n        finally:\n            if self.limiter:\n                self.limiter.release_on_behalf_of(flow_run.id)\n\n        if not task_status._future.done():\n            self.logger.error(\n                f\"Infrastructure returned without reporting flow run '{flow_run.id}' \"\n                \"as started or raising an error. This behavior is not expected and \"\n                \"generally indicates improper implementation of infrastructure. The \"\n                \"flow run will not be marked as failed, but an issue may have occurred.\"\n            )\n            # Mark the task as started to prevent agent crash\n            task_status.started()\n\n        if result.status_code != 0:\n            await self._propose_crashed_state(\n                flow_run,\n                (\n                    \"Flow run infrastructure exited with non-zero status code\"\n                    f\" {result.status_code}.\"\n                ),\n            )\n\n        return result\n\n    async def _propose_pending_state(self, flow_run: FlowRun) -> bool:\n        state = flow_run.state\n        try:\n            state = await propose_state(self.client, Pending(), flow_run_id=flow_run.id)\n        except Abort as exc:\n            self.logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}'. \"\n                    f\"Server sent an abort signal: {exc}\"\n                ),\n            )\n            return False\n        except Exception:\n            self.logger.error(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n                exc_info=True,\n            )\n            return False\n\n        if not state.is_pending():\n            self.logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}': \"\n                    f\"Server returned a non-pending state {state.type.value!r}\"\n                ),\n            )\n            return False\n\n        return True\n\n    async def _propose_failed_state(self, flow_run: FlowRun, exc: Exception) -> None:\n        try:\n            await propose_state(\n                self.client,\n                await exception_to_failed_state(message=\"Submission failed.\", exc=exc),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # We've already failed, no need to note the abort but we don't want it to\n            # raise in the agent process\n            pass\n        except Exception:\n            self.logger.error(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n                exc_info=True,\n            )\n\n    async def _propose_crashed_state(self, flow_run: FlowRun, message: str) -> None:\n        try:\n            state = await propose_state(\n                self.client,\n                Crashed(message=message),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # Flow run already marked as failed\n            pass\n        except Exception:\n            self.logger.exception(f\"Failed to update state of flow run '{flow_run.id}'\")\n        else:\n            if state.is_crashed():\n                self.logger.info(\n                    f\"Reported flow run '{flow_run.id}' as crashed: {message}\"\n                )\n\n    async def _schedule_task(self, __in_seconds: int, fn, *args, **kwargs):\n        \"\"\"\n        Schedule a background task to start after some time.\n\n        These tasks will be run immediately when the agent exits instead of waiting.\n\n        The function may be async or sync. Async functions will be awaited.\n        \"\"\"\n\n        async def wrapper(task_status):\n            # If we are shutting down, do not sleep; otherwise sleep until the scheduled\n            # time or shutdown\n            if self.started:\n                with anyio.CancelScope() as scope:\n                    self.scheduled_task_scopes.add(scope)\n                    task_status.started()\n                    await anyio.sleep(__in_seconds)\n\n                self.scheduled_task_scopes.remove(scope)\n            else:\n                task_status.started()\n\n            result = fn(*args, **kwargs)\n            if inspect.iscoroutine(result):\n                await result\n\n        await self.task_group.start(wrapper)\n\n    # Context management ---------------------------------------------------------------\n\n    async def start(self):\n        self.started = True\n        self.task_group = anyio.create_task_group()\n        self.limiter = (\n            anyio.CapacityLimiter(self.limit) if self.limit is not None else None\n        )\n        self.client = get_client()\n        await self.client.__aenter__()\n        await self.task_group.__aenter__()\n\n    async def shutdown(self, *exc_info):\n        self.started = False\n        # We must cancel scheduled task scopes before closing the task group\n        for scope in self.scheduled_task_scopes:\n            scope.cancel()\n        await self.task_group.__aexit__(*exc_info)\n        await self.client.__aexit__(*exc_info)\n        self.task_group = None\n        self.client = None\n        self.submitting_flow_run_ids.clear()\n        self.cancelling_flow_run_ids.clear()\n        self.scheduled_task_scopes.clear()\n        self._work_queue_cache_expiration = None\n        self._work_queue_cache = []\n\n    async def __aenter__(self):\n        await self.start()\n        return self\n\n    async def __aexit__(self, *exc_info):\n        await self.shutdown(*exc_info)\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent.cancel_run","title":"cancel_run async","text":"

    Cancel a flow run by killing its infrastructure

    Source code in prefect/agent.py
    async def cancel_run(self, flow_run: FlowRun) -> None:\n    \"\"\"\n    Cancel a flow run by killing its infrastructure\n    \"\"\"\n    if not flow_run.infrastructure_pid:\n        self.logger.error(\n            f\"Flow run '{flow_run.id}' does not have an infrastructure pid\"\n            \" attached. Cancellation cannot be guaranteed.\"\n        )\n        await self._mark_flow_run_as_cancelled(\n            flow_run,\n            state_updates={\n                \"message\": (\n                    \"This flow run is missing infrastructure tracking information\"\n                    \" and cancellation cannot be guaranteed.\"\n                )\n            },\n        )\n        return\n\n    try:\n        infrastructure = await self.get_infrastructure(flow_run)\n        if infrastructure.is_using_a_runner:\n            self.logger.info(\n                f\"Skipping cancellation because flow run {str(flow_run.id)!r} is\"\n                \" using enhanced cancellation. A dedicated runner will handle\"\n                \" cancellation.\"\n            )\n            return\n    except Exception:\n        self.logger.exception(\n            f\"Failed to get infrastructure for flow run '{flow_run.id}'. \"\n            \"Flow run cannot be cancelled.\"\n        )\n        # Note: We leave this flow run in the cancelling set because it cannot be\n        #       cancelled and this will prevent additional attempts.\n        return\n\n    if not hasattr(infrastructure, \"kill\"):\n        self.logger.error(\n            f\"Flow run '{flow_run.id}' infrastructure {infrastructure.type!r} \"\n            \"does not support killing created infrastructure. \"\n            \"Cancellation cannot be guaranteed.\"\n        )\n        return\n\n    self.logger.info(\n        f\"Killing {infrastructure.type} {flow_run.infrastructure_pid} for flow run \"\n        f\"'{flow_run.id}'...\"\n    )\n    try:\n        await infrastructure.kill(flow_run.infrastructure_pid)\n    except InfrastructureNotFound as exc:\n        self.logger.warning(f\"{exc} Marking flow run as cancelled.\")\n        await self._mark_flow_run_as_cancelled(flow_run)\n    except InfrastructureNotAvailable as exc:\n        self.logger.warning(f\"{exc} Flow run cannot be cancelled by this agent.\")\n    except Exception:\n        self.logger.exception(\n            \"Encountered exception while killing infrastructure for flow run \"\n            f\"'{flow_run.id}'. Flow run may not be cancelled.\"\n        )\n        # We will try again on generic exceptions\n        self.cancelling_flow_run_ids.remove(flow_run.id)\n        return\n    else:\n        await self._mark_flow_run_as_cancelled(flow_run)\n        self.logger.info(f\"Cancelled flow run '{flow_run.id}'!\")\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent.get_and_submit_flow_runs","title":"get_and_submit_flow_runs async","text":"

    The principle method on agents. Queries for scheduled flow runs and submits them for execution in parallel.

    Source code in prefect/agent.py
    async def get_and_submit_flow_runs(self) -> List[FlowRun]:\n    \"\"\"\n    The principle method on agents. Queries for scheduled flow runs and submits\n    them for execution in parallel.\n    \"\"\"\n    if not self.started:\n        raise RuntimeError(\n            \"Agent is not started. Use `async with PrefectAgent()...`\"\n        )\n\n    self.logger.debug(\"Checking for scheduled flow runs...\")\n\n    before = pendulum.now(\"utc\").add(\n        seconds=self.prefetch_seconds or PREFECT_AGENT_PREFETCH_SECONDS.value()\n    )\n\n    submittable_runs: List[FlowRun] = []\n\n    if self.work_pool_name:\n        responses = await self.client.get_scheduled_flow_runs_for_work_pool(\n            work_pool_name=self.work_pool_name,\n            work_queue_names=[wq.name async for wq in self.get_work_queues()],\n            scheduled_before=before,\n        )\n        submittable_runs.extend([response.flow_run for response in responses])\n\n    else:\n        # load runs from each work queue\n        async for work_queue in self.get_work_queues():\n            # print a nice message if the work queue is paused\n            if work_queue.is_paused:\n                self.logger.info(\n                    f\"Work queue {work_queue.name!r} ({work_queue.id}) is paused.\"\n                )\n\n            else:\n                try:\n                    queue_runs = await self.client.get_runs_in_work_queue(\n                        id=work_queue.id, limit=10, scheduled_before=before\n                    )\n                    submittable_runs.extend(queue_runs)\n                except ObjectNotFound:\n                    self.logger.error(\n                        f\"Work queue {work_queue.name!r} ({work_queue.id}) not\"\n                        \" found.\"\n                    )\n                except Exception as exc:\n                    self.logger.exception(exc)\n\n        submittable_runs.sort(key=lambda run: run.next_scheduled_start_time)\n\n    for flow_run in submittable_runs:\n        # don't resubmit a run\n        if flow_run.id in self.submitting_flow_run_ids:\n            continue\n\n        try:\n            if self.limiter:\n                self.limiter.acquire_on_behalf_of_nowait(flow_run.id)\n        except anyio.WouldBlock:\n            self.logger.info(\n                f\"Flow run limit reached; {self.limiter.borrowed_tokens} flow runs\"\n                \" in progress.\"\n            )\n            break\n        else:\n            self.logger.info(f\"Submitting flow run '{flow_run.id}'\")\n            self.submitting_flow_run_ids.add(flow_run.id)\n            self.task_group.start_soon(\n                self.submit_run,\n                flow_run,\n            )\n\n    return list(\n        filter(lambda run: run.id in self.submitting_flow_run_ids, submittable_runs)\n    )\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent.get_work_queues","title":"get_work_queues async","text":"

    Loads the work queue objects corresponding to the agent's target work queues. If any of them don't exist, they are created.

    Source code in prefect/agent.py
    async def get_work_queues(self) -> AsyncIterator[WorkQueue]:\n    \"\"\"\n    Loads the work queue objects corresponding to the agent's target work\n    queues. If any of them don't exist, they are created.\n    \"\"\"\n\n    # if the queue cache has not expired, yield queues from the cache\n    now = pendulum.now(\"UTC\")\n    if (self._work_queue_cache_expiration or now) > now:\n        for queue in self._work_queue_cache:\n            yield queue\n        return\n\n    # otherwise clear the cache, set the expiration for 30 seconds, and\n    # reload the work queues\n    self._work_queue_cache.clear()\n    self._work_queue_cache_expiration = now.add(seconds=30)\n\n    await self.update_matched_agent_work_queues()\n\n    for name in self.work_queues:\n        try:\n            work_queue = await self.client.read_work_queue_by_name(\n                work_pool_name=self.work_pool_name, name=name\n            )\n        except (ObjectNotFound, Exception):\n            work_queue = None\n\n        # if the work queue wasn't found and the agent is NOT polling\n        # for queues using a regex, try to create it\n        if work_queue is None and not self.work_queue_prefix:\n            try:\n                work_queue = await self.client.create_work_queue(\n                    work_pool_name=self.work_pool_name, name=name\n                )\n            except Exception:\n                # if creating it raises an exception, it was probably just\n                # created by some other agent; rather than entering a re-read\n                # loop with new error handling, we log the exception and\n                # continue.\n                self.logger.exception(f\"Failed to create work queue {name!r}.\")\n                continue\n            else:\n                log_str = f\"Created work queue {name!r}\"\n                if self.work_pool_name:\n                    log_str = (\n                        f\"Created work queue {name!r} in work pool\"\n                        f\" {self.work_pool_name!r}.\"\n                    )\n                else:\n                    log_str = f\"Created work queue '{name}'.\"\n                self.logger.info(log_str)\n\n        if work_queue is None:\n            self.logger.error(\n                f\"Work queue '{name!r}' with prefix {self.work_queue_prefix} wasn't\"\n                \" found\"\n            )\n        else:\n            self._work_queue_cache.append(work_queue)\n            yield work_queue\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/agent/#prefect.agent.PrefectAgent.submit_run","title":"submit_run async","text":"

    Submit a flow run to the infrastructure

    Source code in prefect/agent.py
    async def submit_run(self, flow_run: FlowRun) -> None:\n    \"\"\"\n    Submit a flow run to the infrastructure\n    \"\"\"\n    ready_to_submit = await self._propose_pending_state(flow_run)\n\n    if ready_to_submit:\n        try:\n            infrastructure = await self.get_infrastructure(flow_run)\n        except Exception as exc:\n            self.logger.exception(\n                f\"Failed to get infrastructure for flow run '{flow_run.id}'.\"\n            )\n            await self._propose_failed_state(flow_run, exc)\n            if self.limiter:\n                self.limiter.release_on_behalf_of(flow_run.id)\n        else:\n            # Wait for submission to be completed. Note that the submission function\n            # may continue to run in the background after this exits.\n            readiness_result = await self.task_group.start(\n                self._submit_run_and_capture_errors, flow_run, infrastructure\n            )\n\n            if readiness_result and not isinstance(readiness_result, Exception):\n                try:\n                    await self.client.update_flow_run(\n                        flow_run_id=flow_run.id,\n                        infrastructure_pid=str(readiness_result),\n                    )\n                except Exception:\n                    self.logger.exception(\n                        \"An error occurred while setting the `infrastructure_pid`\"\n                        f\" on flow run {flow_run.id!r}. The flow run will not be\"\n                        \" cancellable.\"\n                    )\n\n            self.logger.info(f\"Completed submission of flow run '{flow_run.id}'\")\n\n    else:\n        # If the run is not ready to submit, release the concurrency slot\n        if self.limiter:\n            self.limiter.release_on_behalf_of(flow_run.id)\n\n    self.submitting_flow_run_ids.remove(flow_run.id)\n
    ","tags":["Python API","agents"]},{"location":"api-ref/prefect/artifacts/","title":"prefect.artifacts","text":"","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/artifacts/#prefect.artifacts","title":"prefect.artifacts","text":"

    Interface for creating and reading artifacts.

    ","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/artifacts/#prefect.artifacts.create_link_artifact","title":"create_link_artifact async","text":"

    Create a link artifact.

    Parameters:

    Name Type Description Default link str

    The link to create.

    required link_text Optional[str]

    The link text.

    None key Optional[str]

    A user-provided string identifier. Required for the artifact to show in the Artifacts page in the UI. The key must only contain lowercase letters, numbers, and dashes.

    None description Optional[str]

    A user-specified description of the artifact.

    None

    Returns:

    Type Description UUID

    The table artifact ID.

    Source code in prefect/artifacts.py
    @sync_compatible\nasync def create_link_artifact(\n    link: str,\n    link_text: Optional[str] = None,\n    key: Optional[str] = None,\n    description: Optional[str] = None,\n) -> UUID:\n    \"\"\"\n    Create a link artifact.\n\n    Arguments:\n        link: The link to create.\n        link_text: The link text.\n        key: A user-provided string identifier.\n          Required for the artifact to show in the Artifacts page in the UI.\n          The key must only contain lowercase letters, numbers, and dashes.\n        description: A user-specified description of the artifact.\n\n\n    Returns:\n        The table artifact ID.\n    \"\"\"\n    formatted_link = f\"[{link_text}]({link})\" if link_text else f\"[{link}]({link})\"\n    artifact = await _create_artifact(\n        key=key,\n        type=\"markdown\",\n        description=description,\n        data=formatted_link,\n    )\n\n    return artifact.id\n
    ","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/artifacts/#prefect.artifacts.create_markdown_artifact","title":"create_markdown_artifact async","text":"

    Create a markdown artifact.

    Parameters:

    Name Type Description Default markdown str

    The markdown to create.

    required key Optional[str]

    A user-provided string identifier. Required for the artifact to show in the Artifacts page in the UI. The key must only contain lowercase letters, numbers, and dashes.

    None description Optional[str]

    A user-specified description of the artifact.

    None

    Returns:

    Type Description UUID

    The table artifact ID.

    Source code in prefect/artifacts.py
    @sync_compatible\nasync def create_markdown_artifact(\n    markdown: str,\n    key: Optional[str] = None,\n    description: Optional[str] = None,\n) -> UUID:\n    \"\"\"\n    Create a markdown artifact.\n\n    Arguments:\n        markdown: The markdown to create.\n        key: A user-provided string identifier.\n          Required for the artifact to show in the Artifacts page in the UI.\n          The key must only contain lowercase letters, numbers, and dashes.\n        description: A user-specified description of the artifact.\n\n    Returns:\n        The table artifact ID.\n    \"\"\"\n    artifact = await _create_artifact(\n        key=key,\n        type=\"markdown\",\n        description=description,\n        data=markdown,\n    )\n\n    return artifact.id\n
    ","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/artifacts/#prefect.artifacts.create_table_artifact","title":"create_table_artifact async","text":"

    Create a table artifact.

    Parameters:

    Name Type Description Default table Union[Dict[str, List[Any]], List[Dict[str, Any]], List[List[Any]]]

    The table to create.

    required key Optional[str]

    A user-provided string identifier. Required for the artifact to show in the Artifacts page in the UI. The key must only contain lowercase letters, numbers, and dashes.

    None description Optional[str]

    A user-specified description of the artifact.

    None

    Returns:

    Type Description UUID

    The table artifact ID.

    Source code in prefect/artifacts.py
    @sync_compatible\nasync def create_table_artifact(\n    table: Union[Dict[str, List[Any]], List[Dict[str, Any]], List[List[Any]]],\n    key: Optional[str] = None,\n    description: Optional[str] = None,\n) -> UUID:\n    \"\"\"\n    Create a table artifact.\n\n    Arguments:\n        table: The table to create.\n        key: A user-provided string identifier.\n          Required for the artifact to show in the Artifacts page in the UI.\n          The key must only contain lowercase letters, numbers, and dashes.\n        description: A user-specified description of the artifact.\n\n    Returns:\n        The table artifact ID.\n    \"\"\"\n\n    def _sanitize_nan_values(item):\n        \"\"\"\n        Sanitize NaN values in a given item. The item can be a dict, list or float.\n        \"\"\"\n\n        if isinstance(item, list):\n            return [_sanitize_nan_values(sub_item) for sub_item in item]\n\n        elif isinstance(item, dict):\n            return {k: _sanitize_nan_values(v) for k, v in item.items()}\n\n        elif isinstance(item, float) and math.isnan(item):\n            return None\n\n        else:\n            return item\n\n    sanitized_table = _sanitize_nan_values(table)\n\n    if isinstance(table, dict) and all(isinstance(v, list) for v in table.values()):\n        pass\n    elif isinstance(table, list) and all(isinstance(v, (list, dict)) for v in table):\n        pass\n    else:\n        raise TypeError(INVALID_TABLE_TYPE_ERROR)\n\n    formatted_table = json.dumps(sanitized_table)\n\n    artifact = await _create_artifact(\n        key=key,\n        type=\"table\",\n        description=description,\n        data=formatted_table,\n    )\n\n    return artifact.id\n
    ","tags":["Python API","artifacts"]},{"location":"api-ref/prefect/context/","title":"prefect.context","text":"","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context","title":"prefect.context","text":"

    Async and thread safe models for passing runtime context data.

    These contexts should never be directly mutated by the user.

    For more user-accessible information about the current run, see prefect.runtime.

    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.ContextModel","title":"ContextModel","text":"

    Bases: BaseModel

    A base model for context data that forbids mutation and extra data while providing a context manager

    Source code in prefect/context.py
    class ContextModel(BaseModel):\n    \"\"\"\n    A base model for context data that forbids mutation and extra data while providing\n    a context manager\n    \"\"\"\n\n    # The context variable for storing data must be defined by the child class\n    __var__: ContextVar\n    _token: Token = PrivateAttr(None)\n\n    class Config:\n        allow_mutation = False\n        arbitrary_types_allowed = True\n        extra = \"forbid\"\n\n    def __enter__(self):\n        if self._token is not None:\n            raise RuntimeError(\n                \"Context already entered. Context enter calls cannot be nested.\"\n            )\n        self._token = self.__var__.set(self)\n        return self\n\n    def __exit__(self, *_):\n        if not self._token:\n            raise RuntimeError(\n                \"Asymmetric use of context. Context exit called without an enter.\"\n            )\n        self.__var__.reset(self._token)\n        self._token = None\n\n    @classmethod\n    def get(cls: Type[T]) -> Optional[T]:\n        return cls.__var__.get(None)\n\n    def copy(self, **kwargs):\n        \"\"\"\n        Duplicate the context model, optionally choosing which fields to include, exclude, or change.\n\n        Attributes:\n            include: Fields to include in new model.\n            exclude: Fields to exclude from new model, as with values this takes precedence over include.\n            update: Values to change/add in the new model. Note: the data is not validated before creating\n                the new model - you should trust this data.\n            deep: Set to `True` to make a deep copy of the model.\n\n        Returns:\n            A new model instance.\n        \"\"\"\n        # Remove the token on copy to avoid re-entrance errors\n        new = super().copy(**kwargs)\n        new._token = None\n        return new\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.ContextModel.copy","title":"copy","text":"

    Duplicate the context model, optionally choosing which fields to include, exclude, or change.

    Attributes:

    Name Type Description include

    Fields to include in new model.

    exclude

    Fields to exclude from new model, as with values this takes precedence over include.

    update

    Values to change/add in the new model. Note: the data is not validated before creating the new model - you should trust this data.

    deep

    Set to True to make a deep copy of the model.

    Returns:

    Type Description

    A new model instance.

    Source code in prefect/context.py
    def copy(self, **kwargs):\n    \"\"\"\n    Duplicate the context model, optionally choosing which fields to include, exclude, or change.\n\n    Attributes:\n        include: Fields to include in new model.\n        exclude: Fields to exclude from new model, as with values this takes precedence over include.\n        update: Values to change/add in the new model. Note: the data is not validated before creating\n            the new model - you should trust this data.\n        deep: Set to `True` to make a deep copy of the model.\n\n    Returns:\n        A new model instance.\n    \"\"\"\n    # Remove the token on copy to avoid re-entrance errors\n    new = super().copy(**kwargs)\n    new._token = None\n    return new\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.FlowRunContext","title":"FlowRunContext","text":"

    Bases: RunContext

    The context for a flow run. Data in this context is only available from within a flow run function.

    Attributes:

    Name Type Description flow Optional[Flow]

    The flow instance associated with the run

    flow_run Optional[FlowRun]

    The API metadata for the flow run

    task_runner BaseTaskRunner

    The task runner instance being used for the flow run

    task_run_futures List[PrefectFuture]

    A list of futures for task runs submitted within this flow run

    task_run_states List[State]

    A list of states for task runs created within this flow run

    task_run_results Dict[int, State]

    A mapping of result ids to task run states for this flow run

    flow_run_states List[State]

    A list of states for flow runs created within this flow run

    sync_portal Optional[BlockingPortal]

    A blocking portal for sync task/flow runs in an async flow

    timeout_scope Optional[CancelScope]

    The cancellation scope for flow level timeouts

    Source code in prefect/context.py
    class FlowRunContext(RunContext):\n    \"\"\"\n    The context for a flow run. Data in this context is only available from within a\n    flow run function.\n\n    Attributes:\n        flow: The flow instance associated with the run\n        flow_run: The API metadata for the flow run\n        task_runner: The task runner instance being used for the flow run\n        task_run_futures: A list of futures for task runs submitted within this flow run\n        task_run_states: A list of states for task runs created within this flow run\n        task_run_results: A mapping of result ids to task run states for this flow run\n        flow_run_states: A list of states for flow runs created within this flow run\n        sync_portal: A blocking portal for sync task/flow runs in an async flow\n        timeout_scope: The cancellation scope for flow level timeouts\n    \"\"\"\n\n    flow: Optional[\"Flow\"] = None\n    flow_run: Optional[FlowRun] = None\n    task_runner: BaseTaskRunner\n    log_prints: bool = False\n    parameters: Dict[str, Any]\n\n    # Result handling\n    result_factory: ResultFactory\n\n    # Counter for task calls allowing unique\n    task_run_dynamic_keys: Dict[str, int] = Field(default_factory=dict)\n\n    # Counter for flow pauses\n    observed_flow_pauses: Dict[str, int] = Field(default_factory=dict)\n\n    # Tracking for objects created by this flow run\n    task_run_futures: List[PrefectFuture] = Field(default_factory=list)\n    task_run_states: List[State] = Field(default_factory=list)\n    task_run_results: Dict[int, State] = Field(default_factory=dict)\n    flow_run_states: List[State] = Field(default_factory=list)\n\n    # The synchronous portal is only created for async flows for creating engine calls\n    # from synchronous task and subflow calls\n    sync_portal: Optional[anyio.abc.BlockingPortal] = None\n    timeout_scope: Optional[anyio.abc.CancelScope] = None\n\n    # Task group that can be used for background tasks during the flow run\n    background_tasks: anyio.abc.TaskGroup\n\n    # Events worker to emit events to Prefect Cloud\n    events: Optional[EventsWorker] = None\n\n    __var__ = ContextVar(\"flow_run\")\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.PrefectObjectRegistry","title":"PrefectObjectRegistry","text":"

    Bases: ContextModel

    A context that acts as a registry for all Prefect objects that are registered during load and execution.

    Attributes:

    Name Type Description start_time DateTimeTZ

    The time the object registry was created.

    block_code_execution bool

    If set, flow calls will be ignored.

    capture_failures bool

    If set, failures during init will be silenced and tracked.

    Source code in prefect/context.py
    class PrefectObjectRegistry(ContextModel):\n    \"\"\"\n    A context that acts as a registry for all Prefect objects that are\n    registered during load and execution.\n\n    Attributes:\n        start_time: The time the object registry was created.\n        block_code_execution: If set, flow calls will be ignored.\n        capture_failures: If set, failures during __init__ will be silenced and tracked.\n    \"\"\"\n\n    start_time: DateTimeTZ = Field(default_factory=lambda: pendulum.now(\"UTC\"))\n\n    _instance_registry: Dict[Type[T], List[T]] = PrivateAttr(\n        default_factory=lambda: defaultdict(list)\n    )\n\n    # Failures will be a tuple of (exception, instance, args, kwargs)\n    _instance_init_failures: Dict[Type[T], List[Tuple[Exception, T, Tuple, Dict]]] = (\n        PrivateAttr(default_factory=lambda: defaultdict(list))\n    )\n\n    block_code_execution: bool = False\n    capture_failures: bool = False\n\n    __var__ = ContextVar(\"object_registry\")\n\n    def get_instances(self, type_: Type[T]) -> List[T]:\n        instances = []\n        for registered_type, type_instances in self._instance_registry.items():\n            if type_ in registered_type.mro():\n                instances.extend(type_instances)\n        return instances\n\n    def get_instance_failures(\n        self, type_: Type[T]\n    ) -> List[Tuple[Exception, T, Tuple, Dict]]:\n        failures = []\n        for type__ in type_.mro():\n            failures.extend(self._instance_init_failures[type__])\n        return failures\n\n    def register_instance(self, object):\n        # TODO: Consider using a 'Set' to avoid duplicate entries\n        self._instance_registry[type(object)].append(object)\n\n    def register_init_failure(\n        self, exc: Exception, object: Any, init_args: Tuple, init_kwargs: Dict\n    ):\n        self._instance_init_failures[type(object)].append(\n            (exc, object, init_args, init_kwargs)\n        )\n\n    @classmethod\n    def register_instances(cls, type_: Type):\n        \"\"\"\n        Decorator for a class that adds registration to the `PrefectObjectRegistry`\n        on initialization of instances.\n        \"\"\"\n        __init__ = type_.__init__\n\n        def __register_init__(__self__, *args, **kwargs):\n            registry = cls.get()\n            try:\n                __init__(__self__, *args, **kwargs)\n            except Exception as exc:\n                if not registry or not registry.capture_failures:\n                    raise\n                else:\n                    registry.register_init_failure(exc, __self__, args, kwargs)\n            else:\n                if registry:\n                    registry.register_instance(__self__)\n\n        type_.__init__ = __register_init__\n        return type_\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.PrefectObjectRegistry.register_instances","title":"register_instances classmethod","text":"

    Decorator for a class that adds registration to the PrefectObjectRegistry on initialization of instances.

    Source code in prefect/context.py
    @classmethod\ndef register_instances(cls, type_: Type):\n    \"\"\"\n    Decorator for a class that adds registration to the `PrefectObjectRegistry`\n    on initialization of instances.\n    \"\"\"\n    __init__ = type_.__init__\n\n    def __register_init__(__self__, *args, **kwargs):\n        registry = cls.get()\n        try:\n            __init__(__self__, *args, **kwargs)\n        except Exception as exc:\n            if not registry or not registry.capture_failures:\n                raise\n            else:\n                registry.register_init_failure(exc, __self__, args, kwargs)\n        else:\n            if registry:\n                registry.register_instance(__self__)\n\n    type_.__init__ = __register_init__\n    return type_\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.RunContext","title":"RunContext","text":"

    Bases: ContextModel

    The base context for a flow or task run. Data in this context will always be available when get_run_context is called.

    Attributes:

    Name Type Description start_time DateTimeTZ

    The time the run context was entered

    client PrefectClient

    The Prefect client instance being used for API communication

    Source code in prefect/context.py
    class RunContext(ContextModel):\n    \"\"\"\n    The base context for a flow or task run. Data in this context will always be\n    available when `get_run_context` is called.\n\n    Attributes:\n        start_time: The time the run context was entered\n        client: The Prefect client instance being used for API communication\n    \"\"\"\n\n    start_time: DateTimeTZ = Field(default_factory=lambda: pendulum.now(\"UTC\"))\n    input_keyset: Optional[Dict[str, Dict[str, str]]] = None\n    client: PrefectClient\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.SettingsContext","title":"SettingsContext","text":"

    Bases: ContextModel

    The context for a Prefect settings.

    This allows for safe concurrent access and modification of settings.

    Attributes:

    Name Type Description profile Profile

    The profile that is in use.

    settings Settings

    The complete settings model.

    Source code in prefect/context.py
    class SettingsContext(ContextModel):\n    \"\"\"\n    The context for a Prefect settings.\n\n    This allows for safe concurrent access and modification of settings.\n\n    Attributes:\n        profile: The profile that is in use.\n        settings: The complete settings model.\n    \"\"\"\n\n    profile: Profile\n    settings: Settings\n\n    __var__ = ContextVar(\"settings\")\n\n    def __hash__(self) -> int:\n        return hash(self.settings)\n\n    def __enter__(self):\n        \"\"\"\n        Upon entrance, we ensure the home directory for the profile exists.\n        \"\"\"\n        return_value = super().__enter__()\n\n        try:\n            prefect_home = Path(self.settings.value_of(PREFECT_HOME))\n            prefect_home.mkdir(mode=0o0700, exist_ok=True)\n        except OSError:\n            warnings.warn(\n                (\n                    \"Failed to create the Prefect home directory at \"\n                    f\"{self.settings.value_of(PREFECT_HOME)}\"\n                ),\n                stacklevel=2,\n            )\n\n        return return_value\n\n    @classmethod\n    def get(cls) -> \"SettingsContext\":\n        # Return the global context instead of `None` if no context exists\n        return super().get() or GLOBAL_SETTINGS_CONTEXT\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.TagsContext","title":"TagsContext","text":"

    Bases: ContextModel

    The context for prefect.tags management.

    Attributes:

    Name Type Description current_tags Set[str]

    A set of current tags in the context

    Source code in prefect/context.py
    class TagsContext(ContextModel):\n    \"\"\"\n    The context for `prefect.tags` management.\n\n    Attributes:\n        current_tags: A set of current tags in the context\n    \"\"\"\n\n    current_tags: Set[str] = Field(default_factory=set)\n\n    @classmethod\n    def get(cls) -> \"TagsContext\":\n        # Return an empty `TagsContext` instead of `None` if no context exists\n        return cls.__var__.get(TagsContext())\n\n    __var__ = ContextVar(\"tags\")\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.TaskRunContext","title":"TaskRunContext","text":"

    Bases: RunContext

    The context for a task run. Data in this context is only available from within a task run function.

    Attributes:

    Name Type Description task Task

    The task instance associated with the task run

    task_run TaskRun

    The API metadata for this task run

    Source code in prefect/context.py
    class TaskRunContext(RunContext):\n    \"\"\"\n    The context for a task run. Data in this context is only available from within a\n    task run function.\n\n    Attributes:\n        task: The task instance associated with the task run\n        task_run: The API metadata for this task run\n    \"\"\"\n\n    task: \"Task\"\n    task_run: TaskRun\n    log_prints: bool = False\n    parameters: Dict[str, Any]\n\n    # Result handling\n    result_factory: ResultFactory\n\n    __var__ = ContextVar(\"task_run\")\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.get_run_context","title":"get_run_context","text":"

    Get the current run context from within a task or flow function.

    Returns:

    Type Description Union[FlowRunContext, TaskRunContext]

    A FlowRunContext or TaskRunContext depending on the function type.

    Raises RuntimeError: If called outside of a flow or task run.

    Source code in prefect/context.py
    def get_run_context() -> Union[FlowRunContext, TaskRunContext]:\n    \"\"\"\n    Get the current run context from within a task or flow function.\n\n    Returns:\n        A `FlowRunContext` or `TaskRunContext` depending on the function type.\n\n    Raises\n        RuntimeError: If called outside of a flow or task run.\n    \"\"\"\n    task_run_ctx = TaskRunContext.get()\n    if task_run_ctx:\n        return task_run_ctx\n\n    flow_run_ctx = FlowRunContext.get()\n    if flow_run_ctx:\n        return flow_run_ctx\n\n    raise MissingContextError(\n        \"No run context available. You are not in a flow or task run context.\"\n    )\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.get_settings_context","title":"get_settings_context","text":"

    Get the current settings context which contains profile information and the settings that are being used.

    Generally, the settings that are being used are a combination of values from the profile and environment. See prefect.context.use_profile for more details.

    Source code in prefect/context.py
    def get_settings_context() -> SettingsContext:\n    \"\"\"\n    Get the current settings context which contains profile information and the\n    settings that are being used.\n\n    Generally, the settings that are being used are a combination of values from the\n    profile and environment. See `prefect.context.use_profile` for more details.\n    \"\"\"\n    settings_ctx = SettingsContext.get()\n\n    if not settings_ctx:\n        raise MissingContextError(\"No settings context found.\")\n\n    return settings_ctx\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.registry_from_script","title":"registry_from_script","text":"

    Return a fresh registry with instances populated from execution of a script.

    Source code in prefect/context.py
    def registry_from_script(\n    path: str,\n    block_code_execution: bool = True,\n    capture_failures: bool = True,\n) -> PrefectObjectRegistry:\n    \"\"\"\n    Return a fresh registry with instances populated from execution of a script.\n    \"\"\"\n    with PrefectObjectRegistry(\n        block_code_execution=block_code_execution,\n        capture_failures=capture_failures,\n    ) as registry:\n        load_script_as_module(path)\n\n    return registry\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.root_settings_context","title":"root_settings_context","text":"

    Return the settings context that will exist as the root context for the module.

    The profile to use is determined with the following precedence - Command line via 'prefect --profile ' - Environment variable via 'PREFECT_PROFILE' - Profiles file via the 'active' key Source code in prefect/context.py

    def root_settings_context():\n    \"\"\"\n    Return the settings context that will exist as the root context for the module.\n\n    The profile to use is determined with the following precedence\n    - Command line via 'prefect --profile <name>'\n    - Environment variable via 'PREFECT_PROFILE'\n    - Profiles file via the 'active' key\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    active_name = profiles.active_name\n    profile_source = \"in the profiles file\"\n\n    if \"PREFECT_PROFILE\" in os.environ:\n        active_name = os.environ[\"PREFECT_PROFILE\"]\n        profile_source = \"by environment variable\"\n\n    if (\n        sys.argv[0].endswith(\"/prefect\")\n        and len(sys.argv) >= 3\n        and sys.argv[1] == \"--profile\"\n    ):\n        active_name = sys.argv[2]\n        profile_source = \"by command line argument\"\n\n    if active_name not in profiles.names:\n        print(\n            (\n                f\"WARNING: Active profile {active_name!r} set {profile_source} not \"\n                \"found. The default profile will be used instead. \"\n            ),\n            file=sys.stderr,\n        )\n        active_name = \"default\"\n\n    with use_profile(\n        profiles[active_name],\n        # Override environment variables if the profile was set by the CLI\n        override_environment_variables=profile_source == \"by command line argument\",\n    ) as settings_context:\n        return settings_context\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.tags","title":"tags","text":"

    Context manager to add tags to flow and task run calls.

    Tags are always combined with any existing tags.

    Yields:

    Type Description Set[str]

    The current set of tags

    Examples:

    >>> from prefect import tags, task, flow\n>>> @task\n>>> def my_task():\n>>>     pass\n

    Run a task with tags

    >>> @flow\n>>> def my_flow():\n>>>     with tags(\"a\", \"b\"):\n>>>         my_task()  # has tags: a, b\n

    Run a flow with tags

    >>> @flow\n>>> def my_flow():\n>>>     pass\n>>> with tags(\"a\", \"b\"):\n>>>     my_flow()  # has tags: a, b\n

    Run a task with nested tag contexts

    >>> @flow\n>>> def my_flow():\n>>>     with tags(\"a\", \"b\"):\n>>>         with tags(\"c\", \"d\"):\n>>>             my_task()  # has tags: a, b, c, d\n>>>         my_task()  # has tags: a, b\n

    Inspect the current tags

    >>> @flow\n>>> def my_flow():\n>>>     with tags(\"c\", \"d\"):\n>>>         with tags(\"e\", \"f\") as current_tags:\n>>>              print(current_tags)\n>>> with tags(\"a\", \"b\"):\n>>>     my_flow()\n{\"a\", \"b\", \"c\", \"d\", \"e\", \"f\"}\n
    Source code in prefect/context.py
    @contextmanager\ndef tags(*new_tags: str) -> Set[str]:\n    \"\"\"\n    Context manager to add tags to flow and task run calls.\n\n    Tags are always combined with any existing tags.\n\n    Yields:\n        The current set of tags\n\n    Examples:\n        >>> from prefect import tags, task, flow\n        >>> @task\n        >>> def my_task():\n        >>>     pass\n\n        Run a task with tags\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     with tags(\"a\", \"b\"):\n        >>>         my_task()  # has tags: a, b\n\n        Run a flow with tags\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     pass\n        >>> with tags(\"a\", \"b\"):\n        >>>     my_flow()  # has tags: a, b\n\n        Run a task with nested tag contexts\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     with tags(\"a\", \"b\"):\n        >>>         with tags(\"c\", \"d\"):\n        >>>             my_task()  # has tags: a, b, c, d\n        >>>         my_task()  # has tags: a, b\n\n        Inspect the current tags\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     with tags(\"c\", \"d\"):\n        >>>         with tags(\"e\", \"f\") as current_tags:\n        >>>              print(current_tags)\n        >>> with tags(\"a\", \"b\"):\n        >>>     my_flow()\n        {\"a\", \"b\", \"c\", \"d\", \"e\", \"f\"}\n    \"\"\"\n    current_tags = TagsContext.get().current_tags\n    new_tags = current_tags.union(new_tags)\n    with TagsContext(current_tags=new_tags):\n        yield new_tags\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/context/#prefect.context.use_profile","title":"use_profile","text":"

    Switch to a profile for the duration of this context.

    Profile contexts are confined to an async context in a single thread.

    Parameters:

    Name Type Description Default profile Union[Profile, str]

    The name of the profile to load or an instance of a Profile.

    required override_environment_variable

    If set, variables in the profile will take precedence over current environment variables. By default, environment variables will override profile settings.

    required include_current_context bool

    If set, the new settings will be constructed with the current settings context as a base. If not set, the use_base settings will be loaded from the environment and defaults.

    True

    Yields:

    Type Description

    The created SettingsContext object

    Source code in prefect/context.py
    @contextmanager\ndef use_profile(\n    profile: Union[Profile, str],\n    override_environment_variables: bool = False,\n    include_current_context: bool = True,\n):\n    \"\"\"\n    Switch to a profile for the duration of this context.\n\n    Profile contexts are confined to an async context in a single thread.\n\n    Args:\n        profile: The name of the profile to load or an instance of a Profile.\n        override_environment_variable: If set, variables in the profile will take\n            precedence over current environment variables. By default, environment\n            variables will override profile settings.\n        include_current_context: If set, the new settings will be constructed\n            with the current settings context as a base. If not set, the use_base settings\n            will be loaded from the environment and defaults.\n\n    Yields:\n        The created `SettingsContext` object\n    \"\"\"\n    if isinstance(profile, str):\n        profiles = prefect.settings.load_profiles()\n        profile = profiles[profile]\n\n    if not isinstance(profile, Profile):\n        raise TypeError(\n            f\"Unexpected type {type(profile).__name__!r} for `profile`. \"\n            \"Expected 'str' or 'Profile'.\"\n        )\n\n    # Create a copy of the profiles settings as we will mutate it\n    profile_settings = profile.settings.copy()\n\n    existing_context = SettingsContext.get()\n    if existing_context and include_current_context:\n        settings = existing_context.settings\n    else:\n        settings = prefect.settings.get_settings_from_env()\n\n    if not override_environment_variables:\n        for key in os.environ:\n            if key in prefect.settings.SETTING_VARIABLES:\n                profile_settings.pop(prefect.settings.SETTING_VARIABLES[key], None)\n\n    new_settings = settings.copy_with_update(updates=profile_settings)\n\n    with SettingsContext(profile=profile, settings=new_settings) as ctx:\n        yield ctx\n
    ","tags":["Python API","flow run context","task run context","context"]},{"location":"api-ref/prefect/engine/","title":"prefect.engine","text":"","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine","title":"prefect.engine","text":"

    Client-side execution and orchestration of flows and tasks.

    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine--engine-process-overview","title":"Engine process overview","text":"","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine--flows","title":"Flows","text":"
    • The flow is called by the user or an existing flow run is executed in a new process.

      See Flow.__call__ and prefect.engine.__main__ (python -m prefect.engine)

    • A synchronous function acts as an entrypoint to the engine. The engine executes on a dedicated \"global loop\" thread. For asynchronous flow calls, we return a coroutine from the entrypoint so the user can enter the engine without blocking their event loop.

      See enter_flow_run_engine_from_flow_call, enter_flow_run_engine_from_subprocess

    • The thread that calls the entrypoint waits until orchestration of the flow run completes. This thread is referred to as the \"user\" thread and is usually the \"main\" thread. The thread is not blocked while waiting \u2014 it allows the engine to send work back to it. This allows us to send calls back to the user thread from the global loop thread.

      See wait_for_call_in_loop_thread and call_soon_in_waiting_thread

    • The asynchronous engine branches depending on if the flow run exists already and if there is a parent flow run in the current context.

      See create_then_begin_flow_run, create_and_begin_subflow_run, and retrieve_flow_then_begin_flow_run

    • The asynchronous engine prepares for execution of the flow run. This includes starting the task runner, preparing context, etc.

      See begin_flow_run

    • The flow run is orchestrated through states, calling the user's function as necessary. Generally the user's function is sent for execution on the user thread. If the flow function cannot be safely executed on the user thread, e.g. it is a synchronous child in an asynchronous parent it will be scheduled on a worker thread instead.

      See orchestrate_flow_run, call_soon_in_waiting_thread, call_soon_in_new_thread

    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine--tasks","title":"Tasks","text":"
    • The task is called or submitted by the user. We require that this is always within a flow.

      See Task.__call__ and Task.submit

    • A synchronous function acts as an entrypoint to the engine. Unlike flow calls, this will not block until completion if submit was used.

      See enter_task_run_engine

    • A future is created for the task call. Creation of the task run and submission to the task runner is scheduled as a background task so submission of many tasks can occur concurrently.

      See create_task_run_future and create_task_run_then_submit

    • The engine branches depending on if a future, state, or result is requested. If a future is requested, it is returned immediately to the user thread. Otherwise, the engine will wait for the task run to complete and return the final state or result.

      See get_task_call_return_value

    • An engine function is submitted to the task runner. The task runner will schedule this function for execution on a worker. When executed, it will prepare for orchestration and wait for completion of the run.

      See create_task_run_then_submit and begin_task_run

    • The task run is orchestrated through states, calling the user's function as necessary. The user's function is always executed in a worker thread for isolation.

      See orchestrate_task_run, call_soon_in_new_thread

      _Ideally, for local and sequential task runners we would send the task run to the user thread as we do for flows. See #9855.

    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.begin_flow_run","title":"begin_flow_run async","text":"

    Begins execution of a flow run; blocks until completion of the flow run

    • Starts a task runner
    • Determines the result storage block to use
    • Orchestrates the flow run (runs the user-function and generates tasks)
    • Waits for tasks to complete / shutsdown the task runner
    • Sets a terminal state for the flow run

    Note that the flow_run contains a parameters attribute which is the serialized parameters sent to the backend while the parameters argument here should be the deserialized and validated dictionary of python objects.

    Returns:

    Type Description State

    The final state of the run

    Source code in prefect/engine.py
    async def begin_flow_run(\n    flow: Flow,\n    flow_run: FlowRun,\n    parameters: Dict[str, Any],\n    client: PrefectClient,\n    user_thread: threading.Thread,\n) -> State:\n    \"\"\"\n    Begins execution of a flow run; blocks until completion of the flow run\n\n    - Starts a task runner\n    - Determines the result storage block to use\n    - Orchestrates the flow run (runs the user-function and generates tasks)\n    - Waits for tasks to complete / shutsdown the task runner\n    - Sets a terminal state for the flow run\n\n    Note that the `flow_run` contains a `parameters` attribute which is the serialized\n    parameters sent to the backend while the `parameters` argument here should be the\n    deserialized and validated dictionary of python objects.\n\n    Returns:\n        The final state of the run\n    \"\"\"\n    logger = flow_run_logger(flow_run, flow)\n\n    log_prints = should_log_prints(flow)\n    flow_run_context = PartialModel(FlowRunContext, log_prints=log_prints)\n\n    async with AsyncExitStack() as stack:\n        await stack.enter_async_context(\n            report_flow_run_crashes(flow_run=flow_run, client=client, flow=flow)\n        )\n\n        # Create a task group for background tasks\n        flow_run_context.background_tasks = await stack.enter_async_context(\n            anyio.create_task_group()\n        )\n\n        # If the flow is async, we need to provide a portal so sync tasks can run\n        flow_run_context.sync_portal = (\n            stack.enter_context(start_blocking_portal()) if flow.isasync else None\n        )\n\n        task_runner = flow.task_runner.duplicate()\n        if task_runner is NotImplemented:\n            # Backwards compatibility; will not support concurrent flow runs\n            task_runner = flow.task_runner\n            logger.warning(\n                f\"Task runner {type(task_runner).__name__!r} does not implement the\"\n                \" `duplicate` method and will fail if used for concurrent execution of\"\n                \" the same flow.\"\n            )\n\n        logger.debug(\n            f\"Starting {type(flow.task_runner).__name__!r}; submitted tasks \"\n            f\"will be run {CONCURRENCY_MESSAGES[flow.task_runner.concurrency_type]}...\"\n        )\n\n        flow_run_context.task_runner = await stack.enter_async_context(\n            task_runner.start()\n        )\n\n        flow_run_context.result_factory = await ResultFactory.from_flow(\n            flow, client=client\n        )\n\n        if log_prints:\n            stack.enter_context(patch_print())\n\n        terminal_or_paused_state = await orchestrate_flow_run(\n            flow,\n            flow_run=flow_run,\n            parameters=parameters,\n            wait_for=None,\n            client=client,\n            partial_flow_run_context=flow_run_context,\n            # Orchestration needs to be interruptible if it has a timeout\n            interruptible=flow.timeout_seconds is not None,\n            user_thread=user_thread,\n        )\n\n    if terminal_or_paused_state.is_paused():\n        timeout = terminal_or_paused_state.state_details.pause_timeout\n        msg = \"Currently paused and suspending execution.\"\n        if timeout:\n            msg += f\" Resume before {timeout.to_rfc3339_string()} to finish execution.\"\n        logger.log(level=logging.INFO, msg=msg)\n        await APILogHandler.aflush()\n\n        return terminal_or_paused_state\n    else:\n        terminal_state = terminal_or_paused_state\n\n    # If debugging, use the more complete `repr` than the usual `str` description\n    display_state = repr(terminal_state) if PREFECT_DEBUG_MODE else str(terminal_state)\n\n    logger.log(\n        level=logging.INFO if terminal_state.is_completed() else logging.ERROR,\n        msg=f\"Finished in state {display_state}\",\n    )\n\n    # When a \"root\" flow run finishes, flush logs so we do not have to rely on handling\n    # during interpreter shutdown\n    await APILogHandler.aflush()\n\n    return terminal_state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.begin_task_map","title":"begin_task_map async","text":"

    Async entrypoint for task mapping

    Source code in prefect/engine.py
    async def begin_task_map(\n    task: Task,\n    flow_run_context: FlowRunContext,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n    task_runner: Optional[BaseTaskRunner],\n) -> List[Union[PrefectFuture, Awaitable[PrefectFuture]]]:\n    \"\"\"Async entrypoint for task mapping\"\"\"\n    # We need to resolve some futures to map over their data, collect the upstream\n    # links beforehand to retain relationship tracking.\n    task_inputs = {\n        k: await collect_task_run_inputs(v, max_depth=0) for k, v in parameters.items()\n    }\n\n    # Resolve the top-level parameters in order to get mappable data of a known length.\n    # Nested parameters will be resolved in each mapped child where their relationships\n    # will also be tracked.\n    parameters = await resolve_inputs(parameters, max_depth=1)\n\n    # Ensure that any parameters in kwargs are expanded before this check\n    parameters = explode_variadic_parameter(task.fn, parameters)\n\n    iterable_parameters = {}\n    static_parameters = {}\n    annotated_parameters = {}\n    for key, val in parameters.items():\n        if isinstance(val, (allow_failure, quote)):\n            # Unwrap annotated parameters to determine if they are iterable\n            annotated_parameters[key] = val\n            val = val.unwrap()\n\n        if isinstance(val, unmapped):\n            static_parameters[key] = val.value\n        elif isiterable(val):\n            iterable_parameters[key] = list(val)\n        else:\n            static_parameters[key] = val\n\n    if not len(iterable_parameters):\n        raise MappingMissingIterable(\n            \"No iterable parameters were received. Parameters for map must \"\n            f\"include at least one iterable. Parameters: {parameters}\"\n        )\n\n    iterable_parameter_lengths = {\n        key: len(val) for key, val in iterable_parameters.items()\n    }\n    lengths = set(iterable_parameter_lengths.values())\n    if len(lengths) > 1:\n        raise MappingLengthMismatch(\n            \"Received iterable parameters with different lengths. Parameters for map\"\n            f\" must all be the same length. Got lengths: {iterable_parameter_lengths}\"\n        )\n\n    map_length = list(lengths)[0]\n\n    task_runs = []\n    for i in range(map_length):\n        call_parameters = {key: value[i] for key, value in iterable_parameters.items()}\n        call_parameters.update({key: value for key, value in static_parameters.items()})\n\n        # Add default values for parameters; these are skipped earlier since they should\n        # not be mapped over\n        for key, value in get_parameter_defaults(task.fn).items():\n            call_parameters.setdefault(key, value)\n\n        # Re-apply annotations to each key again\n        for key, annotation in annotated_parameters.items():\n            call_parameters[key] = annotation.rewrap(call_parameters[key])\n\n        # Collapse any previously exploded kwargs\n        call_parameters = collapse_variadic_parameters(task.fn, call_parameters)\n\n        task_runs.append(\n            partial(\n                get_task_call_return_value,\n                task=task,\n                flow_run_context=flow_run_context,\n                parameters=call_parameters,\n                wait_for=wait_for,\n                return_type=return_type,\n                task_runner=task_runner,\n                extra_task_inputs=task_inputs,\n            )\n        )\n\n    # Maintain the order of the task runs when using the sequential task runner\n    runner = task_runner if task_runner else flow_run_context.task_runner\n    if runner.concurrency_type == TaskConcurrencyType.SEQUENTIAL:\n        return [await task_run() for task_run in task_runs]\n\n    return await gather(*task_runs)\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.begin_task_run","title":"begin_task_run async","text":"

    Entrypoint for task run execution.

    This function is intended for submission to the task runner.

    This method may be called from a worker so we ensure the settings context has been entered. For example, with a runner that is executing tasks in the same event loop, we will likely not enter the context again because the current context already matches:

    main thread: --> Flow called with settings A --> begin_task_run executes same event loop --> Profile A matches and is not entered again

    However, with execution on a remote environment, we are going to need to ensure the settings for the task run are respected by entering the context:

    main thread: --> Flow called with settings A --> begin_task_run is scheduled on a remote worker, settings A is serialized remote worker: --> Remote worker imports Prefect (may not occur) --> Global settings is loaded with default settings --> begin_task_run executes on a different event loop than the flow --> Current settings is not set or does not match, settings A is entered

    Source code in prefect/engine.py
    async def begin_task_run(\n    task: Task,\n    task_run: TaskRun,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    result_factory: ResultFactory,\n    log_prints: bool,\n    settings: prefect.context.SettingsContext,\n):\n    \"\"\"\n    Entrypoint for task run execution.\n\n    This function is intended for submission to the task runner.\n\n    This method may be called from a worker so we ensure the settings context has been\n    entered. For example, with a runner that is executing tasks in the same event loop,\n    we will likely not enter the context again because the current context already\n    matches:\n\n    main thread:\n    --> Flow called with settings A\n    --> `begin_task_run` executes same event loop\n    --> Profile A matches and is not entered again\n\n    However, with execution on a remote environment, we are going to need to ensure the\n    settings for the task run are respected by entering the context:\n\n    main thread:\n    --> Flow called with settings A\n    --> `begin_task_run` is scheduled on a remote worker, settings A is serialized\n    remote worker:\n    --> Remote worker imports Prefect (may not occur)\n    --> Global settings is loaded with default settings\n    --> `begin_task_run` executes on a different event loop than the flow\n    --> Current settings is not set or does not match, settings A is entered\n    \"\"\"\n    maybe_flow_run_context = prefect.context.FlowRunContext.get()\n\n    async with AsyncExitStack() as stack:\n        # The settings context may be null on a remote worker so we use the safe `.get`\n        # method and compare it to the settings required for this task run\n        if prefect.context.SettingsContext.get() != settings:\n            stack.enter_context(settings)\n            setup_logging()\n\n        if maybe_flow_run_context:\n            # Accessible if on a worker that is running in the same thread as the flow\n            client = maybe_flow_run_context.client\n            # Only run the task in an interruptible thread if it in the same thread as\n            # the flow _and_ the flow run has a timeout attached. If the task is on a\n            # worker, the flow run timeout will not be raised in the worker process.\n            interruptible = maybe_flow_run_context.timeout_scope is not None\n        else:\n            # Otherwise, retrieve a new client\n            client = await stack.enter_async_context(get_client())\n            interruptible = False\n            await stack.enter_async_context(anyio.create_task_group())\n\n        await stack.enter_async_context(report_task_run_crashes(task_run, client))\n\n        # TODO: Use the background tasks group to manage logging for this task\n\n        if log_prints:\n            stack.enter_context(patch_print())\n\n        await check_api_reachable(\n            client, f\"Cannot orchestrate task run '{task_run.id}'\"\n        )\n        try:\n            state = await orchestrate_task_run(\n                task=task,\n                task_run=task_run,\n                parameters=parameters,\n                wait_for=wait_for,\n                result_factory=result_factory,\n                log_prints=log_prints,\n                interruptible=interruptible,\n                client=client,\n            )\n\n            if not maybe_flow_run_context:\n                # When a a task run finishes on a remote worker flush logs to prevent\n                # loss if the process exits\n                await APILogHandler.aflush()\n\n        except Abort as abort:\n            # Task run probably already completed, fetch its state\n            task_run = await client.read_task_run(task_run.id)\n\n            if task_run.state.is_final():\n                task_run_logger(task_run).info(\n                    f\"Task run '{task_run.id}' already finished.\"\n                )\n            else:\n                # TODO: This is a concerning case; we should determine when this occurs\n                #       1. This can occur when the flow run is not in a running state\n                task_run_logger(task_run).warning(\n                    f\"Task run '{task_run.id}' received abort during orchestration: \"\n                    f\"{abort} Task run is in {task_run.state.type.value} state.\"\n                )\n            state = task_run.state\n\n        except Pause:\n            # A pause signal here should mean the flow run suspended, so we\n            # should do the same. We'll look up the flow run's pause state to\n            # try and reuse it, so we capture any data like timeouts.\n            flow_run = await client.read_flow_run(task_run.flow_run_id)\n            if flow_run.state and flow_run.state.is_paused():\n                state = flow_run.state\n            else:\n                state = Suspended()\n\n            task_run_logger(task_run).info(\n                \"Task run encountered a pause signal during orchestration.\"\n            )\n\n        return state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.collect_task_run_inputs","title":"collect_task_run_inputs async","text":"

    This function recurses through an expression to generate a set of any discernible task run inputs it finds in the data structure. It produces a set of all inputs found.

    Examples:

    >>> task_inputs = {\n>>>    k: await collect_task_run_inputs(v) for k, v in parameters.items()\n>>> }\n
    Source code in prefect/engine.py
    async def collect_task_run_inputs(expr: Any, max_depth: int = -1) -> Set[TaskRunInput]:\n    \"\"\"\n    This function recurses through an expression to generate a set of any discernible\n    task run inputs it finds in the data structure. It produces a set of all inputs\n    found.\n\n    Examples:\n        >>> task_inputs = {\n        >>>    k: await collect_task_run_inputs(v) for k, v in parameters.items()\n        >>> }\n    \"\"\"\n    # TODO: This function needs to be updated to detect parameters and constants\n\n    inputs = set()\n    futures = set()\n\n    def add_futures_and_states_to_inputs(obj):\n        if isinstance(obj, PrefectFuture):\n            # We need to wait for futures to be submitted before we can get the task\n            # run id but we want to do so asynchronously\n            futures.add(obj)\n        elif is_state(obj):\n            if obj.state_details.task_run_id:\n                inputs.add(TaskRunResult(id=obj.state_details.task_run_id))\n        # Expressions inside quotes should not be traversed\n        elif isinstance(obj, quote):\n            raise StopVisiting\n        else:\n            state = get_state_for_result(obj)\n            if state and state.state_details.task_run_id:\n                inputs.add(TaskRunResult(id=state.state_details.task_run_id))\n\n    visit_collection(\n        expr,\n        visit_fn=add_futures_and_states_to_inputs,\n        return_data=False,\n        max_depth=max_depth,\n    )\n\n    await asyncio.gather(*[future._wait_for_submission() for future in futures])\n    for future in futures:\n        inputs.add(TaskRunResult(id=future.task_run.id))\n\n    return inputs\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.create_and_begin_subflow_run","title":"create_and_begin_subflow_run async","text":"

    Async entrypoint for flows calls within a flow run

    Subflows differ from parent flows in that they - Resolve futures in passed parameters into values - Create a dummy task for representation in the parent flow - Retrieve default result storage from the parent flow rather than the server

    Returns:

    Type Description Any

    The final state of the run

    Source code in prefect/engine.py
    @inject_client\nasync def create_and_begin_subflow_run(\n    flow: Flow,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n    client: PrefectClient,\n    user_thread: threading.Thread,\n) -> Any:\n    \"\"\"\n    Async entrypoint for flows calls within a flow run\n\n    Subflows differ from parent flows in that they\n    - Resolve futures in passed parameters into values\n    - Create a dummy task for representation in the parent flow\n    - Retrieve default result storage from the parent flow rather than the server\n\n    Returns:\n        The final state of the run\n    \"\"\"\n    parent_flow_run_context = FlowRunContext.get()\n    parent_logger = get_run_logger(parent_flow_run_context)\n    log_prints = should_log_prints(flow)\n    terminal_state = None\n\n    parent_logger.debug(f\"Resolving inputs to {flow.name!r}\")\n    task_inputs = {k: await collect_task_run_inputs(v) for k, v in parameters.items()}\n\n    if wait_for:\n        task_inputs[\"wait_for\"] = await collect_task_run_inputs(wait_for)\n\n    rerunning = parent_flow_run_context.flow_run.run_count > 1\n\n    # Generate a task in the parent flow run to represent the result of the subflow run\n    dummy_task = Task(name=flow.name, fn=flow.fn, version=flow.version)\n    parent_task_run = await client.create_task_run(\n        task=dummy_task,\n        flow_run_id=parent_flow_run_context.flow_run.id,\n        dynamic_key=_dynamic_key_for_task_run(parent_flow_run_context, dummy_task),\n        task_inputs=task_inputs,\n        state=Pending(),\n    )\n\n    # Resolve any task futures in the input\n    parameters = await resolve_inputs(parameters)\n\n    if parent_task_run.state.is_final() and not (\n        rerunning and not parent_task_run.state.is_completed()\n    ):\n        # Retrieve the most recent flow run from the database\n        flow_runs = await client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                parent_task_run_id={\"any_\": [parent_task_run.id]}\n            ),\n            sort=FlowRunSort.EXPECTED_START_TIME_ASC,\n        )\n        flow_run = flow_runs[-1]\n\n        # Set up variables required downstream\n        terminal_state = flow_run.state\n        logger = flow_run_logger(flow_run, flow)\n\n    else:\n        flow_run = await client.create_flow_run(\n            flow,\n            parameters=flow.serialize_parameters(parameters),\n            parent_task_run_id=parent_task_run.id,\n            state=parent_task_run.state if not rerunning else Pending(),\n            tags=TagsContext.get().current_tags,\n        )\n\n        parent_logger.info(\n            f\"Created subflow run {flow_run.name!r} for flow {flow.name!r}\"\n        )\n\n        logger = flow_run_logger(flow_run, flow)\n        ui_url = PREFECT_UI_URL.value()\n        if ui_url:\n            logger.info(\n                f\"View at {ui_url}/flow-runs/flow-run/{flow_run.id}\",\n                extra={\"send_to_api\": False},\n            )\n\n        result_factory = await ResultFactory.from_flow(\n            flow, client=parent_flow_run_context.client\n        )\n\n        if flow.should_validate_parameters:\n            try:\n                parameters = flow.validate_parameters(parameters)\n            except Exception:\n                message = \"Validation of flow parameters failed with error:\"\n                logger.exception(message)\n                terminal_state = await propose_state(\n                    client,\n                    state=await exception_to_failed_state(\n                        message=message, result_factory=result_factory\n                    ),\n                    flow_run_id=flow_run.id,\n                )\n\n        if terminal_state is None or not terminal_state.is_final():\n            async with AsyncExitStack() as stack:\n                await stack.enter_async_context(\n                    report_flow_run_crashes(flow_run=flow_run, client=client, flow=flow)\n                )\n\n                task_runner = flow.task_runner.duplicate()\n                if task_runner is NotImplemented:\n                    # Backwards compatibility; will not support concurrent flow runs\n                    task_runner = flow.task_runner\n                    logger.warning(\n                        f\"Task runner {type(task_runner).__name__!r} does not implement\"\n                        \" the `duplicate` method and will fail if used for concurrent\"\n                        \" execution of the same flow.\"\n                    )\n\n                await stack.enter_async_context(task_runner.start())\n\n                if log_prints:\n                    stack.enter_context(patch_print())\n\n                terminal_state = await orchestrate_flow_run(\n                    flow,\n                    flow_run=flow_run,\n                    parameters=parameters,\n                    wait_for=wait_for,\n                    # If the parent flow run has a timeout, then this one needs to be\n                    # interruptible as well\n                    interruptible=parent_flow_run_context.timeout_scope is not None,\n                    client=client,\n                    partial_flow_run_context=PartialModel(\n                        FlowRunContext,\n                        sync_portal=parent_flow_run_context.sync_portal,\n                        task_runner=task_runner,\n                        background_tasks=parent_flow_run_context.background_tasks,\n                        result_factory=result_factory,\n                        log_prints=log_prints,\n                    ),\n                    user_thread=user_thread,\n                )\n\n    # Display the full state (including the result) if debugging\n    display_state = repr(terminal_state) if PREFECT_DEBUG_MODE else str(terminal_state)\n    logger.log(\n        level=logging.INFO if terminal_state.is_completed() else logging.ERROR,\n        msg=f\"Finished in state {display_state}\",\n    )\n\n    # Track the subflow state so the parent flow can use it to determine its final state\n    parent_flow_run_context.flow_run_states.append(terminal_state)\n\n    if return_type == \"state\":\n        return terminal_state\n    elif return_type == \"result\":\n        return await terminal_state.result(fetch=True)\n    else:\n        raise ValueError(f\"Invalid return type for flow engine {return_type!r}.\")\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.create_then_begin_flow_run","title":"create_then_begin_flow_run async","text":"

    Async entrypoint for flow calls

    Creates the flow run in the backend, then enters the main flow run engine.

    Source code in prefect/engine.py
    @inject_client\nasync def create_then_begin_flow_run(\n    flow: Flow,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n    client: PrefectClient,\n    user_thread: threading.Thread,\n) -> Any:\n    \"\"\"\n    Async entrypoint for flow calls\n\n    Creates the flow run in the backend, then enters the main flow run engine.\n    \"\"\"\n    # TODO: Returns a `State` depending on `return_type` and we can add an overload to\n    #       the function signature to clarify this eventually.\n\n    await check_api_reachable(client, \"Cannot create flow run\")\n\n    state = Pending()\n    if flow.should_validate_parameters:\n        try:\n            parameters = flow.validate_parameters(parameters)\n        except Exception:\n            state = await exception_to_failed_state(\n                message=\"Validation of flow parameters failed with error:\"\n            )\n\n    flow_run = await client.create_flow_run(\n        flow,\n        # Send serialized parameters to the backend\n        parameters=flow.serialize_parameters(parameters),\n        state=state,\n        tags=TagsContext.get().current_tags,\n    )\n\n    engine_logger.info(f\"Created flow run {flow_run.name!r} for flow {flow.name!r}\")\n\n    logger = flow_run_logger(flow_run, flow)\n\n    ui_url = PREFECT_UI_URL.value()\n    if ui_url:\n        logger.info(\n            f\"View at {ui_url}/flow-runs/flow-run/{flow_run.id}\",\n            extra={\"send_to_api\": False},\n        )\n\n    if state.is_failed():\n        logger.error(state.message)\n        engine_logger.info(\n            f\"Flow run {flow_run.name!r} received invalid parameters and is marked as\"\n            \" failed.\"\n        )\n    else:\n        state = await begin_flow_run(\n            flow=flow,\n            flow_run=flow_run,\n            parameters=parameters,\n            client=client,\n            user_thread=user_thread,\n        )\n\n    if return_type == \"state\":\n        return state\n    elif return_type == \"result\":\n        return await state.result(fetch=True)\n    else:\n        raise ValueError(f\"Invalid return type for flow engine {return_type!r}.\")\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.enter_flow_run_engine_from_flow_call","title":"enter_flow_run_engine_from_flow_call","text":"

    Sync entrypoint for flow calls.

    This function does the heavy lifting of ensuring we can get into an async context for flow run execution with minimal overhead.

    Source code in prefect/engine.py
    def enter_flow_run_engine_from_flow_call(\n    flow: Flow,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n) -> Union[State, Awaitable[State]]:\n    \"\"\"\n    Sync entrypoint for flow calls.\n\n    This function does the heavy lifting of ensuring we can get into an async context\n    for flow run execution with minimal overhead.\n    \"\"\"\n    setup_logging()\n\n    registry = PrefectObjectRegistry.get()\n    if registry and registry.block_code_execution:\n        engine_logger.warning(\n            f\"Script loading is in progress, flow {flow.name!r} will not be executed.\"\n            \" Consider updating the script to only call the flow if executed\"\n            f' directly:\\n\\n\\tif __name__ == \"__main__\":\\n\\t\\t{flow.fn.__name__}()'\n        )\n        return None\n\n    if TaskRunContext.get():\n        raise RuntimeError(\n            \"Flows cannot be run from within tasks. Did you mean to call this \"\n            \"flow in a flow?\"\n        )\n\n    parent_flow_run_context = FlowRunContext.get()\n    is_subflow_run = parent_flow_run_context is not None\n\n    if wait_for is not None and not is_subflow_run:\n        raise ValueError(\"Only flows run as subflows can wait for dependencies.\")\n\n    begin_run = create_call(\n        create_and_begin_subflow_run if is_subflow_run else create_then_begin_flow_run,\n        flow=flow,\n        parameters=parameters,\n        wait_for=wait_for,\n        return_type=return_type,\n        client=parent_flow_run_context.client if is_subflow_run else None,\n        user_thread=threading.current_thread(),\n    )\n\n    # On completion of root flows, wait for the global thread to ensure that\n    # any work there is complete\n    done_callbacks = (\n        [create_call(wait_for_global_loop_exit)] if not is_subflow_run else None\n    )\n\n    # WARNING: You must define any context managers here to pass to our concurrency\n    # api instead of entering them in here in the engine entrypoint. Otherwise, async\n    # flows will not use the context as this function _exits_ to return an awaitable to\n    # the user. Generally, you should enter contexts _within_ the async `begin_run`\n    # instead but if you need to enter a context from the main thread you'll need to do\n    # it here.\n    contexts = [capture_sigterm()]\n\n    if flow.isasync and (\n        not is_subflow_run or (is_subflow_run and parent_flow_run_context.flow.isasync)\n    ):\n        # return a coro for the user to await if the flow is async\n        # unless it is an async subflow called in a sync flow\n        retval = from_async.wait_for_call_in_loop_thread(\n            begin_run,\n            done_callbacks=done_callbacks,\n            contexts=contexts,\n        )\n\n    else:\n        retval = from_sync.wait_for_call_in_loop_thread(\n            begin_run,\n            done_callbacks=done_callbacks,\n            contexts=contexts,\n        )\n\n    return retval\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.enter_flow_run_engine_from_subprocess","title":"enter_flow_run_engine_from_subprocess","text":"

    Sync entrypoint for flow runs that have been submitted for execution by an agent

    Differs from enter_flow_run_engine_from_flow_call in that we have a flow run id but not a flow object. The flow must be retrieved before execution can begin. Additionally, this assumes that the caller is always in a context without an event loop as this should be called from a fresh process.

    Source code in prefect/engine.py
    def enter_flow_run_engine_from_subprocess(flow_run_id: UUID) -> State:\n    \"\"\"\n    Sync entrypoint for flow runs that have been submitted for execution by an agent\n\n    Differs from `enter_flow_run_engine_from_flow_call` in that we have a flow run id\n    but not a flow object. The flow must be retrieved before execution can begin.\n    Additionally, this assumes that the caller is always in a context without an event\n    loop as this should be called from a fresh process.\n    \"\"\"\n\n    # Ensure collections are imported and have the opportunity to register types before\n    # loading the user code from the deployment\n    prefect.plugins.load_prefect_collections()\n\n    setup_logging()\n\n    state = from_sync.wait_for_call_in_loop_thread(\n        create_call(\n            retrieve_flow_then_begin_flow_run,\n            flow_run_id,\n            user_thread=threading.current_thread(),\n        ),\n        contexts=[capture_sigterm()],\n    )\n\n    APILogHandler.flush()\n    return state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.enter_task_run_engine","title":"enter_task_run_engine","text":"

    Sync entrypoint for task calls

    Source code in prefect/engine.py
    def enter_task_run_engine(\n    task: Task,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    return_type: EngineReturnType,\n    task_runner: Optional[BaseTaskRunner],\n    mapped: bool,\n) -> Union[PrefectFuture, Awaitable[PrefectFuture]]:\n    \"\"\"\n    Sync entrypoint for task calls\n    \"\"\"\n\n    flow_run_context = FlowRunContext.get()\n    if not flow_run_context:\n        raise RuntimeError(\n            \"Tasks cannot be run outside of a flow. To call the underlying task\"\n            \" function outside of a flow use `task.fn()`.\"\n        )\n\n    if TaskRunContext.get():\n        raise RuntimeError(\n            \"Tasks cannot be run from within tasks. Did you mean to call this \"\n            \"task in a flow?\"\n        )\n\n    if flow_run_context.timeout_scope and flow_run_context.timeout_scope.cancel_called:\n        raise TimeoutError(\"Flow run timed out\")\n\n    begin_run = create_call(\n        begin_task_map if mapped else get_task_call_return_value,\n        task=task,\n        flow_run_context=flow_run_context,\n        parameters=parameters,\n        wait_for=wait_for,\n        return_type=return_type,\n        task_runner=task_runner,\n    )\n\n    if task.isasync and flow_run_context.flow.isasync:\n        # return a coro for the user to await if an async task in an async flow\n        return from_async.wait_for_call_in_loop_thread(begin_run)\n    else:\n        return from_sync.wait_for_call_in_loop_thread(begin_run)\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.get_state_for_result","title":"get_state_for_result","text":"

    Get the state related to a result object.

    link_state_to_result must have been called first.

    Source code in prefect/engine.py
    def get_state_for_result(obj: Any) -> Optional[State]:\n    \"\"\"\n    Get the state related to a result object.\n\n    `link_state_to_result` must have been called first.\n    \"\"\"\n    flow_run_context = FlowRunContext.get()\n    if flow_run_context:\n        return flow_run_context.task_run_results.get(id(obj))\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.link_state_to_result","title":"link_state_to_result","text":"

    Caches a link between a state and a result and its components using the id of the components to map to the state. The cache is persisted to the current flow run context since task relationships are limited to within a flow run.

    This allows dependency tracking to occur when results are passed around. Note: Because id is used, we cannot cache links between singleton objects.

    We only cache the relationship between components 1-layer deep. Example: Given the result [1, [\"a\",\"b\"], (\"c\",)], the following elements will be mapped to the state: - [1, [\"a\",\"b\"], (\"c\",)] - [\"a\",\"b\"] - (\"c\",)

    Note: the int `1` will not be mapped to the state because it is a singleton.\n

    Other Notes: We do not hash the result because: - If changes are made to the object in the flow between task calls, we can still track that they are related. - Hashing can be expensive. - Not all objects are hashable.

    We do not set an attribute, e.g. __prefect_state__, on the result because:

    • Mutating user's objects is dangerous.
    • Unrelated equality comparisons can break unexpectedly.
    • The field can be preserved on copy.
    • We cannot set this attribute on Python built-ins.
    Source code in prefect/engine.py
    def link_state_to_result(state: State, result: Any) -> None:\n    \"\"\"\n    Caches a link between a state and a result and its components using\n    the `id` of the components to map to the state. The cache is persisted to the\n    current flow run context since task relationships are limited to within a flow run.\n\n    This allows dependency tracking to occur when results are passed around.\n    Note: Because `id` is used, we cannot cache links between singleton objects.\n\n    We only cache the relationship between components 1-layer deep.\n    Example:\n        Given the result [1, [\"a\",\"b\"], (\"c\",)], the following elements will be\n        mapped to the state:\n        - [1, [\"a\",\"b\"], (\"c\",)]\n        - [\"a\",\"b\"]\n        - (\"c\",)\n\n        Note: the int `1` will not be mapped to the state because it is a singleton.\n\n    Other Notes:\n    We do not hash the result because:\n    - If changes are made to the object in the flow between task calls, we can still\n      track that they are related.\n    - Hashing can be expensive.\n    - Not all objects are hashable.\n\n    We do not set an attribute, e.g. `__prefect_state__`, on the result because:\n\n    - Mutating user's objects is dangerous.\n    - Unrelated equality comparisons can break unexpectedly.\n    - The field can be preserved on copy.\n    - We cannot set this attribute on Python built-ins.\n    \"\"\"\n\n    flow_run_context = FlowRunContext.get()\n\n    def link_if_trackable(obj: Any) -> None:\n        \"\"\"Track connection between a task run result and its associated state if it has a unique ID.\n\n        We cannot track booleans, Ellipsis, None, NotImplemented, or the integers from -5 to 256\n        because they are singletons.\n\n        This function will mutate the State if the object is an untrackable type by setting the value\n        for `State.state_details.untrackable_result` to `True`.\n\n        \"\"\"\n        if (type(obj) in UNTRACKABLE_TYPES) or (\n            isinstance(obj, int) and (-5 <= obj <= 256)\n        ):\n            state.state_details.untrackable_result = True\n            return\n        flow_run_context.task_run_results[id(obj)] = state\n\n    if flow_run_context:\n        visit_collection(expr=result, visit_fn=link_if_trackable, max_depth=1)\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.orchestrate_flow_run","title":"orchestrate_flow_run async","text":"

    Executes a flow run.

    Note on flow timeouts

    Since async flows are run directly in the main event loop, timeout behavior will match that described by anyio. If the flow is awaiting something, it will immediately return; otherwise, the next time it awaits it will exit. Sync flows are being task runner in a worker thread, which cannot be interrupted. The worker thread will exit at the next task call. The worker thread also has access to the status of the cancellation scope at FlowRunContext.timeout_scope.cancel_called which allows it to raise a TimeoutError to respect the timeout.

    Returns:

    Type Description State

    The final state of the run

    Source code in prefect/engine.py
    async def orchestrate_flow_run(\n    flow: Flow,\n    flow_run: FlowRun,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    interruptible: bool,\n    client: PrefectClient,\n    partial_flow_run_context: PartialModel[FlowRunContext],\n    user_thread: threading.Thread,\n) -> State:\n    \"\"\"\n    Executes a flow run.\n\n    Note on flow timeouts:\n        Since async flows are run directly in the main event loop, timeout behavior will\n        match that described by anyio. If the flow is awaiting something, it will\n        immediately return; otherwise, the next time it awaits it will exit. Sync flows\n        are being task runner in a worker thread, which cannot be interrupted. The worker\n        thread will exit at the next task call. The worker thread also has access to the\n        status of the cancellation scope at `FlowRunContext.timeout_scope.cancel_called`\n        which allows it to raise a `TimeoutError` to respect the timeout.\n\n    Returns:\n        The final state of the run\n    \"\"\"\n\n    logger = flow_run_logger(flow_run, flow)\n\n    flow_run_context = None\n    parent_flow_run_context = FlowRunContext.get()\n\n    try:\n        # Resolve futures in any non-data dependencies to ensure they are ready\n        if wait_for is not None:\n            await resolve_inputs({\"wait_for\": wait_for}, return_data=False)\n    except UpstreamTaskError as upstream_exc:\n        return await propose_state(\n            client,\n            Pending(name=\"NotReady\", message=str(upstream_exc)),\n            flow_run_id=flow_run.id,\n            # if orchestrating a run already in a pending state, force orchestration to\n            # update the state name\n            force=flow_run.state.is_pending(),\n        )\n\n    state = await propose_state(client, Running(), flow_run_id=flow_run.id)\n\n    # flag to ensure we only update the flow run name once\n    run_name_set = False\n\n    while state.is_running():\n        waited_for_task_runs = False\n\n        # Update the flow run to the latest data\n        flow_run = await client.read_flow_run(flow_run.id)\n        try:\n            with partial_flow_run_context.finalize(\n                flow=flow,\n                flow_run=flow_run,\n                client=client,\n                parameters=parameters,\n            ) as flow_run_context:\n                # update flow run name\n                if not run_name_set and flow.flow_run_name:\n                    flow_run_name = _resolve_custom_flow_run_name(\n                        flow=flow, parameters=parameters\n                    )\n\n                    await client.update_flow_run(\n                        flow_run_id=flow_run.id, name=flow_run_name\n                    )\n                    logger.extra[\"flow_run_name\"] = flow_run_name\n                    logger.debug(\n                        f\"Renamed flow run {flow_run.name!r} to {flow_run_name!r}\"\n                    )\n                    flow_run.name = flow_run_name\n                    run_name_set = True\n\n                args, kwargs = parameters_to_args_kwargs(flow.fn, parameters)\n                logger.debug(\n                    f\"Executing flow {flow.name!r} for flow run {flow_run.name!r}...\"\n                )\n\n                if PREFECT_DEBUG_MODE:\n                    logger.debug(f\"Executing {call_repr(flow.fn, *args, **kwargs)}\")\n                else:\n                    logger.debug(\n                        \"Beginning execution...\", extra={\"state_message\": True}\n                    )\n\n                flow_call = create_call(flow.fn, *args, **kwargs)\n\n                # This check for a parent call is needed for cases where the engine\n                # was entered directly during testing\n                parent_call = get_current_call()\n\n                if parent_call and (\n                    not parent_flow_run_context\n                    or (\n                        parent_flow_run_context\n                        and parent_flow_run_context.flow.isasync == flow.isasync\n                    )\n                ):\n                    from_async.call_soon_in_waiting_thread(\n                        flow_call, thread=user_thread, timeout=flow.timeout_seconds\n                    )\n                else:\n                    from_async.call_soon_in_new_thread(\n                        flow_call, timeout=flow.timeout_seconds\n                    )\n\n                result = await flow_call.aresult()\n\n                waited_for_task_runs = await wait_for_task_runs_and_report_crashes(\n                    flow_run_context.task_run_futures, client=client\n                )\n        except PausedRun as exc:\n            # could get raised either via utility or by returning Paused from a task run\n            # if a task run pauses, we set its state as the flow's state\n            # to preserve reschedule and timeout behavior\n            paused_flow_run = await client.read_flow_run(flow_run.id)\n            if paused_flow_run.state.is_running():\n                state = await propose_state(\n                    client,\n                    state=exc.state,\n                    flow_run_id=flow_run.id,\n                )\n\n                return state\n            paused_flow_run_state = paused_flow_run.state\n            return paused_flow_run_state\n        except CancelledError as exc:\n            if not flow_call.timedout():\n                # If the flow call was not cancelled by us; this is a crash\n                raise\n            # Construct a new exception as `TimeoutError`\n            original = exc\n            exc = TimeoutError()\n            exc.__cause__ = original\n            logger.exception(\"Encountered exception during execution:\")\n            terminal_state = await exception_to_failed_state(\n                exc,\n                message=f\"Flow run exceeded timeout of {flow.timeout_seconds} seconds\",\n                result_factory=flow_run_context.result_factory,\n                name=\"TimedOut\",\n            )\n        except Exception:\n            # Generic exception in user code\n            logger.exception(\"Encountered exception during execution:\")\n            terminal_state = await exception_to_failed_state(\n                message=\"Flow run encountered an exception.\",\n                result_factory=flow_run_context.result_factory,\n            )\n        else:\n            if result is None:\n                # All tasks and subflows are reference tasks if there is no return value\n                # If there are no tasks, use `None` instead of an empty iterable\n                result = (\n                    flow_run_context.task_run_futures\n                    + flow_run_context.task_run_states\n                    + flow_run_context.flow_run_states\n                ) or None\n\n            terminal_state = await return_value_to_state(\n                await resolve_futures_to_states(result),\n                result_factory=flow_run_context.result_factory,\n            )\n\n        if not waited_for_task_runs:\n            # An exception occurred that prevented us from waiting for task runs to\n            # complete. Ensure that we wait for them before proposing a final state\n            # for the flow run.\n            await wait_for_task_runs_and_report_crashes(\n                flow_run_context.task_run_futures, client=client\n            )\n\n        # Before setting the flow run state, store state.data using\n        # block storage and send the resulting data document to the Prefect API instead.\n        # This prevents the pickled return value of flow runs\n        # from being sent to the Prefect API and stored in the Prefect database.\n        # state.data is left as is, otherwise we would have to load\n        # the data from block storage again after storing.\n        state = await propose_state(\n            client,\n            state=terminal_state,\n            flow_run_id=flow_run.id,\n        )\n\n        await _run_flow_hooks(flow=flow, flow_run=flow_run, state=state)\n\n        if state.type != terminal_state.type and PREFECT_DEBUG_MODE:\n            logger.debug(\n                (\n                    f\"Received new state {state} when proposing final state\"\n                    f\" {terminal_state}\"\n                ),\n                extra={\"send_to_api\": False},\n            )\n\n        if not state.is_final() and not state.is_paused():\n            logger.info(\n                (\n                    f\"Received non-final state {state.name!r} when proposing final\"\n                    f\" state {terminal_state.name!r} and will attempt to run again...\"\n                ),\n                extra={\"send_to_api\": False},\n            )\n            # Attempt to enter a running state again\n            state = await propose_state(client, Running(), flow_run_id=flow_run.id)\n\n    return state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.orchestrate_task_run","title":"orchestrate_task_run async","text":"

    Execute a task run

    This function should be submitted to an task runner. We must construct the context here instead of receiving it already populated since we may be in a new environment.

    Proposes a RUNNING state, then - if accepted, the task user function will be run - if rejected, the received state will be returned

    When the user function is run, the result will be used to determine a final state - if an exception is encountered, it is trapped and stored in a FAILED state - otherwise, return_value_to_state is used to determine the state

    If the final state is COMPLETED, we generate a cache key as specified by the task

    The final state is then proposed - if accepted, this is the final state and will be returned - if rejected and a new final state is provided, it will be returned - if rejected and a non-final state is provided, we will attempt to enter a RUNNING state again

    Returns:

    Type Description State

    The final state of the run

    Source code in prefect/engine.py
    async def orchestrate_task_run(\n    task: Task,\n    task_run: TaskRun,\n    parameters: Dict[str, Any],\n    wait_for: Optional[Iterable[PrefectFuture]],\n    result_factory: ResultFactory,\n    log_prints: bool,\n    interruptible: bool,\n    client: PrefectClient,\n) -> State:\n    \"\"\"\n    Execute a task run\n\n    This function should be submitted to an task runner. We must construct the context\n    here instead of receiving it already populated since we may be in a new environment.\n\n    Proposes a RUNNING state, then\n    - if accepted, the task user function will be run\n    - if rejected, the received state will be returned\n\n    When the user function is run, the result will be used to determine a final state\n    - if an exception is encountered, it is trapped and stored in a FAILED state\n    - otherwise, `return_value_to_state` is used to determine the state\n\n    If the final state is COMPLETED, we generate a cache key as specified by the task\n\n    The final state is then proposed\n    - if accepted, this is the final state and will be returned\n    - if rejected and a new final state is provided, it will be returned\n    - if rejected and a non-final state is provided, we will attempt to enter a RUNNING\n        state again\n\n    Returns:\n        The final state of the run\n    \"\"\"\n    flow_run_context = prefect.context.FlowRunContext.get()\n    if flow_run_context:\n        flow_run = flow_run_context.flow_run\n    else:\n        flow_run = await client.read_flow_run(task_run.flow_run_id)\n    logger = task_run_logger(task_run, task=task, flow_run=flow_run)\n\n    partial_task_run_context = PartialModel(\n        TaskRunContext,\n        task_run=task_run,\n        task=task,\n        client=client,\n        result_factory=result_factory,\n        log_prints=log_prints,\n    )\n    task_introspection_start_time = time.perf_counter()\n    try:\n        # Resolve futures in parameters into data\n        resolved_parameters = await resolve_inputs(parameters)\n        # Resolve futures in any non-data dependencies to ensure they are ready\n        await resolve_inputs({\"wait_for\": wait_for}, return_data=False)\n    except UpstreamTaskError as upstream_exc:\n        return await propose_state(\n            client,\n            Pending(name=\"NotReady\", message=str(upstream_exc)),\n            task_run_id=task_run.id,\n            # if orchestrating a run already in a pending state, force orchestration to\n            # update the state name\n            force=task_run.state.is_pending(),\n        )\n    task_introspection_end_time = time.perf_counter()\n\n    introspection_time = round(\n        task_introspection_end_time - task_introspection_start_time, 3\n    )\n    threshold = PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD.value()\n    if threshold and introspection_time > threshold:\n        logger.warning(\n            f\"Task parameter introspection took {introspection_time} seconds \"\n            f\", exceeding `PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD` of {threshold}. \"\n            \"Try wrapping large task parameters with \"\n            \"`prefect.utilities.annotations.quote` for increased performance, \"\n            \"e.g. `my_task(quote(param))`. To disable this message set \"\n            \"`PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD=0`.\"\n        )\n\n    # Generate the cache key to attach to proposed states\n    # The cache key uses a TaskRunContext that does not include a `timeout_context``\n    cache_key = (\n        task.cache_key_fn(\n            partial_task_run_context.finalize(parameters=resolved_parameters),\n            resolved_parameters,\n        )\n        if task.cache_key_fn\n        else None\n    )\n\n    task_run_context = partial_task_run_context.finalize(parameters=resolved_parameters)\n\n    # Ignore the cached results for a cache key, default = false\n    # Setting on task level overrules the Prefect setting (env var)\n    refresh_cache = (\n        task.refresh_cache\n        if task.refresh_cache is not None\n        else PREFECT_TASKS_REFRESH_CACHE.value()\n    )\n\n    # Emit an event to capture that the task run was in the `PENDING` state.\n    last_event = _emit_task_run_state_change_event(\n        task_run=task_run, initial_state=None, validated_state=task_run.state\n    )\n    last_state = task_run.state\n\n    # Transition from `PENDING` -> `RUNNING`\n    try:\n        state = await propose_state(\n            client,\n            Running(\n                state_details=StateDetails(\n                    cache_key=cache_key, refresh_cache=refresh_cache\n                )\n            ),\n            task_run_id=task_run.id,\n        )\n    except Pause as exc:\n        # We shouldn't get a pause signal without a state, but if this happens,\n        # just use a Paused state to assume an in-process pause.\n        state = exc.state if exc.state else Paused()\n\n        # If a flow submits tasks and then pauses, we may reach this point due\n        # to concurrency timing because the tasks will try to transition after\n        # the flow run has paused. Orchestration will send back a Paused state\n        # for the task runs.\n        if state.state_details.pause_reschedule:\n            # If we're being asked to pause and reschedule, we should exit the\n            # task and expect to be resumed later.\n            raise\n\n    if state.is_paused():\n        BACKOFF_MAX = 10  # Seconds\n        backoff_count = 0\n\n        async def tick():\n            nonlocal backoff_count\n            if backoff_count < BACKOFF_MAX:\n                backoff_count += 1\n            interval = 1 + backoff_count + random.random() * backoff_count\n            await anyio.sleep(interval)\n\n        # Enter a loop to wait for the task run to be resumed, i.e.\n        # become Pending, and then propose a Running state again.\n        while True:\n            await tick()\n\n            # Propose a Running state again. We do this instead of reading the\n            # task run because if the flow run times out, this lets\n            # orchestration fail the task run.\n            try:\n                state = await propose_state(\n                    client,\n                    Running(\n                        state_details=StateDetails(\n                            cache_key=cache_key, refresh_cache=refresh_cache\n                        )\n                    ),\n                    task_run_id=task_run.id,\n                )\n            except Pause as exc:\n                if not exc.state:\n                    continue\n\n                if exc.state.state_details.pause_reschedule:\n                    # If the pause state includes pause_reschedule, we should exit the\n                    # task and expect to be resumed later. We've already checked for this\n                    # above, but we check again here in case the state changed; e.g. the\n                    # flow run suspended.\n                    raise\n                else:\n                    # Propose a Running state again.\n                    continue\n            else:\n                break\n\n    # Emit an event to capture the result of proposing a `RUNNING` state.\n    last_event = _emit_task_run_state_change_event(\n        task_run=task_run,\n        initial_state=last_state,\n        validated_state=state,\n        follows=last_event,\n    )\n    last_state = state\n\n    # flag to ensure we only update the task run name once\n    run_name_set = False\n\n    # Only run the task if we enter a `RUNNING` state\n    while state.is_running():\n        # Retrieve the latest metadata for the task run context\n        task_run = await client.read_task_run(task_run.id)\n\n        with task_run_context.copy(\n            update={\"task_run\": task_run, \"start_time\": pendulum.now(\"UTC\")}\n        ):\n            try:\n                args, kwargs = parameters_to_args_kwargs(task.fn, resolved_parameters)\n                # update task run name\n                if not run_name_set and task.task_run_name:\n                    task_run_name = _resolve_custom_task_run_name(\n                        task=task, parameters=resolved_parameters\n                    )\n                    await client.set_task_run_name(\n                        task_run_id=task_run.id, name=task_run_name\n                    )\n                    logger.extra[\"task_run_name\"] = task_run_name\n                    logger.debug(\n                        f\"Renamed task run {task_run.name!r} to {task_run_name!r}\"\n                    )\n                    task_run.name = task_run_name\n                    run_name_set = True\n\n                if PREFECT_DEBUG_MODE.value():\n                    logger.debug(f\"Executing {call_repr(task.fn, *args, **kwargs)}\")\n                else:\n                    logger.debug(\n                        \"Beginning execution...\", extra={\"state_message\": True}\n                    )\n\n                call = from_async.call_soon_in_new_thread(\n                    create_call(task.fn, *args, **kwargs), timeout=task.timeout_seconds\n                )\n                result = await call.aresult()\n\n            except (CancelledError, asyncio.CancelledError) as exc:\n                if not call.timedout():\n                    # If the task call was not cancelled by us; this is a crash\n                    raise\n                # Construct a new exception as `TimeoutError`\n                original = exc\n                exc = TimeoutError()\n                exc.__cause__ = original\n                logger.exception(\"Encountered exception during execution:\")\n                terminal_state = await exception_to_failed_state(\n                    exc,\n                    message=(\n                        f\"Task run exceeded timeout of {task.timeout_seconds} seconds\"\n                    ),\n                    result_factory=task_run_context.result_factory,\n                    name=\"TimedOut\",\n                )\n            except Exception as exc:\n                logger.exception(\"Encountered exception during execution:\")\n                terminal_state = await exception_to_failed_state(\n                    exc,\n                    message=\"Task run encountered an exception\",\n                    result_factory=task_run_context.result_factory,\n                )\n            else:\n                terminal_state = await return_value_to_state(\n                    result,\n                    result_factory=task_run_context.result_factory,\n                )\n\n                # for COMPLETED tasks, add the cache key and expiration\n                if terminal_state.is_completed():\n                    terminal_state.state_details.cache_expiration = (\n                        (pendulum.now(\"utc\") + task.cache_expiration)\n                        if task.cache_expiration\n                        else None\n                    )\n                    terminal_state.state_details.cache_key = cache_key\n\n            if terminal_state.is_failed():\n                # Defer to user to decide whether failure is retriable\n                terminal_state.state_details.retriable = (\n                    await _check_task_failure_retriable(task, task_run, terminal_state)\n                )\n            state = await propose_state(client, terminal_state, task_run_id=task_run.id)\n\n            last_event = _emit_task_run_state_change_event(\n                task_run=task_run,\n                initial_state=last_state,\n                validated_state=state,\n                follows=last_event,\n            )\n            last_state = state\n\n            await _run_task_hooks(\n                task=task,\n                task_run=task_run,\n                state=state,\n            )\n\n            if state.type != terminal_state.type and PREFECT_DEBUG_MODE:\n                logger.debug(\n                    (\n                        f\"Received new state {state} when proposing final state\"\n                        f\" {terminal_state}\"\n                    ),\n                    extra={\"send_to_api\": False},\n                )\n\n            if not state.is_final() and not state.is_paused():\n                logger.info(\n                    (\n                        f\"Received non-final state {state.name!r} when proposing final\"\n                        f\" state {terminal_state.name!r} and will attempt to run\"\n                        \" again...\"\n                    ),\n                    extra={\"send_to_api\": False},\n                )\n                # Attempt to enter a running state again\n                state = await propose_state(client, Running(), task_run_id=task_run.id)\n                last_event = _emit_task_run_state_change_event(\n                    task_run=task_run,\n                    initial_state=last_state,\n                    validated_state=state,\n                    follows=last_event,\n                )\n                last_state = state\n\n    # If debugging, use the more complete `repr` than the usual `str` description\n    display_state = repr(state) if PREFECT_DEBUG_MODE else str(state)\n\n    logger.log(\n        level=logging.INFO if state.is_completed() else logging.ERROR,\n        msg=f\"Finished in state {display_state}\",\n    )\n\n    return state\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.pause_flow_run","title":"pause_flow_run async","text":"

    Pauses the current flow run by blocking execution until resumed.

    When called within a flow run, execution will block and no downstream tasks will run until the flow is resumed. Task runs that have already started will continue running. A timeout parameter can be passed that will fail the flow run if it has not been resumed within the specified time.

    Parameters:

    Name Type Description Default flow_run_id UUID

    a flow run id. If supplied, this function will attempt to pause the specified flow run outside of the flow run process. When paused, the flow run will continue execution until the NEXT task is orchestrated, at which point the flow will exit. Any tasks that have already started will run until completion. When resumed, the flow run will be rescheduled to finish execution. In order pause a flow run in this way, the flow needs to have an associated deployment and results need to be configured with the persist_results option.

    None timeout int

    the number of seconds to wait for the flow to be resumed before failing. Defaults to 1 hour (3600 seconds). If the pause timeout exceeds any configured flow-level timeout, the flow might fail even after resuming.

    3600 poll_interval int

    The number of seconds between checking whether the flow has been resumed. Defaults to 10 seconds.

    10 reschedule bool

    Flag that will reschedule the flow run if resumed. Instead of blocking execution, the flow will gracefully exit (with no result returned) instead. To use this flag, a flow needs to have an associated deployment and results need to be configured with the persist_results option.

    False key str

    An optional key to prevent calling pauses more than once. This defaults to the number of pauses observed by the flow so far, and prevents pauses that use the \"reschedule\" option from running the same pause twice. A custom key can be supplied for custom pausing behavior.

    None wait_for_input Optional[Type[T]]

    a subclass of RunInput. If provided when the flow pauses, the flow will wait for the input to be provided before resuming. If the flow is resumed without providing the input, the flow will fail. If the flow is resumed with the input, the flow will resume and the input will be loaded and returned from this function.

    None
    @task\ndef task_one():\n    for i in range(3):\n        sleep(1)\n\n@flow\ndef my_flow():\n    terminal_state = task_one.submit(return_state=True)\n    if terminal_state.type == StateType.COMPLETED:\n        print(\"Task one succeeded! Pausing flow run..\")\n        pause_flow_run(timeout=2)\n    else:\n        print(\"Task one failed. Skipping pause flow run..\")\n
    Source code in prefect/engine.py
    @sync_compatible\n@deprecated_parameter(\n    \"flow_run_id\", start_date=\"Dec 2023\", help=\"Use `suspend_flow_run` instead.\"\n)\n@deprecated_parameter(\n    \"reschedule\",\n    start_date=\"Dec 2023\",\n    when=lambda p: p is True,\n    help=\"Use `suspend_flow_run` instead.\",\n)\n@experimental_parameter(\n    \"wait_for_input\", group=\"flow_run_input\", when=lambda y: y is not None\n)\nasync def pause_flow_run(\n    wait_for_input: Optional[Type[T]] = None,\n    flow_run_id: UUID = None,\n    timeout: int = 3600,\n    poll_interval: int = 10,\n    reschedule: bool = False,\n    key: str = None,\n):\n    \"\"\"\n    Pauses the current flow run by blocking execution until resumed.\n\n    When called within a flow run, execution will block and no downstream tasks will\n    run until the flow is resumed. Task runs that have already started will continue\n    running. A timeout parameter can be passed that will fail the flow run if it has not\n    been resumed within the specified time.\n\n    Args:\n        flow_run_id: a flow run id. If supplied, this function will attempt to pause\n            the specified flow run outside of the flow run process. When paused, the\n            flow run will continue execution until the NEXT task is orchestrated, at\n            which point the flow will exit. Any tasks that have already started will\n            run until completion. When resumed, the flow run will be rescheduled to\n            finish execution. In order pause a flow run in this way, the flow needs to\n            have an associated deployment and results need to be configured with the\n            `persist_results` option.\n        timeout: the number of seconds to wait for the flow to be resumed before\n            failing. Defaults to 1 hour (3600 seconds). If the pause timeout exceeds\n            any configured flow-level timeout, the flow might fail even after resuming.\n        poll_interval: The number of seconds between checking whether the flow has been\n            resumed. Defaults to 10 seconds.\n        reschedule: Flag that will reschedule the flow run if resumed. Instead of\n            blocking execution, the flow will gracefully exit (with no result returned)\n            instead. To use this flag, a flow needs to have an associated deployment and\n            results need to be configured with the `persist_results` option.\n        key: An optional key to prevent calling pauses more than once. This defaults to\n            the number of pauses observed by the flow so far, and prevents pauses that\n            use the \"reschedule\" option from running the same pause twice. A custom key\n            can be supplied for custom pausing behavior.\n        wait_for_input: a subclass of `RunInput`. If provided when the flow pauses, the\n            flow will wait for the input to be provided before resuming. If the flow is\n            resumed without providing the input, the flow will fail. If the flow is\n            resumed with the input, the flow will resume and the input will be loaded\n            and returned from this function.\n\n    Example:\n    ```python\n    @task\n    def task_one():\n        for i in range(3):\n            sleep(1)\n\n    @flow\n    def my_flow():\n        terminal_state = task_one.submit(return_state=True)\n        if terminal_state.type == StateType.COMPLETED:\n            print(\"Task one succeeded! Pausing flow run..\")\n            pause_flow_run(timeout=2)\n        else:\n            print(\"Task one failed. Skipping pause flow run..\")\n    ```\n\n    \"\"\"\n    if flow_run_id:\n        if wait_for_input is not None:\n            raise RuntimeError(\"Cannot wait for input when pausing out of process.\")\n\n        return await _out_of_process_pause(\n            flow_run_id=flow_run_id,\n            timeout=timeout,\n            reschedule=reschedule,\n            key=key,\n        )\n    else:\n        return await _in_process_pause(\n            timeout=timeout,\n            poll_interval=poll_interval,\n            reschedule=reschedule,\n            key=key,\n            wait_for_input=wait_for_input,\n        )\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.propose_state","title":"propose_state async","text":"

    Propose a new state for a flow run or task run, invoking Prefect orchestration logic.

    If the proposed state is accepted, the provided state will be augmented with details and returned.

    If the proposed state is rejected, a new state returned by the Prefect API will be returned.

    If the proposed state results in a WAIT instruction from the Prefect API, the function will sleep and attempt to propose the state again.

    If the proposed state results in an ABORT instruction from the Prefect API, an error will be raised.

    Parameters:

    Name Type Description Default state State

    a new state for the task or flow run

    required task_run_id UUID

    an optional task run id, used when proposing task run states

    None flow_run_id UUID

    an optional flow run id, used when proposing flow run states

    None

    Returns:

    Type Description State

    a State model representation of the flow or task run state

    Raises:

    Type Description ValueError

    if neither task_run_id or flow_run_id is provided

    Abort

    if an ABORT instruction is received from the Prefect API

    Source code in prefect/engine.py
    async def propose_state(\n    client: PrefectClient,\n    state: State,\n    force: bool = False,\n    task_run_id: UUID = None,\n    flow_run_id: UUID = None,\n) -> State:\n    \"\"\"\n    Propose a new state for a flow run or task run, invoking Prefect orchestration logic.\n\n    If the proposed state is accepted, the provided `state` will be augmented with\n     details and returned.\n\n    If the proposed state is rejected, a new state returned by the Prefect API will be\n    returned.\n\n    If the proposed state results in a WAIT instruction from the Prefect API, the\n    function will sleep and attempt to propose the state again.\n\n    If the proposed state results in an ABORT instruction from the Prefect API, an\n    error will be raised.\n\n    Args:\n        state: a new state for the task or flow run\n        task_run_id: an optional task run id, used when proposing task run states\n        flow_run_id: an optional flow run id, used when proposing flow run states\n\n    Returns:\n        a [State model][prefect.client.schemas.objects.State] representation of the\n            flow or task run state\n\n    Raises:\n        ValueError: if neither task_run_id or flow_run_id is provided\n        prefect.exceptions.Abort: if an ABORT instruction is received from\n            the Prefect API\n    \"\"\"\n\n    # Determine if working with a task run or flow run\n    if not task_run_id and not flow_run_id:\n        raise ValueError(\"You must provide either a `task_run_id` or `flow_run_id`\")\n\n    # Handle task and sub-flow tracing\n    if state.is_final():\n        if isinstance(state.data, BaseResult) and state.data.has_cached_object():\n            # Avoid fetching the result unless it is cached, otherwise we defeat\n            # the purpose of disabling `cache_result_in_memory`\n            result = await state.result(raise_on_failure=False, fetch=True)\n        else:\n            result = state.data\n\n        link_state_to_result(state, result)\n\n    # Handle repeated WAITs in a loop instead of recursively, to avoid\n    # reaching max recursion depth in extreme cases.\n    async def set_state_and_handle_waits(set_state_func) -> OrchestrationResult:\n        response = await set_state_func()\n        while response.status == SetStateStatus.WAIT:\n            engine_logger.debug(\n                f\"Received wait instruction for {response.details.delay_seconds}s: \"\n                f\"{response.details.reason}\"\n            )\n            await anyio.sleep(response.details.delay_seconds)\n            response = await set_state_func()\n        return response\n\n    # Attempt to set the state\n    if task_run_id:\n        set_state = partial(client.set_task_run_state, task_run_id, state, force=force)\n        response = await set_state_and_handle_waits(set_state)\n    elif flow_run_id:\n        set_state = partial(client.set_flow_run_state, flow_run_id, state, force=force)\n        response = await set_state_and_handle_waits(set_state)\n    else:\n        raise ValueError(\n            \"Neither flow run id or task run id were provided. At least one must \"\n            \"be given.\"\n        )\n\n    # Parse the response to return the new state\n    if response.status == SetStateStatus.ACCEPT:\n        # Update the state with the details if provided\n        state.id = response.state.id\n        state.timestamp = response.state.timestamp\n        if response.state.state_details:\n            state.state_details = response.state.state_details\n        return state\n\n    elif response.status == SetStateStatus.ABORT:\n        raise prefect.exceptions.Abort(response.details.reason)\n\n    elif response.status == SetStateStatus.REJECT:\n        if response.state.is_paused():\n            raise Pause(response.details.reason, state=response.state)\n        return response.state\n\n    else:\n        raise ValueError(\n            f\"Received unexpected `SetStateStatus` from server: {response.status!r}\"\n        )\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.report_flow_run_crashes","title":"report_flow_run_crashes async","text":"

    Detect flow run crashes during this context and update the run to a proper final state.

    This context must reraise the exception to properly exit the run.

    Source code in prefect/engine.py
    @asynccontextmanager\nasync def report_flow_run_crashes(flow_run: FlowRun, client: PrefectClient, flow: Flow):\n    \"\"\"\n    Detect flow run crashes during this context and update the run to a proper final\n    state.\n\n    This context _must_ reraise the exception to properly exit the run.\n    \"\"\"\n\n    try:\n        yield\n    except (Abort, Pause):\n        # Do not capture internal signals as crashes\n        raise\n    except BaseException as exc:\n        state = await exception_to_crashed_state(exc)\n        logger = flow_run_logger(flow_run)\n        with anyio.CancelScope(shield=True):\n            logger.error(f\"Crash detected! {state.message}\")\n            logger.debug(\"Crash details:\", exc_info=exc)\n            flow_run_state = await propose_state(client, state, flow_run_id=flow_run.id)\n            engine_logger.debug(\n                f\"Reported crashed flow run {flow_run.name!r} successfully!\"\n            )\n\n            # Only `on_crashed` and `on_cancellation` flow run state change hooks can be called here.\n            # We call the hooks after the state change proposal to `CRASHED` is validated\n            # or rejected (if it is in a `CANCELLING` state).\n            await _run_flow_hooks(\n                flow=flow,\n                flow_run=flow_run,\n                state=flow_run_state,\n            )\n\n        # Reraise the exception\n        raise\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.report_task_run_crashes","title":"report_task_run_crashes async","text":"

    Detect task run crashes during this context and update the run to a proper final state.

    This context must reraise the exception to properly exit the run.

    Source code in prefect/engine.py
    @asynccontextmanager\nasync def report_task_run_crashes(task_run: TaskRun, client: PrefectClient):\n    \"\"\"\n    Detect task run crashes during this context and update the run to a proper final\n    state.\n\n    This context _must_ reraise the exception to properly exit the run.\n    \"\"\"\n    try:\n        yield\n    except (Abort, Pause):\n        # Do not capture internal signals as crashes\n        raise\n    except BaseException as exc:\n        state = await exception_to_crashed_state(exc)\n        logger = task_run_logger(task_run)\n        with anyio.CancelScope(shield=True):\n            logger.error(f\"Crash detected! {state.message}\")\n            logger.debug(\"Crash details:\", exc_info=exc)\n            await client.set_task_run_state(\n                state=state,\n                task_run_id=task_run.id,\n                force=True,\n            )\n            engine_logger.debug(\n                f\"Reported crashed task run {task_run.name!r} successfully!\"\n            )\n\n        # Reraise the exception\n        raise\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.resolve_inputs","title":"resolve_inputs async","text":"

    Resolve any Quote, PrefectFuture, or State types nested in parameters into data.

    Returns:

    Type Description Dict[str, Any]

    A copy of the parameters with resolved data

    Raises:

    Type Description UpstreamTaskError

    If any of the upstream states are not COMPLETED

    Source code in prefect/engine.py
    async def resolve_inputs(\n    parameters: Dict[str, Any], return_data: bool = True, max_depth: int = -1\n) -> Dict[str, Any]:\n    \"\"\"\n    Resolve any `Quote`, `PrefectFuture`, or `State` types nested in parameters into\n    data.\n\n    Returns:\n        A copy of the parameters with resolved data\n\n    Raises:\n        UpstreamTaskError: If any of the upstream states are not `COMPLETED`\n    \"\"\"\n\n    futures = set()\n    states = set()\n    result_by_state = {}\n\n    if not parameters:\n        return {}\n\n    def collect_futures_and_states(expr, context):\n        # Expressions inside quotes should not be traversed\n        if isinstance(context.get(\"annotation\"), quote):\n            raise StopVisiting()\n\n        if isinstance(expr, PrefectFuture):\n            futures.add(expr)\n        if is_state(expr):\n            states.add(expr)\n\n        return expr\n\n    visit_collection(\n        parameters,\n        visit_fn=collect_futures_and_states,\n        return_data=False,\n        max_depth=max_depth,\n        context={},\n    )\n\n    # Wait for all futures so we do not block when we retrieve the state in `resolve_input`\n    states.update(await asyncio.gather(*[future._wait() for future in futures]))\n\n    # Only retrieve the result if requested as it may be expensive\n    if return_data:\n        finished_states = [state for state in states if state.is_final()]\n\n        state_results = await asyncio.gather(\n            *[\n                state.result(raise_on_failure=False, fetch=True)\n                for state in finished_states\n            ]\n        )\n\n        for state, result in zip(finished_states, state_results):\n            result_by_state[state] = result\n\n    def resolve_input(expr, context):\n        state = None\n\n        # Expressions inside quotes should not be modified\n        if isinstance(context.get(\"annotation\"), quote):\n            raise StopVisiting()\n\n        if isinstance(expr, PrefectFuture):\n            state = expr._final_state\n        elif is_state(expr):\n            state = expr\n        else:\n            return expr\n\n        # Do not allow uncompleted upstreams except failures when `allow_failure` has\n        # been used\n        if not state.is_completed() and not (\n            # TODO: Note that the contextual annotation here is only at the current level\n            #       if `allow_failure` is used then another annotation is used, this will\n            #       incorrectly evaluate to false \u2014 to resolve this, we must track all\n            #       annotations wrapping the current expression but this is not yet\n            #       implemented.\n            isinstance(context.get(\"annotation\"), allow_failure)\n            and state.is_failed()\n        ):\n            raise UpstreamTaskError(\n                f\"Upstream task run '{state.state_details.task_run_id}' did not reach a\"\n                \" 'COMPLETED' state.\"\n            )\n\n        return result_by_state.get(state)\n\n    resolved_parameters = {}\n    for parameter, value in parameters.items():\n        try:\n            resolved_parameters[parameter] = visit_collection(\n                value,\n                visit_fn=resolve_input,\n                return_data=return_data,\n                # we're manually going 1 layer deeper here\n                max_depth=max_depth - 1,\n                remove_annotations=True,\n                context={},\n            )\n        except UpstreamTaskError:\n            raise\n        except Exception as exc:\n            raise PrefectException(\n                f\"Failed to resolve inputs in parameter {parameter!r}. If your\"\n                \" parameter type is not supported, consider using the `quote`\"\n                \" annotation to skip resolution of inputs.\"\n            ) from exc\n\n    return resolved_parameters\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.resume_flow_run","title":"resume_flow_run async","text":"

    Resumes a paused flow.

    Parameters:

    Name Type Description Default flow_run_id

    the flow_run_id to resume

    required run_input Optional[Dict]

    a dictionary of inputs to provide to the flow run.

    None Source code in prefect/engine.py
    @sync_compatible\nasync def resume_flow_run(flow_run_id, run_input: Optional[Dict] = None):\n    \"\"\"\n    Resumes a paused flow.\n\n    Args:\n        flow_run_id: the flow_run_id to resume\n        run_input: a dictionary of inputs to provide to the flow run.\n    \"\"\"\n    client = get_client()\n    flow_run = await client.read_flow_run(flow_run_id)\n\n    if not flow_run.state.is_paused():\n        raise NotPausedError(\"Cannot resume a run that isn't paused!\")\n\n    response = await client.resume_flow_run(flow_run_id, run_input=run_input)\n\n    if response.status == SetStateStatus.REJECT:\n        if response.state.type == StateType.FAILED:\n            raise FlowPauseTimeout(\"Flow run can no longer be resumed.\")\n        else:\n            raise RuntimeError(f\"Cannot resume this run: {response.details.reason}\")\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.retrieve_flow_then_begin_flow_run","title":"retrieve_flow_then_begin_flow_run async","text":"

    Async entrypoint for flow runs that have been submitted for execution by an agent

    • Retrieves the deployment information
    • Loads the flow object using deployment information
    • Updates the flow run version
    Source code in prefect/engine.py
    @inject_client\nasync def retrieve_flow_then_begin_flow_run(\n    flow_run_id: UUID,\n    client: PrefectClient,\n    user_thread: threading.Thread,\n) -> State:\n    \"\"\"\n    Async entrypoint for flow runs that have been submitted for execution by an agent\n\n    - Retrieves the deployment information\n    - Loads the flow object using deployment information\n    - Updates the flow run version\n    \"\"\"\n    flow_run = await client.read_flow_run(flow_run_id)\n    try:\n        flow = await load_flow_from_flow_run(flow_run, client=client)\n    except Exception:\n        message = \"Flow could not be retrieved from deployment.\"\n        flow_run_logger(flow_run).exception(message)\n        state = await exception_to_failed_state(message=message)\n        await client.set_flow_run_state(\n            state=state, flow_run_id=flow_run_id, force=True\n        )\n        return state\n\n    # Update the flow run policy defaults to match settings on the flow\n    # Note: Mutating the flow run object prevents us from performing another read\n    #       operation if these properties are used by the client downstream\n    if flow_run.empirical_policy.retry_delay is None:\n        flow_run.empirical_policy.retry_delay = flow.retry_delay_seconds\n\n    if flow_run.empirical_policy.retries is None:\n        flow_run.empirical_policy.retries = flow.retries\n\n    await client.update_flow_run(\n        flow_run_id=flow_run_id,\n        flow_version=flow.version,\n        empirical_policy=flow_run.empirical_policy,\n    )\n\n    if flow.should_validate_parameters:\n        failed_state = None\n        try:\n            parameters = flow.validate_parameters(flow_run.parameters)\n        except Exception:\n            message = \"Validation of flow parameters failed with error: \"\n            flow_run_logger(flow_run).exception(message)\n            failed_state = await exception_to_failed_state(message=message)\n\n        if failed_state is not None:\n            await propose_state(\n                client,\n                state=failed_state,\n                flow_run_id=flow_run_id,\n            )\n            return failed_state\n    else:\n        parameters = flow_run.parameters\n\n    # Ensure default values are populated\n    parameters = {**get_parameter_defaults(flow.fn), **parameters}\n\n    return await begin_flow_run(\n        flow=flow,\n        flow_run=flow_run,\n        parameters=parameters,\n        client=client,\n        user_thread=user_thread,\n    )\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/engine/#prefect.engine.suspend_flow_run","title":"suspend_flow_run async","text":"

    Suspends a flow run by stopping code execution until resumed.

    When suspended, the flow run will continue execution until the NEXT task is orchestrated, at which point the flow will exit. Any tasks that have already started will run until completion. When resumed, the flow run will be rescheduled to finish execution. In order suspend a flow run in this way, the flow needs to have an associated deployment and results need to be configured with the persist_results option.

    Parameters:

    Name Type Description Default flow_run_id Optional[UUID]

    a flow run id. If supplied, this function will attempt to suspend the specified flow run. If not supplied will attempt to suspend the current flow run.

    None timeout Optional[int]

    the number of seconds to wait for the flow to be resumed before failing. Defaults to 1 hour (3600 seconds). If the pause timeout exceeds any configured flow-level timeout, the flow might fail even after resuming.

    3600 key Optional[str]

    An optional key to prevent calling suspend more than once. This defaults to a random string and prevents suspends from running the same suspend twice. A custom key can be supplied for custom suspending behavior.

    None wait_for_input Optional[Type[T]]

    a subclass of RunInput. If provided when the flow suspends, the flow will wait for the input to be provided before resuming. If the flow is resumed without providing the input, the flow will fail. If the flow is resumed with the input, the flow will resume and the input will be loaded and returned from this function.

    None Source code in prefect/engine.py
    @sync_compatible\n@inject_client\n@experimental_parameter(\n    \"wait_for_input\", group=\"flow_run_input\", when=lambda y: y is not None\n)\nasync def suspend_flow_run(\n    wait_for_input: Optional[Type[T]] = None,\n    flow_run_id: Optional[UUID] = None,\n    timeout: Optional[int] = 3600,\n    key: Optional[str] = None,\n    client: PrefectClient = None,\n):\n    \"\"\"\n    Suspends a flow run by stopping code execution until resumed.\n\n    When suspended, the flow run will continue execution until the NEXT task is\n    orchestrated, at which point the flow will exit. Any tasks that have\n    already started will run until completion. When resumed, the flow run will\n    be rescheduled to finish execution. In order suspend a flow run in this\n    way, the flow needs to have an associated deployment and results need to be\n    configured with the `persist_results` option.\n\n    Args:\n        flow_run_id: a flow run id. If supplied, this function will attempt to\n            suspend the specified flow run. If not supplied will attempt to\n            suspend the current flow run.\n        timeout: the number of seconds to wait for the flow to be resumed before\n            failing. Defaults to 1 hour (3600 seconds). If the pause timeout\n            exceeds any configured flow-level timeout, the flow might fail even\n            after resuming.\n        key: An optional key to prevent calling suspend more than once. This\n            defaults to a random string and prevents suspends from running the\n            same suspend twice. A custom key can be supplied for custom\n            suspending behavior.\n        wait_for_input: a subclass of `RunInput`. If provided when the flow\n            suspends, the flow will wait for the input to be provided before\n            resuming. If the flow is resumed without providing the input, the\n            flow will fail. If the flow is resumed with the input, the flow\n            will resume and the input will be loaded and returned from this\n            function.\n    \"\"\"\n    context = FlowRunContext.get()\n\n    if flow_run_id is None:\n        if TaskRunContext.get():\n            raise RuntimeError(\"Cannot suspend task runs.\")\n\n        if context is None or context.flow_run is None:\n            raise RuntimeError(\n                \"Flow runs can only be suspended from within a flow run.\"\n            )\n\n        logger = get_run_logger(context=context)\n        logger.info(\n            \"Suspending flow run, execution will be rescheduled when this flow run is\"\n            \" resumed.\"\n        )\n        flow_run_id = context.flow_run.id\n        suspending_current_flow_run = True\n        pause_counter = _observed_flow_pauses(context)\n        pause_key = key or str(pause_counter)\n    else:\n        # Since we're suspending another flow run we need to generate a pause\n        # key that won't conflict with whatever suspends/pauses that flow may\n        # have. Since this method won't be called during that flow run it's\n        # okay that this is non-deterministic.\n        suspending_current_flow_run = False\n        pause_key = key or str(uuid4())\n\n    proposed_state = Suspended(timeout_seconds=timeout, pause_key=pause_key)\n\n    if wait_for_input:\n        run_input_keyset = keyset_from_paused_state(proposed_state)\n        proposed_state.state_details.run_input_keyset = run_input_keyset\n\n    try:\n        state = await propose_state(\n            client=client,\n            state=proposed_state,\n            flow_run_id=flow_run_id,\n        )\n    except Abort as exc:\n        # Aborted requests mean the suspension is not allowed\n        raise RuntimeError(f\"Flow run cannot be suspended: {exc}\")\n\n    if state.is_running():\n        # The orchestrator rejected the suspended state which means that this\n        # suspend has happened before and the flow run has been resumed.\n        if wait_for_input:\n            # The flow run wanted input, so we need to load it and return it\n            # to the user.\n            return await wait_for_input.load(run_input_keyset)\n        return\n\n    if not state.is_paused():\n        # If we receive anything but a PAUSED state, we are unable to continue\n        raise RuntimeError(\n            f\"Flow run cannot be suspended. Received unexpected state from API: {state}\"\n        )\n\n    if wait_for_input:\n        await wait_for_input.save(run_input_keyset)\n\n    if suspending_current_flow_run:\n        # Exit this process so the run can be resubmitted later\n        raise Pause()\n
    ","tags":["Python API","flow runs","orchestration","engine","context"]},{"location":"api-ref/prefect/events/","title":"prefect.events","text":"","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events","title":"prefect.events","text":"","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events.Event","title":"Event","text":"

    Bases: PrefectBaseModel

    The client-side view of an event that has happened to a Resource

    Source code in prefect/events/schemas.py
    class Event(PrefectBaseModel):\n    \"\"\"The client-side view of an event that has happened to a Resource\"\"\"\n\n    occurred: DateTimeTZ = Field(\n        default_factory=pendulum.now,\n        description=\"When the event happened from the sender's perspective\",\n    )\n    event: str = Field(\n        description=\"The name of the event that happened\",\n    )\n    resource: Resource = Field(\n        description=\"The primary Resource this event concerns\",\n    )\n    related: List[RelatedResource] = Field(\n        default_factory=list,\n        description=\"A list of additional Resources involved in this event\",\n    )\n    payload: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"An open-ended set of data describing what happened\",\n    )\n    id: UUID = Field(\n        default_factory=uuid4,\n        description=\"The client-provided identifier of this event\",\n    )\n    follows: Optional[UUID] = Field(\n        None,\n        description=(\n            \"The ID of an event that is known to have occurred prior to this \"\n            \"one. If set, this may be used to establish a more precise \"\n            \"ordering of causally-related events when they occur close enough \"\n            \"together in time that the system may receive them out-of-order.\"\n        ),\n    )\n\n    @property\n    def involved_resources(self) -> Iterable[Resource]:\n        return [self.resource] + list(self.related)\n\n    @validator(\"related\")\n    def enforce_maximum_related_resources(cls, value: List[RelatedResource]):\n        if len(value) > MAXIMUM_RELATED_RESOURCES:\n            raise ValueError(\n                \"The maximum number of related resources \"\n                f\"is {MAXIMUM_RELATED_RESOURCES}\"\n            )\n\n        return value\n
    ","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events.RelatedResource","title":"RelatedResource","text":"

    Bases: Resource

    A Resource with a specific role in an Event

    Source code in prefect/events/schemas.py
    class RelatedResource(Resource):\n    \"\"\"A Resource with a specific role in an Event\"\"\"\n\n    @root_validator(pre=True)\n    def requires_resource_role(cls, values: Dict[str, Any]):\n        labels = values.get(\"__root__\")\n        if not isinstance(labels, dict):\n            return values\n\n        labels = cast(Dict[str, str], labels)\n\n        if \"prefect.resource.role\" not in labels:\n            raise ValueError(\n                \"Related Resources must include the prefect.resource.role label\"\n            )\n        if not labels[\"prefect.resource.role\"]:\n            raise ValueError(\"The prefect.resource.role label must be non-empty\")\n\n        return values\n\n    @property\n    def role(self) -> str:\n        return self[\"prefect.resource.role\"]\n
    ","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events.Resource","title":"Resource","text":"

    Bases: Labelled

    An observable business object of interest to the user

    Source code in prefect/events/schemas.py
    class Resource(Labelled):\n    \"\"\"An observable business object of interest to the user\"\"\"\n\n    @root_validator(pre=True)\n    def enforce_maximum_labels(cls, values: Dict[str, Any]):\n        labels = values.get(\"__root__\")\n        if not isinstance(labels, dict):\n            return values\n\n        if len(labels) > MAXIMUM_LABELS_PER_RESOURCE:\n            raise ValueError(\n                \"The maximum number of labels per resource \"\n                f\"is {MAXIMUM_LABELS_PER_RESOURCE}\"\n            )\n\n        return values\n\n    @root_validator(pre=True)\n    def requires_resource_id(cls, values: Dict[str, Any]):\n        labels = values.get(\"__root__\")\n        if not isinstance(labels, dict):\n            return values\n\n        labels = cast(Dict[str, str], labels)\n\n        if \"prefect.resource.id\" not in labels:\n            raise ValueError(\"Resources must include the prefect.resource.id label\")\n        if not labels[\"prefect.resource.id\"]:\n            raise ValueError(\"The prefect.resource.id label must be non-empty\")\n\n        return values\n\n    @property\n    def id(self) -> str:\n        return self[\"prefect.resource.id\"]\n
    ","tags":["Python API","events"]},{"location":"api-ref/prefect/events/#prefect.events.emit_event","title":"emit_event","text":"

    Send an event to Prefect Cloud.

    Parameters:

    Name Type Description Default event str

    The name of the event that happened.

    required resource Dict[str, str]

    The primary Resource this event concerns.

    required occurred Optional[DateTimeTZ]

    When the event happened from the sender's perspective. Defaults to the current datetime.

    None related Optional[Union[List[Dict[str, str]], List[RelatedResource]]]

    A list of additional Resources involved in this event.

    None payload Optional[Dict[str, Any]]

    An open-ended set of data describing what happened.

    None id Optional[UUID]

    The sender-provided identifier for this event. Defaults to a random UUID.

    None follows Optional[Event]

    The event that preceded this one. If the preceding event happened more than 5 minutes prior to this event the follows relationship will not be set.

    None

    Returns:

    Type Description Optional[Event]

    The event that was emitted if worker is using a client that emit

    Optional[Event]

    events, otherwise None.

    Source code in prefect/events/utilities.py
    def emit_event(\n    event: str,\n    resource: Dict[str, str],\n    occurred: Optional[DateTimeTZ] = None,\n    related: Optional[Union[List[Dict[str, str]], List[RelatedResource]]] = None,\n    payload: Optional[Dict[str, Any]] = None,\n    id: Optional[UUID] = None,\n    follows: Optional[Event] = None,\n) -> Optional[Event]:\n    \"\"\"\n    Send an event to Prefect Cloud.\n\n    Args:\n        event: The name of the event that happened.\n        resource: The primary Resource this event concerns.\n        occurred: When the event happened from the sender's perspective.\n                  Defaults to the current datetime.\n        related: A list of additional Resources involved in this event.\n        payload: An open-ended set of data describing what happened.\n        id: The sender-provided identifier for this event. Defaults to a random\n            UUID.\n        follows: The event that preceded this one. If the preceding event\n            happened more than 5 minutes prior to this event the follows\n            relationship will not be set.\n\n    Returns:\n        The event that was emitted if worker is using a client that emit\n        events, otherwise None.\n    \"\"\"\n    operational_clients = [AssertingEventsClient, PrefectCloudEventsClient]\n    worker_instance = EventsWorker.instance()\n\n    if worker_instance.client_type not in operational_clients:\n        return None\n\n    event_kwargs = {\n        \"event\": event,\n        \"resource\": resource,\n    }\n\n    if occurred is None:\n        occurred = pendulum.now(\"UTC\")\n    event_kwargs[\"occurred\"] = occurred\n\n    if related is not None:\n        event_kwargs[\"related\"] = related\n\n    if payload is not None:\n        event_kwargs[\"payload\"] = payload\n\n    if id is not None:\n        event_kwargs[\"id\"] = id\n\n    if follows is not None:\n        if -TIGHT_TIMING < (occurred - follows.occurred) < TIGHT_TIMING:\n            event_kwargs[\"follows\"] = follows.id\n\n    event_obj = Event(**event_kwargs)\n    worker_instance.send(event_obj)\n\n    return event_obj\n
    ","tags":["Python API","events"]},{"location":"api-ref/prefect/exceptions/","title":"prefect.exceptions","text":"","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions","title":"prefect.exceptions","text":"

    Prefect-specific exceptions.

    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.Abort","title":"Abort","text":"

    Bases: PrefectSignal

    Raised when the API sends an 'ABORT' instruction during state proposal.

    Indicates that the run should exit immediately.

    Source code in prefect/exceptions.py
    class Abort(PrefectSignal):\n    \"\"\"\n    Raised when the API sends an 'ABORT' instruction during state proposal.\n\n    Indicates that the run should exit immediately.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.BlockMissingCapabilities","title":"BlockMissingCapabilities","text":"

    Bases: PrefectException

    Raised when a block does not have required capabilities for a given operation.

    Source code in prefect/exceptions.py
    class BlockMissingCapabilities(PrefectException):\n    \"\"\"\n    Raised when a block does not have required capabilities for a given operation.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.CancelledRun","title":"CancelledRun","text":"

    Bases: PrefectException

    Raised when the result from a cancelled run is retrieved and an exception is not attached.

    This occurs when a string is attached to the state instead of an exception or if the state's data is null.

    Source code in prefect/exceptions.py
    class CancelledRun(PrefectException):\n    \"\"\"\n    Raised when the result from a cancelled run is retrieved and an exception\n    is not attached.\n\n    This occurs when a string is attached to the state instead of an exception\n    or if the state's data is null.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.CrashedRun","title":"CrashedRun","text":"

    Bases: PrefectException

    Raised when the result from a crashed run is retrieved.

    This occurs when a string is attached to the state instead of an exception or if the state's data is null.

    Source code in prefect/exceptions.py
    class CrashedRun(PrefectException):\n    \"\"\"\n    Raised when the result from a crashed run is retrieved.\n\n    This occurs when a string is attached to the state instead of an exception or if\n    the state's data is null.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ExternalSignal","title":"ExternalSignal","text":"

    Bases: BaseException

    Base type for external signal-like exceptions that should never be caught by users.

    Source code in prefect/exceptions.py
    class ExternalSignal(BaseException):\n    \"\"\"\n    Base type for external signal-like exceptions that should never be caught by users.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.FailedRun","title":"FailedRun","text":"

    Bases: PrefectException

    Raised when the result from a failed run is retrieved and an exception is not attached.

    This occurs when a string is attached to the state instead of an exception or if the state's data is null.

    Source code in prefect/exceptions.py
    class FailedRun(PrefectException):\n    \"\"\"\n    Raised when the result from a failed run is retrieved and an exception is not\n    attached.\n\n    This occurs when a string is attached to the state instead of an exception or if\n    the state's data is null.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.FlowPauseTimeout","title":"FlowPauseTimeout","text":"

    Bases: PrefectException

    Raised when a flow pause times out

    Source code in prefect/exceptions.py
    class FlowPauseTimeout(PrefectException):\n    \"\"\"Raised when a flow pause times out\"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.FlowScriptError","title":"FlowScriptError","text":"

    Bases: PrefectException

    Raised when a script errors during evaluation while attempting to load a flow.

    Source code in prefect/exceptions.py
    class FlowScriptError(PrefectException):\n    \"\"\"\n    Raised when a script errors during evaluation while attempting to load a flow.\n    \"\"\"\n\n    def __init__(\n        self,\n        user_exc: Exception,\n        script_path: str,\n    ) -> None:\n        message = f\"Flow script at {script_path!r} encountered an exception\"\n        super().__init__(message)\n\n        self.user_exc = user_exc\n\n    def rich_user_traceback(self, **kwargs):\n        trace = Traceback.extract(\n            type(self.user_exc),\n            self.user_exc,\n            self.user_exc.__traceback__.tb_next.tb_next.tb_next.tb_next,\n        )\n        return Traceback(trace, **kwargs)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InfrastructureError","title":"InfrastructureError","text":"

    Bases: PrefectException

    A base class for exceptions related to infrastructure blocks

    Source code in prefect/exceptions.py
    class InfrastructureError(PrefectException):\n    \"\"\"\n    A base class for exceptions related to infrastructure blocks\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InfrastructureNotAvailable","title":"InfrastructureNotAvailable","text":"

    Bases: PrefectException

    Raised when infrastructure is not accessible from the current machine. For example, if a process was spawned on another machine it cannot be managed.

    Source code in prefect/exceptions.py
    class InfrastructureNotAvailable(PrefectException):\n    \"\"\"\n    Raised when infrastructure is not accessible from the current machine. For example,\n    if a process was spawned on another machine it cannot be managed.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InfrastructureNotFound","title":"InfrastructureNotFound","text":"

    Bases: PrefectException

    Raised when infrastructure is missing, likely because it has exited or been deleted.

    Source code in prefect/exceptions.py
    class InfrastructureNotFound(PrefectException):\n    \"\"\"\n    Raised when infrastructure is missing, likely because it has exited or been\n    deleted.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InvalidNameError","title":"InvalidNameError","text":"

    Bases: PrefectException, ValueError

    Raised when a name contains characters that are not permitted.

    Source code in prefect/exceptions.py
    class InvalidNameError(PrefectException, ValueError):\n    \"\"\"\n    Raised when a name contains characters that are not permitted.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.InvalidRepositoryURLError","title":"InvalidRepositoryURLError","text":"

    Bases: PrefectException

    Raised when an incorrect URL is provided to a GitHub filesystem block.

    Source code in prefect/exceptions.py
    class InvalidRepositoryURLError(PrefectException):\n    \"\"\"Raised when an incorrect URL is provided to a GitHub filesystem block.\"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MappingLengthMismatch","title":"MappingLengthMismatch","text":"

    Bases: PrefectException

    Raised when attempting to call Task.map with arguments of different lengths.

    Source code in prefect/exceptions.py
    class MappingLengthMismatch(PrefectException):\n    \"\"\"\n    Raised when attempting to call Task.map with arguments of different lengths.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MappingMissingIterable","title":"MappingMissingIterable","text":"

    Bases: PrefectException

    Raised when attempting to call Task.map with all static arguments

    Source code in prefect/exceptions.py
    class MappingMissingIterable(PrefectException):\n    \"\"\"\n    Raised when attempting to call Task.map with all static arguments\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MissingContextError","title":"MissingContextError","text":"

    Bases: PrefectException, RuntimeError

    Raised when a method is called that requires a task or flow run context to be active but one cannot be found.

    Source code in prefect/exceptions.py
    class MissingContextError(PrefectException, RuntimeError):\n    \"\"\"\n    Raised when a method is called that requires a task or flow run context to be\n    active but one cannot be found.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MissingFlowError","title":"MissingFlowError","text":"

    Bases: PrefectException

    Raised when a given flow name is not found in the expected script.

    Source code in prefect/exceptions.py
    class MissingFlowError(PrefectException):\n    \"\"\"\n    Raised when a given flow name is not found in the expected script.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MissingProfileError","title":"MissingProfileError","text":"

    Bases: PrefectException, ValueError

    Raised when a profile name does not exist.

    Source code in prefect/exceptions.py
    class MissingProfileError(PrefectException, ValueError):\n    \"\"\"\n    Raised when a profile name does not exist.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.MissingResult","title":"MissingResult","text":"

    Bases: PrefectException

    Raised when a result is missing from a state; often when result persistence is disabled and the state is retrieved from the API.

    Source code in prefect/exceptions.py
    class MissingResult(PrefectException):\n    \"\"\"\n    Raised when a result is missing from a state; often when result persistence is\n    disabled and the state is retrieved from the API.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.NotPausedError","title":"NotPausedError","text":"

    Bases: PrefectException

    Raised when attempting to unpause a run that isn't paused.

    Source code in prefect/exceptions.py
    class NotPausedError(PrefectException):\n    \"\"\"Raised when attempting to unpause a run that isn't paused.\"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ObjectAlreadyExists","title":"ObjectAlreadyExists","text":"

    Bases: PrefectException

    Raised when the client receives a 409 (conflict) from the API.

    Source code in prefect/exceptions.py
    class ObjectAlreadyExists(PrefectException):\n    \"\"\"\n    Raised when the client receives a 409 (conflict) from the API.\n    \"\"\"\n\n    def __init__(self, http_exc: Exception, *args, **kwargs):\n        self.http_exc = http_exc\n        super().__init__(*args, **kwargs)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ObjectNotFound","title":"ObjectNotFound","text":"

    Bases: PrefectException

    Raised when the client receives a 404 (not found) from the API.

    Source code in prefect/exceptions.py
    class ObjectNotFound(PrefectException):\n    \"\"\"\n    Raised when the client receives a 404 (not found) from the API.\n    \"\"\"\n\n    def __init__(self, http_exc: Exception, *args, **kwargs):\n        self.http_exc = http_exc\n        super().__init__(*args, **kwargs)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ParameterBindError","title":"ParameterBindError","text":"

    Bases: TypeError, PrefectException

    Raised when args and kwargs cannot be converted to parameters.

    Source code in prefect/exceptions.py
    class ParameterBindError(TypeError, PrefectException):\n    \"\"\"\n    Raised when args and kwargs cannot be converted to parameters.\n    \"\"\"\n\n    def __init__(self, msg: str):\n        super().__init__(msg)\n\n    @classmethod\n    def from_bind_failure(\n        cls, fn: Callable, exc: TypeError, call_args: List, call_kwargs: Dict\n    ) -> Self:\n        fn_signature = str(inspect.signature(fn)).strip(\"()\")\n\n        base = f\"Error binding parameters for function '{fn.__name__}': {exc}\"\n        signature = f\"Function '{fn.__name__}' has signature '{fn_signature}'\"\n        received = f\"received args: {call_args} and kwargs: {list(call_kwargs.keys())}\"\n        msg = f\"{base}.\\n{signature} but {received}.\"\n        return cls(msg)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ParameterTypeError","title":"ParameterTypeError","text":"

    Bases: PrefectException

    Raised when a parameter does not pass Pydantic type validation.

    Source code in prefect/exceptions.py
    class ParameterTypeError(PrefectException):\n    \"\"\"\n    Raised when a parameter does not pass Pydantic type validation.\n    \"\"\"\n\n    def __init__(self, msg: str):\n        super().__init__(msg)\n\n    @classmethod\n    def from_validation_error(cls, exc: pydantic.ValidationError) -> Self:\n        bad_params = [f'{err[\"loc\"][0]}: {err[\"msg\"]}' for err in exc.errors()]\n        msg = \"Flow run received invalid parameters:\\n - \" + \"\\n - \".join(bad_params)\n        return cls(msg)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.Pause","title":"Pause","text":"

    Bases: PrefectSignal

    Raised when a flow run is PAUSED and needs to exit for resubmission.

    Source code in prefect/exceptions.py
    class Pause(PrefectSignal):\n    \"\"\"\n    Raised when a flow run is PAUSED and needs to exit for resubmission.\n    \"\"\"\n\n    def __init__(self, *args, state=None, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.state = state\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PausedRun","title":"PausedRun","text":"

    Bases: PrefectException

    Raised when the result from a paused run is retrieved.

    Source code in prefect/exceptions.py
    class PausedRun(PrefectException):\n    \"\"\"\n    Raised when the result from a paused run is retrieved.\n    \"\"\"\n\n    def __init__(self, *args, state=None, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.state = state\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PrefectException","title":"PrefectException","text":"

    Bases: Exception

    Base exception type for Prefect errors.

    Source code in prefect/exceptions.py
    class PrefectException(Exception):\n    \"\"\"\n    Base exception type for Prefect errors.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PrefectHTTPStatusError","title":"PrefectHTTPStatusError","text":"

    Bases: HTTPStatusError

    Raised when client receives a Response that contains an HTTPStatusError.

    Used to include API error details in the error messages that the client provides users.

    Source code in prefect/exceptions.py
    class PrefectHTTPStatusError(HTTPStatusError):\n    \"\"\"\n    Raised when client receives a `Response` that contains an HTTPStatusError.\n\n    Used to include API error details in the error messages that the client provides users.\n    \"\"\"\n\n    @classmethod\n    def from_httpx_error(cls: Type[Self], httpx_error: HTTPStatusError) -> Self:\n        \"\"\"\n        Generate a `PrefectHTTPStatusError` from an `httpx.HTTPStatusError`.\n        \"\"\"\n        try:\n            details = httpx_error.response.json()\n        except Exception:\n            details = None\n\n        error_message, *more_info = str(httpx_error).split(\"\\n\")\n\n        if details:\n            message_components = [error_message, f\"Response: {details}\", *more_info]\n        else:\n            message_components = [error_message, *more_info]\n\n        new_message = \"\\n\".join(message_components)\n\n        return cls(\n            new_message, request=httpx_error.request, response=httpx_error.response\n        )\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PrefectHTTPStatusError.from_httpx_error","title":"from_httpx_error classmethod","text":"

    Generate a PrefectHTTPStatusError from an httpx.HTTPStatusError.

    Source code in prefect/exceptions.py
    @classmethod\ndef from_httpx_error(cls: Type[Self], httpx_error: HTTPStatusError) -> Self:\n    \"\"\"\n    Generate a `PrefectHTTPStatusError` from an `httpx.HTTPStatusError`.\n    \"\"\"\n    try:\n        details = httpx_error.response.json()\n    except Exception:\n        details = None\n\n    error_message, *more_info = str(httpx_error).split(\"\\n\")\n\n    if details:\n        message_components = [error_message, f\"Response: {details}\", *more_info]\n    else:\n        message_components = [error_message, *more_info]\n\n    new_message = \"\\n\".join(message_components)\n\n    return cls(\n        new_message, request=httpx_error.request, response=httpx_error.response\n    )\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.PrefectSignal","title":"PrefectSignal","text":"

    Bases: BaseException

    Base type for signal-like exceptions that should never be caught by users.

    Source code in prefect/exceptions.py
    class PrefectSignal(BaseException):\n    \"\"\"\n    Base type for signal-like exceptions that should never be caught by users.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ProtectedBlockError","title":"ProtectedBlockError","text":"

    Bases: PrefectException

    Raised when an operation is prevented due to block protection.

    Source code in prefect/exceptions.py
    class ProtectedBlockError(PrefectException):\n    \"\"\"\n    Raised when an operation is prevented due to block protection.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ReservedArgumentError","title":"ReservedArgumentError","text":"

    Bases: PrefectException, TypeError

    Raised when a function used with Prefect has an argument with a name that is reserved for a Prefect feature

    Source code in prefect/exceptions.py
    class ReservedArgumentError(PrefectException, TypeError):\n    \"\"\"\n    Raised when a function used with Prefect has an argument with a name that is\n    reserved for a Prefect feature\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.ScriptError","title":"ScriptError","text":"

    Bases: PrefectException

    Raised when a script errors during evaluation while attempting to load data

    Source code in prefect/exceptions.py
    class ScriptError(PrefectException):\n    \"\"\"\n    Raised when a script errors during evaluation while attempting to load data\n    \"\"\"\n\n    def __init__(\n        self,\n        user_exc: Exception,\n        path: str,\n    ) -> None:\n        message = f\"Script at {str(path)!r} encountered an exception: {user_exc!r}\"\n        super().__init__(message)\n        self.user_exc = user_exc\n\n        # Strip script run information from the traceback\n        self.user_exc.__traceback__ = _trim_traceback(\n            self.user_exc.__traceback__,\n            remove_modules=[prefect.utilities.importtools],\n        )\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.SignatureMismatchError","title":"SignatureMismatchError","text":"

    Bases: PrefectException, TypeError

    Raised when parameters passed to a function do not match its signature.

    Source code in prefect/exceptions.py
    class SignatureMismatchError(PrefectException, TypeError):\n    \"\"\"Raised when parameters passed to a function do not match its signature.\"\"\"\n\n    def __init__(self, msg: str):\n        super().__init__(msg)\n\n    @classmethod\n    def from_bad_params(cls, expected_params: List[str], provided_params: List[str]):\n        msg = (\n            f\"Function expects parameters {expected_params} but was provided with\"\n            f\" parameters {provided_params}\"\n        )\n        return cls(msg)\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.TerminationSignal","title":"TerminationSignal","text":"

    Bases: ExternalSignal

    Raised when a flow run receives a termination signal.

    Source code in prefect/exceptions.py
    class TerminationSignal(ExternalSignal):\n    \"\"\"\n    Raised when a flow run receives a termination signal.\n    \"\"\"\n\n    def __init__(self, signal: int):\n        self.signal = signal\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.UnfinishedRun","title":"UnfinishedRun","text":"

    Bases: PrefectException

    Raised when the result from a run that is not finished is retrieved.

    For example, if a run is in a SCHEDULED, PENDING, CANCELLING, or RUNNING state.

    Source code in prefect/exceptions.py
    class UnfinishedRun(PrefectException):\n    \"\"\"\n    Raised when the result from a run that is not finished is retrieved.\n\n    For example, if a run is in a SCHEDULED, PENDING, CANCELLING, or RUNNING state.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.UnspecifiedFlowError","title":"UnspecifiedFlowError","text":"

    Bases: PrefectException

    Raised when multiple flows are found in the expected script and no name is given.

    Source code in prefect/exceptions.py
    class UnspecifiedFlowError(PrefectException):\n    \"\"\"\n    Raised when multiple flows are found in the expected script and no name is given.\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.UpstreamTaskError","title":"UpstreamTaskError","text":"

    Bases: PrefectException

    Raised when a task relies on the result of another task but that task is not 'COMPLETE'

    Source code in prefect/exceptions.py
    class UpstreamTaskError(PrefectException):\n    \"\"\"\n    Raised when a task relies on the result of another task but that task is not\n    'COMPLETE'\n    \"\"\"\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/exceptions/#prefect.exceptions.exception_traceback","title":"exception_traceback","text":"

    Convert an exception to a printable string with a traceback

    Source code in prefect/exceptions.py
    def exception_traceback(exc: Exception) -> str:\n    \"\"\"\n    Convert an exception to a printable string with a traceback\n    \"\"\"\n    tb = traceback.TracebackException.from_exception(exc)\n    return \"\".join(list(tb.format()))\n
    ","tags":["Python API","exceptions","error handling","errors"]},{"location":"api-ref/prefect/filesystems/","title":"prefect.filesystems","text":"","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems","title":"prefect.filesystems","text":"","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.Azure","title":"Azure","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on Azure Datalake and Azure Blob Storage.

    Example

    Load stored Azure config:

    from prefect.filesystems import Azure\n\naz_block = Azure.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class Azure(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on Azure Datalake and Azure Blob Storage.\n\n    Example:\n        Load stored Azure config:\n        ```python\n        from prefect.filesystems import Azure\n\n        az_block = Azure.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Azure\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/54e3fa7e00197a4fbd1d82ed62494cb58d08c96a-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#azure\"\n\n    bucket_path: str = Field(\n        default=...,\n        description=\"An Azure storage bucket path.\",\n        example=\"my-bucket/a-directory-within\",\n    )\n    azure_storage_connection_string: Optional[SecretStr] = Field(\n        default=None,\n        title=\"Azure storage connection string\",\n        description=(\n            \"Equivalent to the AZURE_STORAGE_CONNECTION_STRING environment variable.\"\n        ),\n    )\n    azure_storage_account_name: Optional[SecretStr] = Field(\n        default=None,\n        title=\"Azure storage account name\",\n        description=(\n            \"Equivalent to the AZURE_STORAGE_ACCOUNT_NAME environment variable.\"\n        ),\n    )\n    azure_storage_account_key: Optional[SecretStr] = Field(\n        default=None,\n        title=\"Azure storage account key\",\n        description=\"Equivalent to the AZURE_STORAGE_ACCOUNT_KEY environment variable.\",\n    )\n    azure_storage_tenant_id: Optional[SecretStr] = Field(\n        None,\n        title=\"Azure storage tenant ID\",\n        description=\"Equivalent to the AZURE_TENANT_ID environment variable.\",\n    )\n    azure_storage_client_id: Optional[SecretStr] = Field(\n        None,\n        title=\"Azure storage client ID\",\n        description=\"Equivalent to the AZURE_CLIENT_ID environment variable.\",\n    )\n    azure_storage_client_secret: Optional[SecretStr] = Field(\n        None,\n        title=\"Azure storage client secret\",\n        description=\"Equivalent to the AZURE_CLIENT_SECRET environment variable.\",\n    )\n    azure_storage_anon: bool = Field(\n        default=True,\n        title=\"Azure storage anonymous connection\",\n        description=(\n            \"Set the 'anon' flag for ADLFS. This should be False for systems that\"\n            \" require ADLFS to use DefaultAzureCredentials.\"\n        ),\n    )\n\n    _remote_file_system: RemoteFileSystem = None\n\n    @property\n    def basepath(self) -> str:\n        return f\"az://{self.bucket_path}\"\n\n    @property\n    def filesystem(self) -> RemoteFileSystem:\n        settings = {}\n        if self.azure_storage_connection_string:\n            settings[\"connection_string\"] = (\n                self.azure_storage_connection_string.get_secret_value()\n            )\n        if self.azure_storage_account_name:\n            settings[\"account_name\"] = (\n                self.azure_storage_account_name.get_secret_value()\n            )\n        if self.azure_storage_account_key:\n            settings[\"account_key\"] = self.azure_storage_account_key.get_secret_value()\n        if self.azure_storage_tenant_id:\n            settings[\"tenant_id\"] = self.azure_storage_tenant_id.get_secret_value()\n        if self.azure_storage_client_id:\n            settings[\"client_id\"] = self.azure_storage_client_id.get_secret_value()\n        if self.azure_storage_client_secret:\n            settings[\"client_secret\"] = (\n                self.azure_storage_client_secret.get_secret_value()\n            )\n        settings[\"anon\"] = self.azure_storage_anon\n        self._remote_file_system = RemoteFileSystem(\n            basepath=f\"az://{self.bucket_path}\", settings=settings\n        )\n        return self._remote_file_system\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> bytes:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        return await self.filesystem.get_directory(\n            from_path=from_path, local_path=local_path\n        )\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        return await self.filesystem.put_directory(\n            local_path=local_path, to_path=to_path, ignore_file=ignore_file\n        )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        return await self.filesystem.read_path(path)\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        return await self.filesystem.write_path(path=path, content=content)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.Azure.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory.

    Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> bytes:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    return await self.filesystem.get_directory(\n        from_path=from_path, local_path=local_path\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.Azure.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory.

    Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    return await self.filesystem.put_directory(\n        local_path=local_path, to_path=to_path, ignore_file=ignore_file\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GCS","title":"GCS","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on Google Cloud Storage.

    Example

    Load stored GCS config:

    from prefect.filesystems import GCS\n\ngcs_block = GCS.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class GCS(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on Google Cloud Storage.\n\n    Example:\n        Load stored GCS config:\n        ```python\n        from prefect.filesystems import GCS\n\n        gcs_block = GCS.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/422d13bb838cf247eb2b2cf229ce6a2e717d601b-256x256.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#gcs\"\n\n    bucket_path: str = Field(\n        default=...,\n        description=\"A GCS bucket path.\",\n        example=\"my-bucket/a-directory-within\",\n    )\n    service_account_info: Optional[SecretStr] = Field(\n        default=None,\n        description=\"The contents of a service account keyfile as a JSON string.\",\n    )\n    project: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The project the GCS bucket resides in. If not provided, the project will\"\n            \" be inferred from the credentials or environment.\"\n        ),\n    )\n\n    @property\n    def basepath(self) -> str:\n        return f\"gcs://{self.bucket_path}\"\n\n    @property\n    def filesystem(self) -> RemoteFileSystem:\n        settings = {}\n        if self.service_account_info:\n            try:\n                settings[\"token\"] = json.loads(\n                    self.service_account_info.get_secret_value()\n                )\n            except json.JSONDecodeError:\n                raise ValueError(\n                    \"Unable to load provided service_account_info. Please make sure\"\n                    \" that the provided value is a valid JSON string.\"\n                )\n        remote_file_system = RemoteFileSystem(\n            basepath=f\"gcs://{self.bucket_path}\", settings=settings\n        )\n        return remote_file_system\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> bytes:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        return await self.filesystem.get_directory(\n            from_path=from_path, local_path=local_path\n        )\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        return await self.filesystem.put_directory(\n            local_path=local_path, to_path=to_path, ignore_file=ignore_file\n        )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        return await self.filesystem.read_path(path)\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        return await self.filesystem.write_path(path=path, content=content)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GCS.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory.

    Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> bytes:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    return await self.filesystem.get_directory(\n        from_path=from_path, local_path=local_path\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GCS.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory.

    Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    return await self.filesystem.put_directory(\n        local_path=local_path, to_path=to_path, ignore_file=ignore_file\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GitHub","title":"GitHub","text":"

    Bases: ReadableDeploymentStorage

    Interact with files stored on GitHub repositories.

    Source code in prefect/filesystems.py
    class GitHub(ReadableDeploymentStorage):\n    \"\"\"\n    Interact with files stored on GitHub repositories.\n    \"\"\"\n\n    _block_type_name = \"GitHub\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/41971cfecfea5f79ff334164f06ecb34d1038dd4-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#github\"\n\n    repository: str = Field(\n        default=...,\n        description=(\n            \"The URL of a GitHub repository to read from, in either HTTPS or SSH\"\n            \" format.\"\n        ),\n    )\n    reference: Optional[str] = Field(\n        default=None,\n        description=\"An optional reference to pin to; can be a branch name or tag.\",\n    )\n    access_token: Optional[SecretStr] = Field(\n        name=\"Personal Access Token\",\n        default=None,\n        description=(\n            \"A GitHub Personal Access Token (PAT) with repo scope.\"\n            \" To use a fine-grained PAT, provide '{username}:{PAT}' as the value.\"\n        ),\n    )\n    include_git_objects: bool = Field(\n        default=True,\n        description=(\n            \"Whether to include git objects when copying the repo contents to a\"\n            \" directory.\"\n        ),\n    )\n\n    @validator(\"access_token\")\n    def _ensure_credentials_go_with_https(cls, v: str, values: dict) -> str:\n        \"\"\"Ensure that credentials are not provided with 'SSH' formatted GitHub URLs.\n\n        Note: validates `access_token` specifically so that it only fires when\n        private repositories are used.\n        \"\"\"\n        if v is not None:\n            if urllib.parse.urlparse(values[\"repository\"]).scheme != \"https\":\n                raise InvalidRepositoryURLError(\n                    \"Crendentials can only be used with GitHub repositories \"\n                    \"using the 'HTTPS' format. You must either remove the \"\n                    \"credential if you wish to use the 'SSH' format and are not \"\n                    \"using a private repository, or you must change the repository \"\n                    \"URL to the 'HTTPS' format. \"\n                )\n\n        return v\n\n    def _create_repo_url(self) -> str:\n        \"\"\"Format the URL provided to the `git clone` command.\n\n        For private repos: https://<oauth-key>@github.com/<username>/<repo>.git\n        All other repos should be the same as `self.repository`.\n        \"\"\"\n        url_components = urllib.parse.urlparse(self.repository)\n        if url_components.scheme == \"https\" and self.access_token is not None:\n            updated_components = url_components._replace(\n                netloc=f\"{self.access_token.get_secret_value()}@{url_components.netloc}\"\n            )\n            full_url = urllib.parse.urlunparse(updated_components)\n        else:\n            full_url = self.repository\n\n        return full_url\n\n    @staticmethod\n    def _get_paths(\n        dst_dir: Union[str, None], src_dir: str, sub_directory: str\n    ) -> Tuple[str, str]:\n        \"\"\"Returns the fully formed paths for GitHubRepository contents in the form\n        (content_source, content_destination).\n        \"\"\"\n        if dst_dir is None:\n            content_destination = Path(\".\").absolute()\n        else:\n            content_destination = Path(dst_dir)\n\n        content_source = Path(src_dir)\n\n        if sub_directory:\n            content_destination = content_destination.joinpath(sub_directory)\n            content_source = content_source.joinpath(sub_directory)\n\n        return str(content_source), str(content_destination)\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> None:\n        \"\"\"\n        Clones a GitHub project specified in `from_path` to the provided `local_path`;\n        defaults to cloning the repository reference configured on the Block to the\n        present working directory.\n\n        Args:\n            from_path: If provided, interpreted as a subdirectory of the underlying\n                repository that will be copied to the provided local path.\n            local_path: A local path to clone to; defaults to present working directory.\n        \"\"\"\n        # CONSTRUCT COMMAND\n        cmd = [\"git\", \"clone\", self._create_repo_url()]\n        if self.reference:\n            cmd += [\"-b\", self.reference]\n\n        # Limit git history\n        cmd += [\"--depth\", \"1\"]\n\n        # Clone to a temporary directory and move the subdirectory over\n        with TemporaryDirectory(suffix=\"prefect\") as tmp_dir:\n            cmd.append(tmp_dir)\n\n            err_stream = io.StringIO()\n            out_stream = io.StringIO()\n            process = await run_process(cmd, stream_output=(out_stream, err_stream))\n            if process.returncode != 0:\n                err_stream.seek(0)\n                raise OSError(f\"Failed to pull from remote:\\n {err_stream.read()}\")\n\n            content_source, content_destination = self._get_paths(\n                dst_dir=local_path, src_dir=tmp_dir, sub_directory=from_path\n            )\n\n            ignore_func = None\n            if not self.include_git_objects:\n                ignore_func = ignore_patterns(\".git\")\n\n            copytree(\n                src=content_source,\n                dst=content_destination,\n                dirs_exist_ok=True,\n                ignore=ignore_func,\n            )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.GitHub.get_directory","title":"get_directory async","text":"

    Clones a GitHub project specified in from_path to the provided local_path; defaults to cloning the repository reference configured on the Block to the present working directory.

    Parameters:

    Name Type Description Default from_path Optional[str]

    If provided, interpreted as a subdirectory of the underlying repository that will be copied to the provided local path.

    None local_path Optional[str]

    A local path to clone to; defaults to present working directory.

    None Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> None:\n    \"\"\"\n    Clones a GitHub project specified in `from_path` to the provided `local_path`;\n    defaults to cloning the repository reference configured on the Block to the\n    present working directory.\n\n    Args:\n        from_path: If provided, interpreted as a subdirectory of the underlying\n            repository that will be copied to the provided local path.\n        local_path: A local path to clone to; defaults to present working directory.\n    \"\"\"\n    # CONSTRUCT COMMAND\n    cmd = [\"git\", \"clone\", self._create_repo_url()]\n    if self.reference:\n        cmd += [\"-b\", self.reference]\n\n    # Limit git history\n    cmd += [\"--depth\", \"1\"]\n\n    # Clone to a temporary directory and move the subdirectory over\n    with TemporaryDirectory(suffix=\"prefect\") as tmp_dir:\n        cmd.append(tmp_dir)\n\n        err_stream = io.StringIO()\n        out_stream = io.StringIO()\n        process = await run_process(cmd, stream_output=(out_stream, err_stream))\n        if process.returncode != 0:\n            err_stream.seek(0)\n            raise OSError(f\"Failed to pull from remote:\\n {err_stream.read()}\")\n\n        content_source, content_destination = self._get_paths(\n            dst_dir=local_path, src_dir=tmp_dir, sub_directory=from_path\n        )\n\n        ignore_func = None\n        if not self.include_git_objects:\n            ignore_func = ignore_patterns(\".git\")\n\n        copytree(\n            src=content_source,\n            dst=content_destination,\n            dirs_exist_ok=True,\n            ignore=ignore_func,\n        )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.LocalFileSystem","title":"LocalFileSystem","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on a local file system.

    Example

    Load stored local file system config:

    from prefect.filesystems import LocalFileSystem\n\nlocal_file_system_block = LocalFileSystem.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class LocalFileSystem(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on a local file system.\n\n    Example:\n        Load stored local file system config:\n        ```python\n        from prefect.filesystems import LocalFileSystem\n\n        local_file_system_block = LocalFileSystem.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Local File System\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/ad39089fa66d273b943394a68f003f7a19aa850e-48x48.png\"\n    _documentation_url = (\n        \"https://docs.prefect.io/concepts/filesystems/#local-filesystem\"\n    )\n\n    basepath: Optional[str] = Field(\n        default=None, description=\"Default local path for this block to write to.\"\n    )\n\n    @validator(\"basepath\", pre=True)\n    def cast_pathlib(cls, value):\n        if isinstance(value, Path):\n            return str(value)\n        return value\n\n    def _resolve_path(self, path: str) -> Path:\n        # Only resolve the base path at runtime, default to the current directory\n        basepath = (\n            Path(self.basepath).expanduser().resolve()\n            if self.basepath\n            else Path(\".\").resolve()\n        )\n\n        # Determine the path to access relative to the base path, ensuring that paths\n        # outside of the base path are off limits\n        if path is None:\n            return basepath\n\n        path: Path = Path(path).expanduser()\n\n        if not path.is_absolute():\n            path = basepath / path\n        else:\n            path = path.resolve()\n            if basepath not in path.parents and (basepath != path):\n                raise ValueError(\n                    f\"Provided path {path} is outside of the base path {basepath}.\"\n                )\n\n        return path\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: str = None, local_path: str = None\n    ) -> None:\n        \"\"\"\n        Copies a directory from one place to another on the local filesystem.\n\n        Defaults to copying the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        if not from_path:\n            from_path = Path(self.basepath).expanduser().resolve()\n        else:\n            from_path = self._resolve_path(from_path)\n\n        if not local_path:\n            local_path = Path(\".\").resolve()\n        else:\n            local_path = Path(local_path).resolve()\n\n        if from_path == local_path:\n            # If the paths are the same there is no need to copy\n            # and we avoid shutil.copytree raising an error\n            return\n\n        # .prefectignore exists in the original location, not the current location which\n        # is most likely temporary\n        if (from_path / Path(\".prefectignore\")).exists():\n            ignore_func = await self._get_ignore_func(\n                local_path=from_path, ignore_file=from_path / Path(\".prefectignore\")\n            )\n        else:\n            ignore_func = None\n\n        copytree(from_path, local_path, dirs_exist_ok=True, ignore=ignore_func)\n\n    async def _get_ignore_func(self, local_path: str, ignore_file: str):\n        with open(ignore_file, \"r\") as f:\n            ignore_patterns = f.readlines()\n        included_files = filter_files(root=local_path, ignore_patterns=ignore_patterns)\n\n        def ignore_func(directory, files):\n            relative_path = Path(directory).relative_to(local_path)\n\n            files_to_ignore = [\n                f for f in files if str(relative_path / f) not in included_files\n            ]\n            return files_to_ignore\n\n        return ignore_func\n\n    @sync_compatible\n    async def put_directory(\n        self, local_path: str = None, to_path: str = None, ignore_file: str = None\n    ) -> None:\n        \"\"\"\n        Copies a directory from one place to another on the local filesystem.\n\n        Defaults to copying the entire contents of the current working directory to the block's basepath.\n        An `ignore_file` path may be provided that can include gitignore style expressions for filepaths to ignore.\n        \"\"\"\n        destination_path = self._resolve_path(to_path)\n\n        if not local_path:\n            local_path = Path(\".\").absolute()\n\n        if ignore_file:\n            ignore_func = await self._get_ignore_func(\n                local_path=local_path, ignore_file=ignore_file\n            )\n        else:\n            ignore_func = None\n\n        if local_path == destination_path:\n            pass\n        else:\n            copytree(\n                src=local_path,\n                dst=destination_path,\n                ignore=ignore_func,\n                dirs_exist_ok=True,\n            )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        path: Path = self._resolve_path(path)\n\n        # Check if the path exists\n        if not path.exists():\n            raise ValueError(f\"Path {path} does not exist.\")\n\n        # Validate that its a file\n        if not path.is_file():\n            raise ValueError(f\"Path {path} is not a file.\")\n\n        async with await anyio.open_file(str(path), mode=\"rb\") as f:\n            content = await f.read()\n\n        return content\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        path: Path = self._resolve_path(path)\n\n        # Construct the path if it does not exist\n        path.parent.mkdir(exist_ok=True, parents=True)\n\n        # Check if the file already exists\n        if path.exists() and not path.is_file():\n            raise ValueError(f\"Path {path} already exists and is not a file.\")\n\n        async with await anyio.open_file(path, mode=\"wb\") as f:\n            await f.write(content)\n        # Leave path stringify to the OS\n        return str(path)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.LocalFileSystem.get_directory","title":"get_directory async","text":"

    Copies a directory from one place to another on the local filesystem.

    Defaults to copying the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: str = None, local_path: str = None\n) -> None:\n    \"\"\"\n    Copies a directory from one place to another on the local filesystem.\n\n    Defaults to copying the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    if not from_path:\n        from_path = Path(self.basepath).expanduser().resolve()\n    else:\n        from_path = self._resolve_path(from_path)\n\n    if not local_path:\n        local_path = Path(\".\").resolve()\n    else:\n        local_path = Path(local_path).resolve()\n\n    if from_path == local_path:\n        # If the paths are the same there is no need to copy\n        # and we avoid shutil.copytree raising an error\n        return\n\n    # .prefectignore exists in the original location, not the current location which\n    # is most likely temporary\n    if (from_path / Path(\".prefectignore\")).exists():\n        ignore_func = await self._get_ignore_func(\n            local_path=from_path, ignore_file=from_path / Path(\".prefectignore\")\n        )\n    else:\n        ignore_func = None\n\n    copytree(from_path, local_path, dirs_exist_ok=True, ignore=ignore_func)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.LocalFileSystem.put_directory","title":"put_directory async","text":"

    Copies a directory from one place to another on the local filesystem.

    Defaults to copying the entire contents of the current working directory to the block's basepath. An ignore_file path may be provided that can include gitignore style expressions for filepaths to ignore.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self, local_path: str = None, to_path: str = None, ignore_file: str = None\n) -> None:\n    \"\"\"\n    Copies a directory from one place to another on the local filesystem.\n\n    Defaults to copying the entire contents of the current working directory to the block's basepath.\n    An `ignore_file` path may be provided that can include gitignore style expressions for filepaths to ignore.\n    \"\"\"\n    destination_path = self._resolve_path(to_path)\n\n    if not local_path:\n        local_path = Path(\".\").absolute()\n\n    if ignore_file:\n        ignore_func = await self._get_ignore_func(\n            local_path=local_path, ignore_file=ignore_file\n        )\n    else:\n        ignore_func = None\n\n    if local_path == destination_path:\n        pass\n    else:\n        copytree(\n            src=local_path,\n            dst=destination_path,\n            ignore=ignore_func,\n            dirs_exist_ok=True,\n        )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.RemoteFileSystem","title":"RemoteFileSystem","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on a remote file system.

    Supports any remote file system supported by fsspec. The file system is specified using a protocol. For example, \"s3://my-bucket/my-folder/\" will use S3.

    Example

    Load stored remote file system config:

    from prefect.filesystems import RemoteFileSystem\n\nremote_file_system_block = RemoteFileSystem.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class RemoteFileSystem(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on a remote file system.\n\n    Supports any remote file system supported by `fsspec`. The file system is specified\n    using a protocol. For example, \"s3://my-bucket/my-folder/\" will use S3.\n\n    Example:\n        Load stored remote file system config:\n        ```python\n        from prefect.filesystems import RemoteFileSystem\n\n        remote_file_system_block = RemoteFileSystem.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Remote File System\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/e86b41bc0f9c99ba9489abeee83433b43d5c9365-48x48.png\"\n    _documentation_url = (\n        \"https://docs.prefect.io/concepts/filesystems/#remote-file-system\"\n    )\n\n    basepath: str = Field(\n        default=...,\n        description=\"Default path for this block to write to.\",\n        example=\"s3://my-bucket/my-folder/\",\n    )\n    settings: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Additional settings to pass through to fsspec.\",\n    )\n\n    # Cache for the configured fsspec file system used for access\n    _filesystem: fsspec.AbstractFileSystem = None\n\n    @validator(\"basepath\")\n    def check_basepath(cls, value):\n        scheme, netloc, _, _, _ = urllib.parse.urlsplit(value)\n\n        if not scheme:\n            raise ValueError(f\"Base path must start with a scheme. Got {value!r}.\")\n\n        if not netloc:\n            raise ValueError(\n                f\"Base path must include a location after the scheme. Got {value!r}.\"\n            )\n\n        if scheme == \"file\":\n            raise ValueError(\n                \"Base path scheme cannot be 'file'. Use `LocalFileSystem` instead for\"\n                \" local file access.\"\n            )\n\n        return value\n\n    def _resolve_path(self, path: str) -> str:\n        base_scheme, base_netloc, base_urlpath, _, _ = urllib.parse.urlsplit(\n            self.basepath\n        )\n        scheme, netloc, urlpath, _, _ = urllib.parse.urlsplit(path)\n\n        # Confirm that absolute paths are valid\n        if scheme:\n            if scheme != base_scheme:\n                raise ValueError(\n                    f\"Path {path!r} with scheme {scheme!r} must use the same scheme as\"\n                    f\" the base path {base_scheme!r}.\"\n                )\n\n        if netloc:\n            if (netloc != base_netloc) or not urlpath.startswith(base_urlpath):\n                raise ValueError(\n                    f\"Path {path!r} is outside of the base path {self.basepath!r}.\"\n                )\n\n        return f\"{self.basepath.rstrip('/')}/{urlpath.lstrip('/')}\"\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> None:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        if from_path is None:\n            from_path = str(self.basepath)\n        else:\n            from_path = self._resolve_path(from_path)\n\n        if local_path is None:\n            local_path = Path(\".\").absolute()\n\n        # validate that from_path has a trailing slash for proper fsspec behavior across versions\n        if not from_path.endswith(\"/\"):\n            from_path += \"/\"\n\n        return self.filesystem.get(from_path, local_path, recursive=True)\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n        overwrite: bool = True,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        if to_path is None:\n            to_path = str(self.basepath)\n        else:\n            to_path = self._resolve_path(to_path)\n\n        if local_path is None:\n            local_path = \".\"\n\n        included_files = None\n        if ignore_file:\n            with open(ignore_file, \"r\") as f:\n                ignore_patterns = f.readlines()\n\n            included_files = filter_files(\n                local_path, ignore_patterns, include_dirs=True\n            )\n\n        counter = 0\n        for f in Path(local_path).rglob(\"*\"):\n            relative_path = f.relative_to(local_path)\n            if included_files and str(relative_path) not in included_files:\n                continue\n\n            if to_path.endswith(\"/\"):\n                fpath = to_path + relative_path.as_posix()\n            else:\n                fpath = to_path + \"/\" + relative_path.as_posix()\n\n            if f.is_dir():\n                pass\n            else:\n                f = f.as_posix()\n                if overwrite:\n                    self.filesystem.put_file(f, fpath, overwrite=True)\n                else:\n                    self.filesystem.put_file(f, fpath)\n\n                counter += 1\n\n        return counter\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        path = self._resolve_path(path)\n\n        with self.filesystem.open(path, \"rb\") as file:\n            content = await run_sync_in_worker_thread(file.read)\n\n        return content\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        path = self._resolve_path(path)\n        dirpath = path[: path.rindex(\"/\")]\n\n        self.filesystem.makedirs(dirpath, exist_ok=True)\n\n        with self.filesystem.open(path, \"wb\") as file:\n            await run_sync_in_worker_thread(file.write, content)\n        return path\n\n    @property\n    def filesystem(self) -> fsspec.AbstractFileSystem:\n        if not self._filesystem:\n            scheme, _, _, _, _ = urllib.parse.urlsplit(self.basepath)\n\n            try:\n                self._filesystem = fsspec.filesystem(scheme, **self.settings)\n            except ImportError as exc:\n                # The path is a remote file system that uses a lib that is not installed\n                raise RuntimeError(\n                    f\"File system created with scheme {scheme!r} from base path \"\n                    f\"{self.basepath!r} could not be created. \"\n                    \"You are likely missing a Python module required to use the given \"\n                    \"storage protocol.\"\n                ) from exc\n\n        return self._filesystem\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.RemoteFileSystem.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory.

    Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> None:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    if from_path is None:\n        from_path = str(self.basepath)\n    else:\n        from_path = self._resolve_path(from_path)\n\n    if local_path is None:\n        local_path = Path(\".\").absolute()\n\n    # validate that from_path has a trailing slash for proper fsspec behavior across versions\n    if not from_path.endswith(\"/\"):\n        from_path += \"/\"\n\n    return self.filesystem.get(from_path, local_path, recursive=True)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.RemoteFileSystem.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory.

    Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n    overwrite: bool = True,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    if to_path is None:\n        to_path = str(self.basepath)\n    else:\n        to_path = self._resolve_path(to_path)\n\n    if local_path is None:\n        local_path = \".\"\n\n    included_files = None\n    if ignore_file:\n        with open(ignore_file, \"r\") as f:\n            ignore_patterns = f.readlines()\n\n        included_files = filter_files(\n            local_path, ignore_patterns, include_dirs=True\n        )\n\n    counter = 0\n    for f in Path(local_path).rglob(\"*\"):\n        relative_path = f.relative_to(local_path)\n        if included_files and str(relative_path) not in included_files:\n            continue\n\n        if to_path.endswith(\"/\"):\n            fpath = to_path + relative_path.as_posix()\n        else:\n            fpath = to_path + \"/\" + relative_path.as_posix()\n\n        if f.is_dir():\n            pass\n        else:\n            f = f.as_posix()\n            if overwrite:\n                self.filesystem.put_file(f, fpath, overwrite=True)\n            else:\n                self.filesystem.put_file(f, fpath)\n\n            counter += 1\n\n    return counter\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.S3","title":"S3","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on AWS S3.

    Example

    Load stored S3 config:

    from prefect.filesystems import S3\n\ns3_block = S3.load(\"BLOCK_NAME\")\n

    Source code in prefect/filesystems.py
    class S3(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on AWS S3.\n\n    Example:\n        Load stored S3 config:\n        ```python\n        from prefect.filesystems import S3\n\n        s3_block = S3.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"S3\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/d74b16fe84ce626345adf235a47008fea2869a60-225x225.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#s3\"\n\n    bucket_path: str = Field(\n        default=...,\n        description=\"An S3 bucket path.\",\n        example=\"my-bucket/a-directory-within\",\n    )\n    aws_access_key_id: Optional[SecretStr] = Field(\n        default=None,\n        title=\"AWS Access Key ID\",\n        description=\"Equivalent to the AWS_ACCESS_KEY_ID environment variable.\",\n        example=\"AKIAIOSFODNN7EXAMPLE\",\n    )\n    aws_secret_access_key: Optional[SecretStr] = Field(\n        default=None,\n        title=\"AWS Secret Access Key\",\n        description=\"Equivalent to the AWS_SECRET_ACCESS_KEY environment variable.\",\n        example=\"wJalrXUtnFEMI/K7MDENG/bPxRfiCYEXAMPLEKEY\",\n    )\n\n    _remote_file_system: RemoteFileSystem = None\n\n    @property\n    def basepath(self) -> str:\n        return f\"s3://{self.bucket_path}\"\n\n    @property\n    def filesystem(self) -> RemoteFileSystem:\n        settings = {}\n        if self.aws_access_key_id:\n            settings[\"key\"] = self.aws_access_key_id.get_secret_value()\n        if self.aws_secret_access_key:\n            settings[\"secret\"] = self.aws_secret_access_key.get_secret_value()\n        self._remote_file_system = RemoteFileSystem(\n            basepath=f\"s3://{self.bucket_path}\", settings=settings\n        )\n        return self._remote_file_system\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> bytes:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        return await self.filesystem.get_directory(\n            from_path=from_path, local_path=local_path\n        )\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        return await self.filesystem.put_directory(\n            local_path=local_path, to_path=to_path, ignore_file=ignore_file\n        )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        return await self.filesystem.read_path(path)\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        return await self.filesystem.write_path(path=path, content=content)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.S3.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory.

    Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> bytes:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    return await self.filesystem.get_directory(\n        from_path=from_path, local_path=local_path\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.S3.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory.

    Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    return await self.filesystem.put_directory(\n        local_path=local_path, to_path=to_path, ignore_file=ignore_file\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.SMB","title":"SMB","text":"

    Bases: WritableFileSystem, WritableDeploymentStorage

    Store data as a file on a SMB share.

    Example

    Load stored SMB config:

    from prefect.filesystems import SMB\nsmb_block = SMB.load(\"BLOCK_NAME\")\n
    Source code in prefect/filesystems.py
    class SMB(WritableFileSystem, WritableDeploymentStorage):\n    \"\"\"\n    Store data as a file on a SMB share.\n\n    Example:\n        Load stored SMB config:\n\n        ```python\n        from prefect.filesystems import SMB\n        smb_block = SMB.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"SMB\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/3f624663f7beb97d011d011bffd51ecf6c499efc-195x195.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/filesystems/#smb\"\n\n    share_path: str = Field(\n        default=...,\n        description=\"SMB target (requires <SHARE>, followed by <PATH>).\",\n        example=\"/SHARE/dir/subdir\",\n    )\n    smb_username: Optional[SecretStr] = Field(\n        default=None,\n        title=\"SMB Username\",\n        description=\"Username with access to the target SMB SHARE.\",\n    )\n    smb_password: Optional[SecretStr] = Field(\n        default=None, title=\"SMB Password\", description=\"Password for SMB access.\"\n    )\n    smb_host: str = Field(\n        default=..., tile=\"SMB server/hostname\", description=\"SMB server/hostname.\"\n    )\n    smb_port: Optional[int] = Field(\n        default=None, title=\"SMB port\", description=\"SMB port (default: 445).\"\n    )\n\n    _remote_file_system: RemoteFileSystem = None\n\n    @property\n    def basepath(self) -> str:\n        return f\"smb://{self.smb_host.rstrip('/')}/{self.share_path.lstrip('/')}\"\n\n    @property\n    def filesystem(self) -> RemoteFileSystem:\n        settings = {}\n        if self.smb_username:\n            settings[\"username\"] = self.smb_username.get_secret_value()\n        if self.smb_password:\n            settings[\"password\"] = self.smb_password.get_secret_value()\n        if self.smb_host:\n            settings[\"host\"] = self.smb_host\n        if self.smb_port:\n            settings[\"port\"] = self.smb_port\n        self._remote_file_system = RemoteFileSystem(\n            basepath=f\"smb://{self.smb_host.rstrip('/')}/{self.share_path.lstrip('/')}\",\n            settings=settings,\n        )\n        return self._remote_file_system\n\n    @sync_compatible\n    async def get_directory(\n        self, from_path: Optional[str] = None, local_path: Optional[str] = None\n    ) -> bytes:\n        \"\"\"\n        Downloads a directory from a given remote path to a local directory.\n        Defaults to downloading the entire contents of the block's basepath to the current working directory.\n        \"\"\"\n        return await self.filesystem.get_directory(\n            from_path=from_path, local_path=local_path\n        )\n\n    @sync_compatible\n    async def put_directory(\n        self,\n        local_path: Optional[str] = None,\n        to_path: Optional[str] = None,\n        ignore_file: Optional[str] = None,\n    ) -> int:\n        \"\"\"\n        Uploads a directory from a given local path to a remote directory.\n        Defaults to uploading the entire contents of the current working directory to the block's basepath.\n        \"\"\"\n        return await self.filesystem.put_directory(\n            local_path=local_path,\n            to_path=to_path,\n            ignore_file=ignore_file,\n            overwrite=False,\n        )\n\n    @sync_compatible\n    async def read_path(self, path: str) -> bytes:\n        return await self.filesystem.read_path(path)\n\n    @sync_compatible\n    async def write_path(self, path: str, content: bytes) -> str:\n        return await self.filesystem.write_path(path=path, content=content)\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.SMB.get_directory","title":"get_directory async","text":"

    Downloads a directory from a given remote path to a local directory. Defaults to downloading the entire contents of the block's basepath to the current working directory.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def get_directory(\n    self, from_path: Optional[str] = None, local_path: Optional[str] = None\n) -> bytes:\n    \"\"\"\n    Downloads a directory from a given remote path to a local directory.\n    Defaults to downloading the entire contents of the block's basepath to the current working directory.\n    \"\"\"\n    return await self.filesystem.get_directory(\n        from_path=from_path, local_path=local_path\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/filesystems/#prefect.filesystems.SMB.put_directory","title":"put_directory async","text":"

    Uploads a directory from a given local path to a remote directory. Defaults to uploading the entire contents of the current working directory to the block's basepath.

    Source code in prefect/filesystems.py
    @sync_compatible\nasync def put_directory(\n    self,\n    local_path: Optional[str] = None,\n    to_path: Optional[str] = None,\n    ignore_file: Optional[str] = None,\n) -> int:\n    \"\"\"\n    Uploads a directory from a given local path to a remote directory.\n    Defaults to uploading the entire contents of the current working directory to the block's basepath.\n    \"\"\"\n    return await self.filesystem.put_directory(\n        local_path=local_path,\n        to_path=to_path,\n        ignore_file=ignore_file,\n        overwrite=False,\n    )\n
    ","tags":["Python API","filesystems","LocalFileSystem","RemoteFileSystem"]},{"location":"api-ref/prefect/flows/","title":"prefect.flows","text":"","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows","title":"prefect.flows","text":"

    Module containing the base workflow class and decorator - for most use cases, using the @flow decorator is preferred.

    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow","title":"Flow","text":"

    Bases: Generic[P, R]

    A Prefect workflow definition.

    Note

    We recommend using the @flow decorator for most use-cases.

    Wraps a function with an entrypoint to the Prefect engine. To preserve the input and output types, we use the generic type variables P and R for \"Parameters\" and \"Returns\" respectively.

    Parameters:

    Name Type Description Default fn Callable[P, R]

    The function defining the workflow.

    required name Optional[str]

    An optional name for the flow; if not provided, the name will be inferred from the given function.

    None version Optional[str]

    An optional version string for the flow; if not provided, we will attempt to create a version string as a hash of the file containing the wrapped function; if the file cannot be located, the version will be null.

    None flow_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this flow; this name can be provided as a string template with the flow's parameters as variables, or a function that returns a string.

    None task_runner Union[Type[BaseTaskRunner], BaseTaskRunner]

    An optional task runner to use for task execution within the flow; if not provided, a ConcurrentTaskRunner will be used.

    ConcurrentTaskRunner description str

    An optional string description for the flow; if not provided, the description will be pulled from the docstring for the decorated function.

    None timeout_seconds Union[int, float]

    An optional number of seconds indicating a maximum runtime for the flow. If the flow exceeds this runtime, it will be marked as failed. Flow execution may continue until the next task is called.

    None validate_parameters bool

    By default, parameters passed to flows are validated by Pydantic. This will check that input values conform to the annotated types on the function. Where possible, values will be coerced into the correct type; for example, if a parameter is defined as x: int and \"5\" is passed, it will be resolved to 5. If set to False, no validation will be performed on flow parameters.

    True retries Optional[int]

    An optional number of times to retry on flow run failure.

    None retry_delay_seconds Optional[Union[int, float]]

    An optional number of seconds to wait before retrying the flow after failure. This is only applicable if retries is nonzero.

    None persist_result Optional[bool]

    An optional toggle indicating whether the result of this flow should be persisted to result storage. Defaults to None, which indicates that Prefect should choose whether the result should be persisted depending on the features being used.

    None result_storage Optional[ResultStorage]

    An optional block to use to persist the result of this flow. This value will be used as the default for any tasks in this flow. If not provided, the local file system will be used unless called as a subflow, at which point the default will be loaded from the parent flow.

    None result_serializer Optional[ResultSerializer]

    An optional serializer to use to serialize the result of this flow for persistence. This value will be used as the default for any tasks in this flow. If not provided, the value of PREFECT_RESULTS_DEFAULT_SERIALIZER will be used unless called as a subflow, at which point the default will be loaded from the parent flow.

    None on_failure Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of callables to run when the flow enters a failed state.

    None on_completion Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of callables to run when the flow enters a completed state.

    None on_cancellation Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of callables to run when the flow enters a cancelling state.

    None on_crashed Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of callables to run when the flow enters a crashed state.

    None Source code in prefect/flows.py
    @PrefectObjectRegistry.register_instances\nclass Flow(Generic[P, R]):\n    \"\"\"\n    A Prefect workflow definition.\n\n    !!! note\n        We recommend using the [`@flow` decorator][prefect.flows.flow] for most use-cases.\n\n    Wraps a function with an entrypoint to the Prefect engine. To preserve the input\n    and output types, we use the generic type variables `P` and `R` for \"Parameters\" and\n    \"Returns\" respectively.\n\n    Args:\n        fn: The function defining the workflow.\n        name: An optional name for the flow; if not provided, the name will be inferred\n            from the given function.\n        version: An optional version string for the flow; if not provided, we will\n            attempt to create a version string as a hash of the file containing the\n            wrapped function; if the file cannot be located, the version will be null.\n        flow_run_name: An optional name to distinguish runs of this flow; this name can\n            be provided as a string template with the flow's parameters as variables,\n            or a function that returns a string.\n        task_runner: An optional task runner to use for task execution within the flow;\n            if not provided, a `ConcurrentTaskRunner` will be used.\n        description: An optional string description for the flow; if not provided, the\n            description will be pulled from the docstring for the decorated function.\n        timeout_seconds: An optional number of seconds indicating a maximum runtime for\n            the flow. If the flow exceeds this runtime, it will be marked as failed.\n            Flow execution may continue until the next task is called.\n        validate_parameters: By default, parameters passed to flows are validated by\n            Pydantic. This will check that input values conform to the annotated types\n            on the function. Where possible, values will be coerced into the correct\n            type; for example, if a parameter is defined as `x: int` and \"5\" is passed,\n            it will be resolved to `5`. If set to `False`, no validation will be\n            performed on flow parameters.\n        retries: An optional number of times to retry on flow run failure.\n        retry_delay_seconds: An optional number of seconds to wait before retrying the\n            flow after failure. This is only applicable if `retries` is nonzero.\n        persist_result: An optional toggle indicating whether the result of this flow\n            should be persisted to result storage. Defaults to `None`, which indicates\n            that Prefect should choose whether the result should be persisted depending on\n            the features being used.\n        result_storage: An optional block to use to persist the result of this flow.\n            This value will be used as the default for any tasks in this flow.\n            If not provided, the local file system will be used unless called as\n            a subflow, at which point the default will be loaded from the parent flow.\n        result_serializer: An optional serializer to use to serialize the result of this\n            flow for persistence. This value will be used as the default for any tasks\n            in this flow. If not provided, the value of `PREFECT_RESULTS_DEFAULT_SERIALIZER`\n            will be used unless called as a subflow, at which point the default will be\n            loaded from the parent flow.\n        on_failure: An optional list of callables to run when the flow enters a failed state.\n        on_completion: An optional list of callables to run when the flow enters a completed state.\n        on_cancellation: An optional list of callables to run when the flow enters a cancelling state.\n        on_crashed: An optional list of callables to run when the flow enters a crashed state.\n    \"\"\"\n\n    # NOTE: These parameters (types, defaults, and docstrings) should be duplicated\n    #       exactly in the @flow decorator\n    def __init__(\n        self,\n        fn: Callable[P, R],\n        name: Optional[str] = None,\n        version: Optional[str] = None,\n        flow_run_name: Optional[Union[Callable[[], str], str]] = None,\n        retries: Optional[int] = None,\n        retry_delay_seconds: Optional[Union[int, float]] = None,\n        task_runner: Union[Type[BaseTaskRunner], BaseTaskRunner] = ConcurrentTaskRunner,\n        description: str = None,\n        timeout_seconds: Union[int, float] = None,\n        validate_parameters: bool = True,\n        persist_result: Optional[bool] = None,\n        result_storage: Optional[ResultStorage] = None,\n        result_serializer: Optional[ResultSerializer] = None,\n        cache_result_in_memory: bool = True,\n        log_prints: Optional[bool] = None,\n        on_completion: Optional[\n            List[Callable[[FlowSchema, FlowRun, State], None]]\n        ] = None,\n        on_failure: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n        on_cancellation: Optional[\n            List[Callable[[FlowSchema, FlowRun, State], None]]\n        ] = None,\n        on_crashed: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    ):\n        if name is not None and not isinstance(name, str):\n            raise TypeError(\n                \"Expected string for flow parameter 'name'; got {} instead. {}\".format(\n                    type(name).__name__,\n                    (\n                        \"Perhaps you meant to call it? e.g.\"\n                        \" '@flow(name=get_flow_run_name())'\"\n                        if callable(name)\n                        else \"\"\n                    ),\n                )\n            )\n\n        # Validate if hook passed is list and contains callables\n        hook_categories = [on_completion, on_failure, on_cancellation, on_crashed]\n        hook_names = [\"on_completion\", \"on_failure\", \"on_cancellation\", \"on_crashed\"]\n        for hooks, hook_name in zip(hook_categories, hook_names):\n            if hooks is not None:\n                if not hooks:\n                    raise ValueError(f\"Empty list passed for '{hook_name}'\")\n                try:\n                    hooks = list(hooks)\n                except TypeError:\n                    raise TypeError(\n                        f\"Expected iterable for '{hook_name}'; got\"\n                        f\" {type(hooks).__name__} instead. Please provide a list of\"\n                        f\" hooks to '{hook_name}':\\n\\n\"\n                        f\"@flow({hook_name}=[hook1, hook2])\\ndef\"\n                        \" my_flow():\\n\\tpass\"\n                    )\n\n                for hook in hooks:\n                    if not callable(hook):\n                        raise TypeError(\n                            f\"Expected callables in '{hook_name}'; got\"\n                            f\" {type(hook).__name__} instead. Please provide a list of\"\n                            f\" hooks to '{hook_name}':\\n\\n\"\n                            f\"@flow({hook_name}=[hook1, hook2])\\ndef\"\n                            \" my_flow():\\n\\tpass\"\n                        )\n\n        if not callable(fn):\n            raise TypeError(\"'fn' must be callable\")\n\n        # Validate name if given\n        if name:\n            raise_on_name_with_banned_characters(name)\n\n        self.name = name or fn.__name__.replace(\"_\", \"-\")\n\n        if flow_run_name is not None:\n            if not isinstance(flow_run_name, str) and not callable(flow_run_name):\n                raise TypeError(\n                    \"Expected string or callable for 'flow_run_name'; got\"\n                    f\" {type(flow_run_name).__name__} instead.\"\n                )\n        self.flow_run_name = flow_run_name\n\n        task_runner = task_runner or ConcurrentTaskRunner()\n        self.task_runner = (\n            task_runner() if isinstance(task_runner, type) else task_runner\n        )\n\n        self.log_prints = log_prints\n\n        self.description = description or inspect.getdoc(fn)\n        update_wrapper(self, fn)\n        self.fn = fn\n        self.isasync = is_async_fn(self.fn)\n\n        raise_for_reserved_arguments(self.fn, [\"return_state\", \"wait_for\"])\n\n        # Version defaults to a hash of the function's file\n        flow_file = inspect.getsourcefile(self.fn)\n        if not version:\n            try:\n                version = file_hash(flow_file)\n            except (FileNotFoundError, TypeError, OSError):\n                pass  # `getsourcefile` can return null values and \"<stdin>\" for objects in repls\n        self.version = version\n\n        self.timeout_seconds = float(timeout_seconds) if timeout_seconds else None\n\n        # FlowRunPolicy settings\n        # TODO: We can instantiate a `FlowRunPolicy` and add Pydantic bound checks to\n        #       validate that the user passes positive numbers here\n        self.retries = (\n            retries if retries is not None else PREFECT_FLOW_DEFAULT_RETRIES.value()\n        )\n\n        self.retry_delay_seconds = (\n            retry_delay_seconds\n            if retry_delay_seconds is not None\n            else PREFECT_FLOW_DEFAULT_RETRY_DELAY_SECONDS.value()\n        )\n\n        self.parameters = parameter_schema(self.fn)\n        self.should_validate_parameters = validate_parameters\n\n        if self.should_validate_parameters:\n            # Try to create the validated function now so that incompatibility can be\n            # raised at declaration time rather than at runtime\n            # We cannot, however, store the validated function on the flow because it\n            # is not picklable in some environments\n            try:\n                ValidatedFunction(self.fn, config={\"arbitrary_types_allowed\": True})\n            except pydantic.ConfigError as exc:\n                raise ValueError(\n                    \"Flow function is not compatible with `validate_parameters`. \"\n                    \"Disable validation or change the argument names.\"\n                ) from exc\n\n        self.persist_result = persist_result\n        self.result_storage = result_storage\n        self.result_serializer = result_serializer\n        self.cache_result_in_memory = cache_result_in_memory\n\n        # Check for collision in the registry\n        registry = PrefectObjectRegistry.get()\n\n        if registry and any(\n            other\n            for other in registry.get_instances(Flow)\n            if other.name == self.name and id(other.fn) != id(self.fn)\n        ):\n            file = inspect.getsourcefile(self.fn)\n            line_number = inspect.getsourcelines(self.fn)[1]\n            warnings.warn(\n                f\"A flow named {self.name!r} and defined at '{file}:{line_number}' \"\n                \"conflicts with another flow. Consider specifying a unique `name` \"\n                \"parameter in the flow definition:\\n\\n \"\n                \"`@flow(name='my_unique_name', ...)`\"\n            )\n        self.on_completion = on_completion\n        self.on_failure = on_failure\n        self.on_cancellation = on_cancellation\n        self.on_crashed = on_crashed\n\n        # Used for flows loaded from remote storage\n        self._storage: Optional[RunnerStorage] = None\n        self._entrypoint: Optional[str] = None\n\n    def with_options(\n        self,\n        *,\n        name: str = None,\n        version: str = None,\n        retries: int = 0,\n        retry_delay_seconds: Union[int, float] = 0,\n        description: str = None,\n        flow_run_name: Optional[Union[Callable[[], str], str]] = None,\n        task_runner: Union[Type[BaseTaskRunner], BaseTaskRunner] = None,\n        timeout_seconds: Union[int, float] = None,\n        validate_parameters: bool = None,\n        persist_result: Optional[bool] = NotSet,\n        result_storage: Optional[ResultStorage] = NotSet,\n        result_serializer: Optional[ResultSerializer] = NotSet,\n        cache_result_in_memory: bool = None,\n        log_prints: Optional[bool] = NotSet,\n        on_completion: Optional[\n            List[Callable[[FlowSchema, FlowRun, State], None]]\n        ] = None,\n        on_failure: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n        on_cancellation: Optional[\n            List[Callable[[FlowSchema, FlowRun, State], None]]\n        ] = None,\n        on_crashed: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    ) -> Self:\n        \"\"\"\n        Create a new flow from the current object, updating provided options.\n\n        Args:\n            name: A new name for the flow.\n            version: A new version for the flow.\n            description: A new description for the flow.\n            flow_run_name: An optional name to distinguish runs of this flow; this name\n                can be provided as a string template with the flow's parameters as variables,\n                or a function that returns a string.\n            task_runner: A new task runner for the flow.\n            timeout_seconds: A new number of seconds to fail the flow after if still\n                running.\n            validate_parameters: A new value indicating if flow calls should validate\n                given parameters.\n            retries: A new number of times to retry on flow run failure.\n            retry_delay_seconds: A new number of seconds to wait before retrying the\n                flow after failure. This is only applicable if `retries` is nonzero.\n            persist_result: A new option for enabling or disabling result persistence.\n            result_storage: A new storage type to use for results.\n            result_serializer: A new serializer to use for results.\n            cache_result_in_memory: A new value indicating if the flow's result should\n                be cached in memory.\n            on_failure: A new list of callables to run when the flow enters a failed state.\n            on_completion: A new list of callables to run when the flow enters a completed state.\n            on_cancellation: A new list of callables to run when the flow enters a cancelling state.\n            on_crashed: A new list of callables to run when the flow enters a crashed state.\n\n        Returns:\n            A new `Flow` instance.\n\n        Examples:\n\n            Create a new flow from an existing flow and update the name:\n\n            >>> @flow(name=\"My flow\")\n            >>> def my_flow():\n            >>>     return 1\n            >>>\n            >>> new_flow = my_flow.with_options(name=\"My new flow\")\n\n            Create a new flow from an existing flow, update the task runner, and call\n            it without an intermediate variable:\n\n            >>> from prefect.task_runners import SequentialTaskRunner\n            >>>\n            >>> @flow\n            >>> def my_flow(x, y):\n            >>>     return x + y\n            >>>\n            >>> state = my_flow.with_options(task_runner=SequentialTaskRunner)(1, 3)\n            >>> assert state.result() == 4\n\n        \"\"\"\n        new_flow = Flow(\n            fn=self.fn,\n            name=name or self.name,\n            description=description or self.description,\n            flow_run_name=flow_run_name,\n            version=version or self.version,\n            task_runner=task_runner or self.task_runner,\n            retries=retries or self.retries,\n            retry_delay_seconds=retry_delay_seconds or self.retry_delay_seconds,\n            timeout_seconds=(\n                timeout_seconds if timeout_seconds is not None else self.timeout_seconds\n            ),\n            validate_parameters=(\n                validate_parameters\n                if validate_parameters is not None\n                else self.should_validate_parameters\n            ),\n            persist_result=(\n                persist_result if persist_result is not NotSet else self.persist_result\n            ),\n            result_storage=(\n                result_storage if result_storage is not NotSet else self.result_storage\n            ),\n            result_serializer=(\n                result_serializer\n                if result_serializer is not NotSet\n                else self.result_serializer\n            ),\n            cache_result_in_memory=(\n                cache_result_in_memory\n                if cache_result_in_memory is not None\n                else self.cache_result_in_memory\n            ),\n            log_prints=log_prints if log_prints is not NotSet else self.log_prints,\n            on_completion=on_completion or self.on_completion,\n            on_failure=on_failure or self.on_failure,\n            on_cancellation=on_cancellation or self.on_cancellation,\n            on_crashed=on_crashed or self.on_crashed,\n        )\n        new_flow._storage = self._storage\n        new_flow._entrypoint = self._entrypoint\n        return new_flow\n\n    def validate_parameters(self, parameters: Dict[str, Any]) -> Dict[str, Any]:\n        \"\"\"\n        Validate parameters for compatibility with the flow by attempting to cast the inputs to the\n        associated types specified by the function's type annotations.\n\n        Returns:\n            A new dict of parameters that have been cast to the appropriate types\n\n        Raises:\n            ParameterTypeError: if the provided parameters are not valid\n        \"\"\"\n        args, kwargs = parameters_to_args_kwargs(self.fn, parameters)\n\n        if HAS_PYDANTIC_V2:\n            has_v1_models = any(isinstance(o, V1BaseModel) for o in args) or any(\n                isinstance(o, V1BaseModel) for o in kwargs.values()\n            )\n            has_v2_models = any(isinstance(o, V2BaseModel) for o in args) or any(\n                isinstance(o, V2BaseModel) for o in kwargs.values()\n            )\n\n            if has_v1_models and has_v2_models:\n                raise ParameterTypeError(\n                    \"Cannot mix Pydantic v1 and v2 models as arguments to a flow.\"\n                )\n\n            if has_v1_models:\n                validated_fn = V1ValidatedFunction(\n                    self.fn, config={\"arbitrary_types_allowed\": True}\n                )\n            else:\n                validated_fn = V2ValidatedFunction(\n                    self.fn, config={\"arbitrary_types_allowed\": True}\n                )\n\n        else:\n            validated_fn = ValidatedFunction(\n                self.fn, config={\"arbitrary_types_allowed\": True}\n            )\n\n        try:\n            model = validated_fn.init_model_instance(*args, **kwargs)\n        except pydantic.ValidationError as exc:\n            # We capture the pydantic exception and raise our own because the pydantic\n            # exception is not picklable when using a cythonized pydantic installation\n            raise ParameterTypeError.from_validation_error(exc) from None\n        except V2ValidationError as exc:\n            # We capture the pydantic exception and raise our own because the pydantic\n            # exception is not picklable when using a cythonized pydantic installation\n            raise ParameterTypeError.from_validation_error(exc) from None\n\n        # Get the updated parameter dict with cast values from the model\n        cast_parameters = {\n            k: v\n            for k, v in model._iter()\n            if k in model.__fields_set__ or model.__fields__[k].default_factory\n        }\n        return cast_parameters\n\n    def serialize_parameters(self, parameters: Dict[str, Any]) -> Dict[str, Any]:\n        \"\"\"\n        Convert parameters to a serializable form.\n\n        Uses FastAPI's `jsonable_encoder` to convert to JSON compatible objects without\n        converting everything directly to a string. This maintains basic types like\n        integers during API roundtrips.\n        \"\"\"\n        serialized_parameters = {}\n        for key, value in parameters.items():\n            try:\n                serialized_parameters[key] = jsonable_encoder(value)\n            except (TypeError, ValueError):\n                logger.debug(\n                    f\"Parameter {key!r} for flow {self.name!r} is of unserializable \"\n                    f\"type {type(value).__name__!r} and will not be stored \"\n                    \"in the backend.\"\n                )\n                serialized_parameters[key] = f\"<{type(value).__name__}>\"\n        return serialized_parameters\n\n    @sync_compatible\n    async def to_deployment(\n        self,\n        name: str,\n        interval: Optional[Union[int, float, datetime.timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        work_pool_name: Optional[str] = None,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[Dict[str, Any]] = None,\n    ):\n        \"\"\"\n        Creates a runner deployment object for this flow.\n\n        Args:\n            name: The name to give the created deployment.\n            interval: An interval on which to execute the new deployment. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this deployment.\n            rrule: An rrule schedule of when to execute runs of this deployment.\n            timezone: A timezone to use for the schedule. Defaults to UTC.\n            triggers: A list of triggers that will kick off runs of this deployment.\n            schedule: A schedule object defining when to execute runs of this deployment.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            parameters: A dictionary of default parameter values to pass to runs of this deployment.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for the created deployment.\n            work_pool_name: The name of the work pool to use for this deployment.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n\n        Examples:\n            Prepare two deployments and serve them:\n\n            ```python\n            from prefect import flow, serve\n\n            @flow\n            def my_flow(name):\n                print(f\"hello {name}\")\n\n            @flow\n            def my_other_flow(name):\n                print(f\"goodbye {name}\")\n\n            if __name__ == \"__main__\":\n                hello_deploy = my_flow.to_deployment(\"hello\", tags=[\"dev\"])\n                bye_deploy = my_other_flow.to_deployment(\"goodbye\", tags=[\"dev\"])\n                serve(hello_deploy, bye_deploy)\n            ```\n        \"\"\"\n        from prefect.deployments.runner import RunnerDeployment\n\n        if not name.endswith(\".py\"):\n            raise_on_name_with_banned_characters(name)\n        if self._storage and self._entrypoint:\n            return await RunnerDeployment.from_storage(\n                storage=self._storage,\n                entrypoint=self._entrypoint,\n                name=name,\n                interval=interval,\n                cron=cron,\n                rrule=rrule,\n                schedule=schedule,\n                is_schedule_active=is_schedule_active,\n                tags=tags,\n                triggers=triggers,\n                parameters=parameters or {},\n                description=description,\n                version=version,\n                enforce_parameter_schema=enforce_parameter_schema,\n                work_pool_name=work_pool_name,\n                work_queue_name=work_queue_name,\n                job_variables=job_variables,\n            )\n        else:\n            return RunnerDeployment.from_flow(\n                self,\n                name=name,\n                interval=interval,\n                cron=cron,\n                rrule=rrule,\n                schedule=schedule,\n                is_schedule_active=is_schedule_active,\n                tags=tags,\n                triggers=triggers,\n                parameters=parameters or {},\n                description=description,\n                version=version,\n                enforce_parameter_schema=enforce_parameter_schema,\n                work_pool_name=work_pool_name,\n                work_queue_name=work_queue_name,\n                job_variables=job_variables,\n            )\n\n    @sync_compatible\n    async def serve(\n        self,\n        name: str,\n        interval: Optional[Union[int, float, datetime.timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        parameters: Optional[dict] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        pause_on_shutdown: bool = True,\n        print_starting_message: bool = True,\n        webserver: bool = False,\n    ):\n        \"\"\"\n        Creates a deployment for this flow and starts a runner to monitor for scheduled work.\n\n        Args:\n            name: The name to give the created deployment.\n            interval: An interval on which to execute the new deployment. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this deployment.\n            rrule: An rrule schedule of when to execute runs of this deployment.\n            triggers: A list of triggers that will kick off runs of this deployment.\n            schedule: A schedule object defining when to execute runs of this deployment. Used to\n                define additional scheduling options like `timezone`.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            parameters: A dictionary of default parameter values to pass to runs of this deployment.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for the created deployment.\n            pause_on_shutdown: If True, provided schedule will be paused when the serve function is stopped.\n                If False, the schedules will continue running.\n            print_starting_message: Whether or not to print the starting message when flow is served.\n            webserver: Whether or not to start a monitoring webserver for this flow.\n\n        Examples:\n            Serve a flow:\n\n            ```python\n            from prefect import flow\n\n            @flow\n            def my_flow(name):\n                print(f\"hello {name}\")\n\n            if __name__ == \"__main__\":\n                my_flow.serve(\"example-deployment\")\n            ```\n\n            Serve a flow and run it every hour:\n\n            ```python\n            from prefect import flow\n\n            @flow\n            def my_flow(name):\n                print(f\"hello {name}\")\n\n            if __name__ == \"__main__\":\n                my_flow.serve(\"example-deployment\", interval=3600)\n            ```\n        \"\"\"\n        from prefect.runner import Runner\n\n        # Handling for my_flow.serve(__file__)\n        # Will set name to name of file where my_flow.serve() without the extension\n        # Non filepath strings will pass through unchanged\n        name = Path(name).stem\n\n        runner = Runner(name=name, pause_on_shutdown=pause_on_shutdown)\n        deployment_id = await runner.add_flow(\n            self,\n            name=name,\n            triggers=triggers,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            parameters=parameters,\n            description=description,\n            tags=tags,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n        )\n        if print_starting_message:\n            help_message = (\n                f\"[green]Your flow {self.name!r} is being served and polling for\"\n                \" scheduled runs!\\n[/]\\nTo trigger a run for this flow, use the\"\n                \" following command:\\n[blue]\\n\\t$ prefect deployment run\"\n                f\" '{self.name}/{name}'\\n[/]\"\n            )\n            if PREFECT_UI_URL:\n                help_message += (\n                    \"\\nYou can also run your flow via the Prefect UI:\"\n                    f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_id}[/]\\n\"\n                )\n\n            console = Console()\n            console.print(Panel(help_message))\n        await runner.start(webserver=webserver)\n\n    @classmethod\n    @sync_compatible\n    async def from_source(\n        cls: Type[F],\n        source: Union[str, RunnerStorage, ReadableDeploymentStorage],\n        entrypoint: str,\n    ) -> F:\n        \"\"\"\n        Loads a flow from a remote s ource.\n\n        Args:\n            source: Either a URL to a git repository or a storage object.\n            entrypoint:  The path to a file containing a flow and the name of the flow function in\n                the format `./path/to/file.py:flow_func_name`.\n\n        Returns:\n            A new `Flow` instance.\n\n        Examples:\n            Load a flow from a public git repository:\n\n\n            ```python\n            from prefect import flow\n            from prefect.runner.storage import GitRepository\n            from prefect.blocks.system import Secret\n\n            my_flow = flow.from_source(\n                source=\"https://github.com/org/repo.git\",\n                entrypoint=\"flows.py:my_flow\",\n            )\n\n            my_flow()\n            ```\n\n            Load a flow from a private git repository:\n\n            ```python\n            from prefect import flow\n            from prefect.runner.storage import GitRepository\n            from prefect.blocks.system import Secret\n\n            my_flow = flow.from_source(\n                source=GitRepository(\n                    url=\"https://github.com/org/repo.git\",\n                    access_token=Secret.load(\"github-access-token\").get(),\n                ),\n                entrypoint=\"flows.py:my_flow\",\n            )\n\n            my_flow()\n            ```\n        \"\"\"\n        if isinstance(source, str):\n            storage = create_storage_from_url(source)\n        elif isinstance(source, RunnerStorage):\n            storage = source\n        elif hasattr(source, \"get_directory\"):\n            storage = BlockStorageAdapter(source)\n        else:\n            raise TypeError(\n                f\"Unsupported source type {type(source).__name__!r}. Please provide a\"\n                \" URL to remote storage or a storage object.\"\n            )\n        with tempfile.TemporaryDirectory() as tmpdir:\n            storage.set_base_path(Path(tmpdir))\n            await storage.pull_code()\n\n            full_entrypoint = str(storage.destination / entrypoint)\n            flow: \"Flow\" = await from_async.wait_for_call_in_new_thread(\n                create_call(load_flow_from_entrypoint, full_entrypoint)\n            )\n            flow._storage = storage\n            flow._entrypoint = entrypoint\n\n        return flow\n\n    @sync_compatible\n    async def deploy(\n        self,\n        name: str,\n        work_pool_name: Optional[str] = None,\n        image: Optional[Union[str, DeploymentImage]] = None,\n        build: bool = True,\n        push: bool = True,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[dict] = None,\n        interval: Optional[Union[int, float, datetime.timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        parameters: Optional[dict] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        print_next_steps: bool = True,\n    ) -> UUID:\n        \"\"\"\n        Deploys a flow to run on dynamic infrastructure via a work pool.\n\n        By default, calling this method will build a Docker image for the flow, push it to a registry,\n        and create a deployment via the Prefect API that will run the flow on the given schedule.\n\n        If you want to use an existing image, you can pass `build=False` to skip building and pushing\n        an image.\n\n        Args:\n            name: The name to give the created deployment.\n            work_pool_name: The name of the work pool to use for this deployment. Defaults to\n                the value of `PREFECT_DEFAULT_WORK_POOL_NAME`.\n            image: The name of the Docker image to build, including the registry and\n                repository. Pass a DeploymentImage instance to customize the Dockerfile used\n                and build arguments.\n            build: Whether or not to build a new image for the flow. If False, the provided\n                image will be used as-is and pulled at runtime.\n            push: Whether or not to skip pushing the built image to a registry.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n                available settings.\n            interval: An interval on which to execute the new deployment. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this deployment.\n            rrule: An rrule schedule of when to execute runs of this deployment.\n            triggers: A list of triggers that will kick off runs of this deployment.\n            schedule: A schedule object defining when to execute runs of this deployment. Used to\n                define additional scheduling options like `timezone`.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            parameters: A dictionary of default parameter values to pass to runs of this deployment.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for the created deployment.\n            print_next_steps_message: Whether or not to print a message with next steps\n            after deploying the deployments.\n\n        Returns:\n            The ID of the created/updated deployment.\n\n        Examples:\n            Deploy a local flow to a work pool:\n\n            ```python\n            from prefect import flow\n\n            @flow\n            def my_flow(name):\n                print(f\"hello {name}\")\n\n            if __name__ == \"__main__\":\n                my_flow.deploy(\n                    \"example-deployment\",\n                    work_pool_name=\"my-work-pool\",\n                    image=\"my-repository/my-image:dev\",\n                )\n            ```\n\n            Deploy a remotely stored flow to a work pool:\n\n            ```python\n            from prefect import flow\n\n            if __name__ == \"__main__\":\n                flow.from_source(\n                    source=\"https://github.com/org/repo.git\",\n                    entrypoint=\"flows.py:my_flow\",\n                ).deploy(\n                    \"example-deployment\",\n                    work_pool_name=\"my-work-pool\",\n                    image=\"my-repository/my-image:dev\",\n                )\n            ```\n        \"\"\"\n        work_pool_name = work_pool_name or PREFECT_DEFAULT_WORK_POOL_NAME.value()\n\n        try:\n            async with get_client() as client:\n                work_pool = await client.read_work_pool(work_pool_name)\n        except ObjectNotFound as exc:\n            raise ValueError(\n                f\"Could not find work pool {work_pool_name!r}. Please create it before\"\n                \" deploying this flow.\"\n            ) from exc\n\n        deployment = await self.to_deployment(\n            name=name,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            triggers=triggers,\n            parameters=parameters,\n            description=description,\n            tags=tags,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n\n        deployment_ids = await deploy(\n            deployment,\n            work_pool_name=work_pool_name,\n            image=image,\n            build=build,\n            push=push,\n            print_next_steps_message=False,\n        )\n\n        if print_next_steps:\n            console = Console()\n            if not work_pool.is_push_pool and not work_pool.is_managed_pool:\n                console.print(\n                    \"\\nTo execute flow runs from this deployment, start a worker in a\"\n                    \" separate terminal that pulls work from the\"\n                    f\" {work_pool_name!r} work pool:\"\n                )\n                console.print(\n                    f\"\\n\\t$ prefect worker start --pool {work_pool_name!r}\",\n                    style=\"blue\",\n                )\n            console.print(\n                \"\\nTo schedule a run for this deployment, use the following command:\"\n            )\n            console.print(\n                f\"\\n\\t$ prefect deployment run '{self.name}/{name}'\\n\",\n                style=\"blue\",\n            )\n            if PREFECT_UI_URL:\n                console.print(\n                    \"\\nYou can also run your flow via the Prefect UI:\"\n                    f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_ids[0]}[/]\\n\"\n                )\n\n        return deployment_ids[0]\n\n    @overload\n    def __call__(self: \"Flow[P, NoReturn]\", *args: P.args, **kwargs: P.kwargs) -> None:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def __call__(\n        self: \"Flow[P, Coroutine[Any, Any, T]]\", *args: P.args, **kwargs: P.kwargs\n    ) -> Awaitable[T]:\n        ...\n\n    @overload\n    def __call__(\n        self: \"Flow[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> T:\n        ...\n\n    @overload\n    def __call__(\n        self: \"Flow[P, T]\",\n        *args: P.args,\n        return_state: Literal[True],\n        **kwargs: P.kwargs,\n    ) -> State[T]:\n        ...\n\n    def __call__(\n        self,\n        *args: \"P.args\",\n        return_state: bool = False,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: \"P.kwargs\",\n    ):\n        \"\"\"\n        Run the flow and return its result.\n\n\n        Flow parameter values must be serializable by Pydantic.\n\n        If writing an async flow, this call must be awaited.\n\n        This will create a new flow run in the API.\n\n        Args:\n            *args: Arguments to run the flow with.\n            return_state: Return a Prefect State containing the result of the\n                flow run.\n            wait_for: Upstream task futures to wait for before starting the flow if called as a subflow\n            **kwargs: Keyword arguments to run the flow with.\n\n        Returns:\n            If `return_state` is False, returns the result of the flow run.\n            If `return_state` is True, returns the result of the flow run\n                wrapped in a Prefect State which provides error handling.\n\n        Examples:\n\n            Define a flow\n\n            >>> @flow\n            >>> def my_flow(name):\n            >>>     print(f\"hello {name}\")\n            >>>     return f\"goodbye {name}\"\n\n            Run a flow\n\n            >>> my_flow(\"marvin\")\n            hello marvin\n            \"goodbye marvin\"\n\n            Run a flow with additional tags\n\n            >>> from prefect import tags\n            >>> with tags(\"db\", \"blue\"):\n            >>>     my_flow(\"foo\")\n        \"\"\"\n        from prefect.engine import enter_flow_run_engine_from_flow_call\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n\n        return_type = \"state\" if return_state else \"result\"\n\n        task_viz_tracker = get_task_viz_tracker()\n        if task_viz_tracker:\n            # this is a subflow, for now return a single task and do not go further\n            # we can add support for exploring subflows for tasks in the future.\n            return track_viz_task(self.isasync, self.name, parameters)\n\n        return enter_flow_run_engine_from_flow_call(\n            self,\n            parameters,\n            wait_for=wait_for,\n            return_type=return_type,\n        )\n\n    @overload\n    def _run(self: \"Flow[P, NoReturn]\", *args: P.args, **kwargs: P.kwargs) -> State[T]:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def _run(\n        self: \"Flow[P, Coroutine[Any, Any, T]]\", *args: P.args, **kwargs: P.kwargs\n    ) -> Awaitable[T]:\n        ...\n\n    @overload\n    def _run(self: \"Flow[P, T]\", *args: P.args, **kwargs: P.kwargs) -> State[T]:\n        ...\n\n    def _run(\n        self,\n        *args: \"P.args\",\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: \"P.kwargs\",\n    ):\n        \"\"\"\n        Run the flow and return its final state.\n\n        Examples:\n\n            Run a flow and get the returned result\n\n            >>> state = my_flow._run(\"marvin\")\n            >>> state.result()\n           \"goodbye marvin\"\n        \"\"\"\n        from prefect.engine import enter_flow_run_engine_from_flow_call\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n\n        return enter_flow_run_engine_from_flow_call(\n            self,\n            parameters,\n            wait_for=wait_for,\n            return_type=\"state\",\n        )\n\n    @sync_compatible\n    async def visualize(self, *args, **kwargs):\n        \"\"\"\n        Generates a graphviz object representing the current flow. In IPython notebooks,\n        it's rendered inline, otherwise in a new window as a PNG.\n\n        Raises:\n            - ImportError: If `graphviz` isn't installed.\n            - GraphvizExecutableNotFoundError: If the `dot` executable isn't found.\n            - FlowVisualizationError: If the flow can't be visualized for any other reason.\n        \"\"\"\n        if not PREFECT_UNIT_TEST_MODE:\n            warnings.warn(\n                \"`flow.visualize()` will execute code inside of your flow that is not\"\n                \" decorated with `@task` or `@flow`.\"\n            )\n\n        try:\n            with TaskVizTracker() as tracker:\n                if self.isasync:\n                    await self.fn(*args, **kwargs)\n                else:\n                    self.fn(*args, **kwargs)\n\n                graph = build_task_dependencies(tracker)\n\n                visualize_task_dependencies(graph, self.name)\n\n        except GraphvizImportError:\n            raise\n        except GraphvizExecutableNotFoundError:\n            raise\n        except VisualizationUnsupportedError:\n            raise\n        except FlowVisualizationError:\n            raise\n        except Exception as e:\n            msg = (\n                \"It's possible you are trying to visualize a flow that contains \"\n                \"code that directly interacts with the result of a task\"\n                \" inside of the flow. \\nTry passing a `viz_return_value` \"\n                \"to the task decorator, e.g. `@task(viz_return_value=[1, 2, 3]).`\"\n            )\n\n            new_exception = type(e)(str(e) + \"\\n\" + msg)\n            # Copy traceback information from the original exception\n            new_exception.__traceback__ = e.__traceback__\n            raise new_exception\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.deploy","title":"deploy async","text":"

    Deploys a flow to run on dynamic infrastructure via a work pool.

    By default, calling this method will build a Docker image for the flow, push it to a registry, and create a deployment via the Prefect API that will run the flow on the given schedule.

    If you want to use an existing image, you can pass build=False to skip building and pushing an image.

    Parameters:

    Name Type Description Default name str

    The name to give the created deployment.

    required work_pool_name Optional[str]

    The name of the work pool to use for this deployment. Defaults to the value of PREFECT_DEFAULT_WORK_POOL_NAME.

    None image Optional[Union[str, DeploymentImage]]

    The name of the Docker image to build, including the registry and repository. Pass a DeploymentImage instance to customize the Dockerfile used and build arguments.

    None build bool

    Whether or not to build a new image for the flow. If False, the provided image will be used as-is and pulled at runtime.

    True push bool

    Whether or not to skip pushing the built image to a registry.

    True work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[dict]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    None interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the new deployment. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this deployment.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this deployment.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that will kick off runs of this deployment.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object defining when to execute runs of this deployment. Used to define additional scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this deployment.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for the created deployment.

    False print_next_steps_message

    Whether or not to print a message with next steps

    required

    Returns:

    Type Description UUID

    The ID of the created/updated deployment.

    Examples:

    Deploy a local flow to a work pool:

    from prefect import flow\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\nif __name__ == \"__main__\":\n    my_flow.deploy(\n        \"example-deployment\",\n        work_pool_name=\"my-work-pool\",\n        image=\"my-repository/my-image:dev\",\n    )\n

    Deploy a remotely stored flow to a work pool:

    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=\"https://github.com/org/repo.git\",\n        entrypoint=\"flows.py:my_flow\",\n    ).deploy(\n        \"example-deployment\",\n        work_pool_name=\"my-work-pool\",\n        image=\"my-repository/my-image:dev\",\n    )\n
    Source code in prefect/flows.py
    @sync_compatible\nasync def deploy(\n    self,\n    name: str,\n    work_pool_name: Optional[str] = None,\n    image: Optional[Union[str, DeploymentImage]] = None,\n    build: bool = True,\n    push: bool = True,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[dict] = None,\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    parameters: Optional[dict] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    print_next_steps: bool = True,\n) -> UUID:\n    \"\"\"\n    Deploys a flow to run on dynamic infrastructure via a work pool.\n\n    By default, calling this method will build a Docker image for the flow, push it to a registry,\n    and create a deployment via the Prefect API that will run the flow on the given schedule.\n\n    If you want to use an existing image, you can pass `build=False` to skip building and pushing\n    an image.\n\n    Args:\n        name: The name to give the created deployment.\n        work_pool_name: The name of the work pool to use for this deployment. Defaults to\n            the value of `PREFECT_DEFAULT_WORK_POOL_NAME`.\n        image: The name of the Docker image to build, including the registry and\n            repository. Pass a DeploymentImage instance to customize the Dockerfile used\n            and build arguments.\n        build: Whether or not to build a new image for the flow. If False, the provided\n            image will be used as-is and pulled at runtime.\n        push: Whether or not to skip pushing the built image to a registry.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n        interval: An interval on which to execute the new deployment. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this deployment.\n        rrule: An rrule schedule of when to execute runs of this deployment.\n        triggers: A list of triggers that will kick off runs of this deployment.\n        schedule: A schedule object defining when to execute runs of this deployment. Used to\n            define additional scheduling options like `timezone`.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        parameters: A dictionary of default parameter values to pass to runs of this deployment.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for the created deployment.\n        print_next_steps_message: Whether or not to print a message with next steps\n        after deploying the deployments.\n\n    Returns:\n        The ID of the created/updated deployment.\n\n    Examples:\n        Deploy a local flow to a work pool:\n\n        ```python\n        from prefect import flow\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        if __name__ == \"__main__\":\n            my_flow.deploy(\n                \"example-deployment\",\n                work_pool_name=\"my-work-pool\",\n                image=\"my-repository/my-image:dev\",\n            )\n        ```\n\n        Deploy a remotely stored flow to a work pool:\n\n        ```python\n        from prefect import flow\n\n        if __name__ == \"__main__\":\n            flow.from_source(\n                source=\"https://github.com/org/repo.git\",\n                entrypoint=\"flows.py:my_flow\",\n            ).deploy(\n                \"example-deployment\",\n                work_pool_name=\"my-work-pool\",\n                image=\"my-repository/my-image:dev\",\n            )\n        ```\n    \"\"\"\n    work_pool_name = work_pool_name or PREFECT_DEFAULT_WORK_POOL_NAME.value()\n\n    try:\n        async with get_client() as client:\n            work_pool = await client.read_work_pool(work_pool_name)\n    except ObjectNotFound as exc:\n        raise ValueError(\n            f\"Could not find work pool {work_pool_name!r}. Please create it before\"\n            \" deploying this flow.\"\n        ) from exc\n\n    deployment = await self.to_deployment(\n        name=name,\n        interval=interval,\n        cron=cron,\n        rrule=rrule,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        triggers=triggers,\n        parameters=parameters,\n        description=description,\n        tags=tags,\n        version=version,\n        enforce_parameter_schema=enforce_parameter_schema,\n        work_queue_name=work_queue_name,\n        job_variables=job_variables,\n    )\n\n    deployment_ids = await deploy(\n        deployment,\n        work_pool_name=work_pool_name,\n        image=image,\n        build=build,\n        push=push,\n        print_next_steps_message=False,\n    )\n\n    if print_next_steps:\n        console = Console()\n        if not work_pool.is_push_pool and not work_pool.is_managed_pool:\n            console.print(\n                \"\\nTo execute flow runs from this deployment, start a worker in a\"\n                \" separate terminal that pulls work from the\"\n                f\" {work_pool_name!r} work pool:\"\n            )\n            console.print(\n                f\"\\n\\t$ prefect worker start --pool {work_pool_name!r}\",\n                style=\"blue\",\n            )\n        console.print(\n            \"\\nTo schedule a run for this deployment, use the following command:\"\n        )\n        console.print(\n            f\"\\n\\t$ prefect deployment run '{self.name}/{name}'\\n\",\n            style=\"blue\",\n        )\n        if PREFECT_UI_URL:\n            console.print(\n                \"\\nYou can also run your flow via the Prefect UI:\"\n                f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_ids[0]}[/]\\n\"\n            )\n\n    return deployment_ids[0]\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.from_source","title":"from_source async classmethod","text":"

    Loads a flow from a remote s ource.

    Parameters:

    Name Type Description Default source Union[str, RunnerStorage, ReadableDeploymentStorage]

    Either a URL to a git repository or a storage object.

    required entrypoint str

    The path to a file containing a flow and the name of the flow function in the format ./path/to/file.py:flow_func_name.

    required

    Returns:

    Type Description F

    A new Flow instance.

    Examples:

    Load a flow from a public git repository:

    from prefect import flow\nfrom prefect.runner.storage import GitRepository\nfrom prefect.blocks.system import Secret\n\nmy_flow = flow.from_source(\n    source=\"https://github.com/org/repo.git\",\n    entrypoint=\"flows.py:my_flow\",\n)\n\nmy_flow()\n

    Load a flow from a private git repository:

    from prefect import flow\nfrom prefect.runner.storage import GitRepository\nfrom prefect.blocks.system import Secret\n\nmy_flow = flow.from_source(\n    source=GitRepository(\n        url=\"https://github.com/org/repo.git\",\n        access_token=Secret.load(\"github-access-token\").get(),\n    ),\n    entrypoint=\"flows.py:my_flow\",\n)\n\nmy_flow()\n
    Source code in prefect/flows.py
    @classmethod\n@sync_compatible\nasync def from_source(\n    cls: Type[F],\n    source: Union[str, RunnerStorage, ReadableDeploymentStorage],\n    entrypoint: str,\n) -> F:\n    \"\"\"\n    Loads a flow from a remote s ource.\n\n    Args:\n        source: Either a URL to a git repository or a storage object.\n        entrypoint:  The path to a file containing a flow and the name of the flow function in\n            the format `./path/to/file.py:flow_func_name`.\n\n    Returns:\n        A new `Flow` instance.\n\n    Examples:\n        Load a flow from a public git repository:\n\n\n        ```python\n        from prefect import flow\n        from prefect.runner.storage import GitRepository\n        from prefect.blocks.system import Secret\n\n        my_flow = flow.from_source(\n            source=\"https://github.com/org/repo.git\",\n            entrypoint=\"flows.py:my_flow\",\n        )\n\n        my_flow()\n        ```\n\n        Load a flow from a private git repository:\n\n        ```python\n        from prefect import flow\n        from prefect.runner.storage import GitRepository\n        from prefect.blocks.system import Secret\n\n        my_flow = flow.from_source(\n            source=GitRepository(\n                url=\"https://github.com/org/repo.git\",\n                access_token=Secret.load(\"github-access-token\").get(),\n            ),\n            entrypoint=\"flows.py:my_flow\",\n        )\n\n        my_flow()\n        ```\n    \"\"\"\n    if isinstance(source, str):\n        storage = create_storage_from_url(source)\n    elif isinstance(source, RunnerStorage):\n        storage = source\n    elif hasattr(source, \"get_directory\"):\n        storage = BlockStorageAdapter(source)\n    else:\n        raise TypeError(\n            f\"Unsupported source type {type(source).__name__!r}. Please provide a\"\n            \" URL to remote storage or a storage object.\"\n        )\n    with tempfile.TemporaryDirectory() as tmpdir:\n        storage.set_base_path(Path(tmpdir))\n        await storage.pull_code()\n\n        full_entrypoint = str(storage.destination / entrypoint)\n        flow: \"Flow\" = await from_async.wait_for_call_in_new_thread(\n            create_call(load_flow_from_entrypoint, full_entrypoint)\n        )\n        flow._storage = storage\n        flow._entrypoint = entrypoint\n\n    return flow\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.serialize_parameters","title":"serialize_parameters","text":"

    Convert parameters to a serializable form.

    Uses FastAPI's jsonable_encoder to convert to JSON compatible objects without converting everything directly to a string. This maintains basic types like integers during API roundtrips.

    Source code in prefect/flows.py
    def serialize_parameters(self, parameters: Dict[str, Any]) -> Dict[str, Any]:\n    \"\"\"\n    Convert parameters to a serializable form.\n\n    Uses FastAPI's `jsonable_encoder` to convert to JSON compatible objects without\n    converting everything directly to a string. This maintains basic types like\n    integers during API roundtrips.\n    \"\"\"\n    serialized_parameters = {}\n    for key, value in parameters.items():\n        try:\n            serialized_parameters[key] = jsonable_encoder(value)\n        except (TypeError, ValueError):\n            logger.debug(\n                f\"Parameter {key!r} for flow {self.name!r} is of unserializable \"\n                f\"type {type(value).__name__!r} and will not be stored \"\n                \"in the backend.\"\n            )\n            serialized_parameters[key] = f\"<{type(value).__name__}>\"\n    return serialized_parameters\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.serve","title":"serve async","text":"

    Creates a deployment for this flow and starts a runner to monitor for scheduled work.

    Parameters:

    Name Type Description Default name str

    The name to give the created deployment.

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the new deployment. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this deployment.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this deployment.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that will kick off runs of this deployment.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object defining when to execute runs of this deployment. Used to define additional scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this deployment.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for the created deployment.

    False pause_on_shutdown bool

    If True, provided schedule will be paused when the serve function is stopped. If False, the schedules will continue running.

    True print_starting_message bool

    Whether or not to print the starting message when flow is served.

    True webserver bool

    Whether or not to start a monitoring webserver for this flow.

    False

    Examples:

    Serve a flow:

    from prefect import flow\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\nif __name__ == \"__main__\":\n    my_flow.serve(\"example-deployment\")\n

    Serve a flow and run it every hour:

    from prefect import flow\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\nif __name__ == \"__main__\":\n    my_flow.serve(\"example-deployment\", interval=3600)\n
    Source code in prefect/flows.py
    @sync_compatible\nasync def serve(\n    self,\n    name: str,\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    parameters: Optional[dict] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    pause_on_shutdown: bool = True,\n    print_starting_message: bool = True,\n    webserver: bool = False,\n):\n    \"\"\"\n    Creates a deployment for this flow and starts a runner to monitor for scheduled work.\n\n    Args:\n        name: The name to give the created deployment.\n        interval: An interval on which to execute the new deployment. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this deployment.\n        rrule: An rrule schedule of when to execute runs of this deployment.\n        triggers: A list of triggers that will kick off runs of this deployment.\n        schedule: A schedule object defining when to execute runs of this deployment. Used to\n            define additional scheduling options like `timezone`.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        parameters: A dictionary of default parameter values to pass to runs of this deployment.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for the created deployment.\n        pause_on_shutdown: If True, provided schedule will be paused when the serve function is stopped.\n            If False, the schedules will continue running.\n        print_starting_message: Whether or not to print the starting message when flow is served.\n        webserver: Whether or not to start a monitoring webserver for this flow.\n\n    Examples:\n        Serve a flow:\n\n        ```python\n        from prefect import flow\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        if __name__ == \"__main__\":\n            my_flow.serve(\"example-deployment\")\n        ```\n\n        Serve a flow and run it every hour:\n\n        ```python\n        from prefect import flow\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        if __name__ == \"__main__\":\n            my_flow.serve(\"example-deployment\", interval=3600)\n        ```\n    \"\"\"\n    from prefect.runner import Runner\n\n    # Handling for my_flow.serve(__file__)\n    # Will set name to name of file where my_flow.serve() without the extension\n    # Non filepath strings will pass through unchanged\n    name = Path(name).stem\n\n    runner = Runner(name=name, pause_on_shutdown=pause_on_shutdown)\n    deployment_id = await runner.add_flow(\n        self,\n        name=name,\n        triggers=triggers,\n        interval=interval,\n        cron=cron,\n        rrule=rrule,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        parameters=parameters,\n        description=description,\n        tags=tags,\n        version=version,\n        enforce_parameter_schema=enforce_parameter_schema,\n    )\n    if print_starting_message:\n        help_message = (\n            f\"[green]Your flow {self.name!r} is being served and polling for\"\n            \" scheduled runs!\\n[/]\\nTo trigger a run for this flow, use the\"\n            \" following command:\\n[blue]\\n\\t$ prefect deployment run\"\n            f\" '{self.name}/{name}'\\n[/]\"\n        )\n        if PREFECT_UI_URL:\n            help_message += (\n                \"\\nYou can also run your flow via the Prefect UI:\"\n                f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_id}[/]\\n\"\n            )\n\n        console = Console()\n        console.print(Panel(help_message))\n    await runner.start(webserver=webserver)\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.to_deployment","title":"to_deployment async","text":"

    Creates a runner deployment object for this flow.

    Parameters:

    Name Type Description Default name str

    The name to give the created deployment.

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the new deployment. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this deployment.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this deployment.

    None timezone

    A timezone to use for the schedule. Defaults to UTC.

    required triggers Optional[List[DeploymentTrigger]]

    A list of triggers that will kick off runs of this deployment.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object defining when to execute runs of this deployment.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this deployment.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for the created deployment.

    False work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[Dict[str, Any]]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for

    None

    Examples:

    Prepare two deployments and serve them:

    from prefect import flow, serve\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\n@flow\ndef my_other_flow(name):\n    print(f\"goodbye {name}\")\n\nif __name__ == \"__main__\":\n    hello_deploy = my_flow.to_deployment(\"hello\", tags=[\"dev\"])\n    bye_deploy = my_other_flow.to_deployment(\"goodbye\", tags=[\"dev\"])\n    serve(hello_deploy, bye_deploy)\n
    Source code in prefect/flows.py
    @sync_compatible\nasync def to_deployment(\n    self,\n    name: str,\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    work_pool_name: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[Dict[str, Any]] = None,\n):\n    \"\"\"\n    Creates a runner deployment object for this flow.\n\n    Args:\n        name: The name to give the created deployment.\n        interval: An interval on which to execute the new deployment. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this deployment.\n        rrule: An rrule schedule of when to execute runs of this deployment.\n        timezone: A timezone to use for the schedule. Defaults to UTC.\n        triggers: A list of triggers that will kick off runs of this deployment.\n        schedule: A schedule object defining when to execute runs of this deployment.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        parameters: A dictionary of default parameter values to pass to runs of this deployment.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for the created deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n\n    Examples:\n        Prepare two deployments and serve them:\n\n        ```python\n        from prefect import flow, serve\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        @flow\n        def my_other_flow(name):\n            print(f\"goodbye {name}\")\n\n        if __name__ == \"__main__\":\n            hello_deploy = my_flow.to_deployment(\"hello\", tags=[\"dev\"])\n            bye_deploy = my_other_flow.to_deployment(\"goodbye\", tags=[\"dev\"])\n            serve(hello_deploy, bye_deploy)\n        ```\n    \"\"\"\n    from prefect.deployments.runner import RunnerDeployment\n\n    if not name.endswith(\".py\"):\n        raise_on_name_with_banned_characters(name)\n    if self._storage and self._entrypoint:\n        return await RunnerDeployment.from_storage(\n            storage=self._storage,\n            entrypoint=self._entrypoint,\n            name=name,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags,\n            triggers=triggers,\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n    else:\n        return RunnerDeployment.from_flow(\n            self,\n            name=name,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags,\n            triggers=triggers,\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.validate_parameters","title":"validate_parameters","text":"

    Validate parameters for compatibility with the flow by attempting to cast the inputs to the associated types specified by the function's type annotations.

    Returns:

    Type Description Dict[str, Any]

    A new dict of parameters that have been cast to the appropriate types

    Raises:

    Type Description ParameterTypeError

    if the provided parameters are not valid

    Source code in prefect/flows.py
    def validate_parameters(self, parameters: Dict[str, Any]) -> Dict[str, Any]:\n    \"\"\"\n    Validate parameters for compatibility with the flow by attempting to cast the inputs to the\n    associated types specified by the function's type annotations.\n\n    Returns:\n        A new dict of parameters that have been cast to the appropriate types\n\n    Raises:\n        ParameterTypeError: if the provided parameters are not valid\n    \"\"\"\n    args, kwargs = parameters_to_args_kwargs(self.fn, parameters)\n\n    if HAS_PYDANTIC_V2:\n        has_v1_models = any(isinstance(o, V1BaseModel) for o in args) or any(\n            isinstance(o, V1BaseModel) for o in kwargs.values()\n        )\n        has_v2_models = any(isinstance(o, V2BaseModel) for o in args) or any(\n            isinstance(o, V2BaseModel) for o in kwargs.values()\n        )\n\n        if has_v1_models and has_v2_models:\n            raise ParameterTypeError(\n                \"Cannot mix Pydantic v1 and v2 models as arguments to a flow.\"\n            )\n\n        if has_v1_models:\n            validated_fn = V1ValidatedFunction(\n                self.fn, config={\"arbitrary_types_allowed\": True}\n            )\n        else:\n            validated_fn = V2ValidatedFunction(\n                self.fn, config={\"arbitrary_types_allowed\": True}\n            )\n\n    else:\n        validated_fn = ValidatedFunction(\n            self.fn, config={\"arbitrary_types_allowed\": True}\n        )\n\n    try:\n        model = validated_fn.init_model_instance(*args, **kwargs)\n    except pydantic.ValidationError as exc:\n        # We capture the pydantic exception and raise our own because the pydantic\n        # exception is not picklable when using a cythonized pydantic installation\n        raise ParameterTypeError.from_validation_error(exc) from None\n    except V2ValidationError as exc:\n        # We capture the pydantic exception and raise our own because the pydantic\n        # exception is not picklable when using a cythonized pydantic installation\n        raise ParameterTypeError.from_validation_error(exc) from None\n\n    # Get the updated parameter dict with cast values from the model\n    cast_parameters = {\n        k: v\n        for k, v in model._iter()\n        if k in model.__fields_set__ or model.__fields__[k].default_factory\n    }\n    return cast_parameters\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.visualize","title":"visualize async","text":"

    Generates a graphviz object representing the current flow. In IPython notebooks, it's rendered inline, otherwise in a new window as a PNG.

    Raises:

    Type Description -ImportError

    If graphviz isn't installed.

    -GraphvizExecutableNotFoundError

    If the dot executable isn't found.

    -FlowVisualizationError

    If the flow can't be visualized for any other reason.

    Source code in prefect/flows.py
    @sync_compatible\nasync def visualize(self, *args, **kwargs):\n    \"\"\"\n    Generates a graphviz object representing the current flow. In IPython notebooks,\n    it's rendered inline, otherwise in a new window as a PNG.\n\n    Raises:\n        - ImportError: If `graphviz` isn't installed.\n        - GraphvizExecutableNotFoundError: If the `dot` executable isn't found.\n        - FlowVisualizationError: If the flow can't be visualized for any other reason.\n    \"\"\"\n    if not PREFECT_UNIT_TEST_MODE:\n        warnings.warn(\n            \"`flow.visualize()` will execute code inside of your flow that is not\"\n            \" decorated with `@task` or `@flow`.\"\n        )\n\n    try:\n        with TaskVizTracker() as tracker:\n            if self.isasync:\n                await self.fn(*args, **kwargs)\n            else:\n                self.fn(*args, **kwargs)\n\n            graph = build_task_dependencies(tracker)\n\n            visualize_task_dependencies(graph, self.name)\n\n    except GraphvizImportError:\n        raise\n    except GraphvizExecutableNotFoundError:\n        raise\n    except VisualizationUnsupportedError:\n        raise\n    except FlowVisualizationError:\n        raise\n    except Exception as e:\n        msg = (\n            \"It's possible you are trying to visualize a flow that contains \"\n            \"code that directly interacts with the result of a task\"\n            \" inside of the flow. \\nTry passing a `viz_return_value` \"\n            \"to the task decorator, e.g. `@task(viz_return_value=[1, 2, 3]).`\"\n        )\n\n        new_exception = type(e)(str(e) + \"\\n\" + msg)\n        # Copy traceback information from the original exception\n        new_exception.__traceback__ = e.__traceback__\n        raise new_exception\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.Flow.with_options","title":"with_options","text":"

    Create a new flow from the current object, updating provided options.

    Parameters:

    Name Type Description Default name str

    A new name for the flow.

    None version str

    A new version for the flow.

    None description str

    A new description for the flow.

    None flow_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this flow; this name can be provided as a string template with the flow's parameters as variables, or a function that returns a string.

    None task_runner Union[Type[BaseTaskRunner], BaseTaskRunner]

    A new task runner for the flow.

    None timeout_seconds Union[int, float]

    A new number of seconds to fail the flow after if still running.

    None validate_parameters bool

    A new value indicating if flow calls should validate given parameters.

    None retries int

    A new number of times to retry on flow run failure.

    0 retry_delay_seconds Union[int, float]

    A new number of seconds to wait before retrying the flow after failure. This is only applicable if retries is nonzero.

    0 persist_result Optional[bool]

    A new option for enabling or disabling result persistence.

    NotSet result_storage Optional[ResultStorage]

    A new storage type to use for results.

    NotSet result_serializer Optional[ResultSerializer]

    A new serializer to use for results.

    NotSet cache_result_in_memory bool

    A new value indicating if the flow's result should be cached in memory.

    None on_failure Optional[List[Callable[[Flow, FlowRun, State], None]]]

    A new list of callables to run when the flow enters a failed state.

    None on_completion Optional[List[Callable[[Flow, FlowRun, State], None]]]

    A new list of callables to run when the flow enters a completed state.

    None on_cancellation Optional[List[Callable[[Flow, FlowRun, State], None]]]

    A new list of callables to run when the flow enters a cancelling state.

    None on_crashed Optional[List[Callable[[Flow, FlowRun, State], None]]]

    A new list of callables to run when the flow enters a crashed state.

    None

    Returns:

    Type Description Self

    A new Flow instance.

    Create a new flow from an existing flow and update the name:\n\n>>> @flow(name=\"My flow\")\n>>> def my_flow():\n>>>     return 1\n>>>\n>>> new_flow = my_flow.with_options(name=\"My new flow\")\n\nCreate a new flow from an existing flow, update the task runner, and call\nit without an intermediate variable:\n\n>>> from prefect.task_runners import SequentialTaskRunner\n>>>\n>>> @flow\n>>> def my_flow(x, y):\n>>>     return x + y\n>>>\n>>> state = my_flow.with_options(task_runner=SequentialTaskRunner)(1, 3)\n>>> assert state.result() == 4\n
    Source code in prefect/flows.py
    def with_options(\n    self,\n    *,\n    name: str = None,\n    version: str = None,\n    retries: int = 0,\n    retry_delay_seconds: Union[int, float] = 0,\n    description: str = None,\n    flow_run_name: Optional[Union[Callable[[], str], str]] = None,\n    task_runner: Union[Type[BaseTaskRunner], BaseTaskRunner] = None,\n    timeout_seconds: Union[int, float] = None,\n    validate_parameters: bool = None,\n    persist_result: Optional[bool] = NotSet,\n    result_storage: Optional[ResultStorage] = NotSet,\n    result_serializer: Optional[ResultSerializer] = NotSet,\n    cache_result_in_memory: bool = None,\n    log_prints: Optional[bool] = NotSet,\n    on_completion: Optional[\n        List[Callable[[FlowSchema, FlowRun, State], None]]\n    ] = None,\n    on_failure: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    on_cancellation: Optional[\n        List[Callable[[FlowSchema, FlowRun, State], None]]\n    ] = None,\n    on_crashed: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n) -> Self:\n    \"\"\"\n    Create a new flow from the current object, updating provided options.\n\n    Args:\n        name: A new name for the flow.\n        version: A new version for the flow.\n        description: A new description for the flow.\n        flow_run_name: An optional name to distinguish runs of this flow; this name\n            can be provided as a string template with the flow's parameters as variables,\n            or a function that returns a string.\n        task_runner: A new task runner for the flow.\n        timeout_seconds: A new number of seconds to fail the flow after if still\n            running.\n        validate_parameters: A new value indicating if flow calls should validate\n            given parameters.\n        retries: A new number of times to retry on flow run failure.\n        retry_delay_seconds: A new number of seconds to wait before retrying the\n            flow after failure. This is only applicable if `retries` is nonzero.\n        persist_result: A new option for enabling or disabling result persistence.\n        result_storage: A new storage type to use for results.\n        result_serializer: A new serializer to use for results.\n        cache_result_in_memory: A new value indicating if the flow's result should\n            be cached in memory.\n        on_failure: A new list of callables to run when the flow enters a failed state.\n        on_completion: A new list of callables to run when the flow enters a completed state.\n        on_cancellation: A new list of callables to run when the flow enters a cancelling state.\n        on_crashed: A new list of callables to run when the flow enters a crashed state.\n\n    Returns:\n        A new `Flow` instance.\n\n    Examples:\n\n        Create a new flow from an existing flow and update the name:\n\n        >>> @flow(name=\"My flow\")\n        >>> def my_flow():\n        >>>     return 1\n        >>>\n        >>> new_flow = my_flow.with_options(name=\"My new flow\")\n\n        Create a new flow from an existing flow, update the task runner, and call\n        it without an intermediate variable:\n\n        >>> from prefect.task_runners import SequentialTaskRunner\n        >>>\n        >>> @flow\n        >>> def my_flow(x, y):\n        >>>     return x + y\n        >>>\n        >>> state = my_flow.with_options(task_runner=SequentialTaskRunner)(1, 3)\n        >>> assert state.result() == 4\n\n    \"\"\"\n    new_flow = Flow(\n        fn=self.fn,\n        name=name or self.name,\n        description=description or self.description,\n        flow_run_name=flow_run_name,\n        version=version or self.version,\n        task_runner=task_runner or self.task_runner,\n        retries=retries or self.retries,\n        retry_delay_seconds=retry_delay_seconds or self.retry_delay_seconds,\n        timeout_seconds=(\n            timeout_seconds if timeout_seconds is not None else self.timeout_seconds\n        ),\n        validate_parameters=(\n            validate_parameters\n            if validate_parameters is not None\n            else self.should_validate_parameters\n        ),\n        persist_result=(\n            persist_result if persist_result is not NotSet else self.persist_result\n        ),\n        result_storage=(\n            result_storage if result_storage is not NotSet else self.result_storage\n        ),\n        result_serializer=(\n            result_serializer\n            if result_serializer is not NotSet\n            else self.result_serializer\n        ),\n        cache_result_in_memory=(\n            cache_result_in_memory\n            if cache_result_in_memory is not None\n            else self.cache_result_in_memory\n        ),\n        log_prints=log_prints if log_prints is not NotSet else self.log_prints,\n        on_completion=on_completion or self.on_completion,\n        on_failure=on_failure or self.on_failure,\n        on_cancellation=on_cancellation or self.on_cancellation,\n        on_crashed=on_crashed or self.on_crashed,\n    )\n    new_flow._storage = self._storage\n    new_flow._entrypoint = self._entrypoint\n    return new_flow\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.flow","title":"flow","text":"

    Decorator to designate a function as a Prefect workflow.

    This decorator may be used for asynchronous or synchronous functions.

    Flow parameters must be serializable by Pydantic.

    Parameters:

    Name Type Description Default name Optional[str]

    An optional name for the flow; if not provided, the name will be inferred from the given function.

    None version Optional[str]

    An optional version string for the flow; if not provided, we will attempt to create a version string as a hash of the file containing the wrapped function; if the file cannot be located, the version will be null.

    None flow_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this flow; this name can be provided as a string template with the flow's parameters as variables, or a function that returns a string.

    None task_runner BaseTaskRunner

    An optional task runner to use for task execution within the flow; if not provided, a ConcurrentTaskRunner will be instantiated.

    ConcurrentTaskRunner description str

    An optional string description for the flow; if not provided, the description will be pulled from the docstring for the decorated function.

    None timeout_seconds Union[int, float]

    An optional number of seconds indicating a maximum runtime for the flow. If the flow exceeds this runtime, it will be marked as failed. Flow execution may continue until the next task is called.

    None validate_parameters bool

    By default, parameters passed to flows are validated by Pydantic. This will check that input values conform to the annotated types on the function. Where possible, values will be coerced into the correct type; for example, if a parameter is defined as x: int and \"5\" is passed, it will be resolved to 5. If set to False, no validation will be performed on flow parameters.

    True retries int

    An optional number of times to retry on flow run failure.

    None retry_delay_seconds Union[int, float]

    An optional number of seconds to wait before retrying the flow after failure. This is only applicable if retries is nonzero.

    None persist_result Optional[bool]

    An optional toggle indicating whether the result of this flow should be persisted to result storage. Defaults to None, which indicates that Prefect should choose whether the result should be persisted depending on the features being used.

    None result_storage Optional[ResultStorage]

    An optional block to use to persist the result of this flow. This value will be used as the default for any tasks in this flow. If not provided, the local file system will be used unless called as a subflow, at which point the default will be loaded from the parent flow.

    None result_serializer Optional[ResultSerializer]

    An optional serializer to use to serialize the result of this flow for persistence. This value will be used as the default for any tasks in this flow. If not provided, the value of PREFECT_RESULTS_DEFAULT_SERIALIZER will be used unless called as a subflow, at which point the default will be loaded from the parent flow.

    None log_prints Optional[bool]

    If set, print statements in the flow will be redirected to the Prefect logger for the flow run. Defaults to None, which indicates that the value from the parent flow should be used. If this is a parent flow, the default is pulled from the PREFECT_LOGGING_LOG_PRINTS setting.

    None on_completion Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of functions to call when the flow run is completed. Each function should accept three arguments: the flow, the flow run, and the final state of the flow run.

    None on_failure Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of functions to call when the flow run fails. Each function should accept three arguments: the flow, the flow run, and the final state of the flow run.

    None on_cancellation Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of functions to call when the flow run is cancelled. These functions will be passed the flow, flow run, and final state.

    None on_crashed Optional[List[Callable[[Flow, FlowRun, State], None]]]

    An optional list of functions to call when the flow run crashes. Each function should accept three arguments: the flow, the flow run, and the final state of the flow run.

    None

    Returns:

    Type Description

    A callable Flow object which, when called, will run the flow and return its

    final state.

    Examples:

    Define a simple flow

    >>> from prefect import flow\n>>> @flow\n>>> def add(x, y):\n>>>     return x + y\n

    Define an async flow

    >>> @flow\n>>> async def add(x, y):\n>>>     return x + y\n

    Define a flow with a version and description

    >>> @flow(version=\"first-flow\", description=\"This flow is empty!\")\n>>> def my_flow():\n>>>     pass\n

    Define a flow with a custom name

    >>> @flow(name=\"The Ultimate Flow\")\n>>> def my_flow():\n>>>     pass\n

    Define a flow that submits its tasks to dask

    >>> from prefect_dask.task_runners import DaskTaskRunner\n>>>\n>>> @flow(task_runner=DaskTaskRunner)\n>>> def my_flow():\n>>>     pass\n
    Source code in prefect/flows.py
    def flow(\n    __fn=None,\n    *,\n    name: Optional[str] = None,\n    version: Optional[str] = None,\n    flow_run_name: Optional[Union[Callable[[], str], str]] = None,\n    retries: int = None,\n    retry_delay_seconds: Union[int, float] = None,\n    task_runner: BaseTaskRunner = ConcurrentTaskRunner,\n    description: str = None,\n    timeout_seconds: Union[int, float] = None,\n    validate_parameters: bool = True,\n    persist_result: Optional[bool] = None,\n    result_storage: Optional[ResultStorage] = None,\n    result_serializer: Optional[ResultSerializer] = None,\n    cache_result_in_memory: bool = True,\n    log_prints: Optional[bool] = None,\n    on_completion: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    on_failure: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n    on_cancellation: Optional[\n        List[Callable[[FlowSchema, FlowRun, State], None]]\n    ] = None,\n    on_crashed: Optional[List[Callable[[FlowSchema, FlowRun, State], None]]] = None,\n):\n    \"\"\"\n    Decorator to designate a function as a Prefect workflow.\n\n    This decorator may be used for asynchronous or synchronous functions.\n\n    Flow parameters must be serializable by Pydantic.\n\n    Args:\n        name: An optional name for the flow; if not provided, the name will be inferred\n            from the given function.\n        version: An optional version string for the flow; if not provided, we will\n            attempt to create a version string as a hash of the file containing the\n            wrapped function; if the file cannot be located, the version will be null.\n        flow_run_name: An optional name to distinguish runs of this flow; this name can\n            be provided as a string template with the flow's parameters as variables,\n            or a function that returns a string.\n        task_runner: An optional task runner to use for task execution within the flow; if\n            not provided, a `ConcurrentTaskRunner` will be instantiated.\n        description: An optional string description for the flow; if not provided, the\n            description will be pulled from the docstring for the decorated function.\n        timeout_seconds: An optional number of seconds indicating a maximum runtime for\n            the flow. If the flow exceeds this runtime, it will be marked as failed.\n            Flow execution may continue until the next task is called.\n        validate_parameters: By default, parameters passed to flows are validated by\n            Pydantic. This will check that input values conform to the annotated types\n            on the function. Where possible, values will be coerced into the correct\n            type; for example, if a parameter is defined as `x: int` and \"5\" is passed,\n            it will be resolved to `5`. If set to `False`, no validation will be\n            performed on flow parameters.\n        retries: An optional number of times to retry on flow run failure.\n        retry_delay_seconds: An optional number of seconds to wait before retrying the\n            flow after failure. This is only applicable if `retries` is nonzero.\n        persist_result: An optional toggle indicating whether the result of this flow\n            should be persisted to result storage. Defaults to `None`, which indicates\n            that Prefect should choose whether the result should be persisted depending on\n            the features being used.\n        result_storage: An optional block to use to persist the result of this flow.\n            This value will be used as the default for any tasks in this flow.\n            If not provided, the local file system will be used unless called as\n            a subflow, at which point the default will be loaded from the parent flow.\n        result_serializer: An optional serializer to use to serialize the result of this\n            flow for persistence. This value will be used as the default for any tasks\n            in this flow. If not provided, the value of `PREFECT_RESULTS_DEFAULT_SERIALIZER`\n            will be used unless called as a subflow, at which point the default will be\n            loaded from the parent flow.\n        log_prints: If set, `print` statements in the flow will be redirected to the\n            Prefect logger for the flow run. Defaults to `None`, which indicates that\n            the value from the parent flow should be used. If this is a parent flow,\n            the default is pulled from the `PREFECT_LOGGING_LOG_PRINTS` setting.\n        on_completion: An optional list of functions to call when the flow run is\n            completed. Each function should accept three arguments: the flow, the flow\n            run, and the final state of the flow run.\n        on_failure: An optional list of functions to call when the flow run fails. Each\n            function should accept three arguments: the flow, the flow run, and the\n            final state of the flow run.\n        on_cancellation: An optional list of functions to call when the flow run is\n            cancelled. These functions will be passed the flow, flow run, and final state.\n        on_crashed: An optional list of functions to call when the flow run crashes. Each\n            function should accept three arguments: the flow, the flow run, and the\n            final state of the flow run.\n\n    Returns:\n        A callable `Flow` object which, when called, will run the flow and return its\n        final state.\n\n    Examples:\n        Define a simple flow\n\n        >>> from prefect import flow\n        >>> @flow\n        >>> def add(x, y):\n        >>>     return x + y\n\n        Define an async flow\n\n        >>> @flow\n        >>> async def add(x, y):\n        >>>     return x + y\n\n        Define a flow with a version and description\n\n        >>> @flow(version=\"first-flow\", description=\"This flow is empty!\")\n        >>> def my_flow():\n        >>>     pass\n\n        Define a flow with a custom name\n\n        >>> @flow(name=\"The Ultimate Flow\")\n        >>> def my_flow():\n        >>>     pass\n\n        Define a flow that submits its tasks to dask\n\n        >>> from prefect_dask.task_runners import DaskTaskRunner\n        >>>\n        >>> @flow(task_runner=DaskTaskRunner)\n        >>> def my_flow():\n        >>>     pass\n    \"\"\"\n    if __fn:\n        return cast(\n            Flow[P, R],\n            Flow(\n                fn=__fn,\n                name=name,\n                version=version,\n                flow_run_name=flow_run_name,\n                task_runner=task_runner,\n                description=description,\n                timeout_seconds=timeout_seconds,\n                validate_parameters=validate_parameters,\n                retries=retries,\n                retry_delay_seconds=retry_delay_seconds,\n                persist_result=persist_result,\n                result_storage=result_storage,\n                result_serializer=result_serializer,\n                cache_result_in_memory=cache_result_in_memory,\n                log_prints=log_prints,\n                on_completion=on_completion,\n                on_failure=on_failure,\n                on_cancellation=on_cancellation,\n                on_crashed=on_crashed,\n            ),\n        )\n    else:\n        return cast(\n            Callable[[Callable[P, R]], Flow[P, R]],\n            partial(\n                flow,\n                name=name,\n                version=version,\n                flow_run_name=flow_run_name,\n                task_runner=task_runner,\n                description=description,\n                timeout_seconds=timeout_seconds,\n                validate_parameters=validate_parameters,\n                retries=retries,\n                retry_delay_seconds=retry_delay_seconds,\n                persist_result=persist_result,\n                result_storage=result_storage,\n                result_serializer=result_serializer,\n                cache_result_in_memory=cache_result_in_memory,\n                log_prints=log_prints,\n                on_completion=on_completion,\n                on_failure=on_failure,\n                on_cancellation=on_cancellation,\n                on_crashed=on_crashed,\n            ),\n        )\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.load_flow_from_entrypoint","title":"load_flow_from_entrypoint","text":"

    Extract a flow object from a script at an entrypoint by running all of the code in the file.

    Parameters:

    Name Type Description Default entrypoint str

    a string in the format <path_to_script>:<flow_func_name>

    required

    Returns:

    Type Description Flow

    The flow object from the script

    Raises:

    Type Description FlowScriptError

    If an exception is encountered while running the script

    MissingFlowError

    If the flow function specified in the entrypoint does not exist

    Source code in prefect/flows.py
    def load_flow_from_entrypoint(entrypoint: str) -> Flow:\n    \"\"\"\n    Extract a flow object from a script at an entrypoint by running all of the code in the file.\n\n    Args:\n        entrypoint: a string in the format `<path_to_script>:<flow_func_name>`\n\n    Returns:\n        The flow object from the script\n\n    Raises:\n        FlowScriptError: If an exception is encountered while running the script\n        MissingFlowError: If the flow function specified in the entrypoint does not exist\n    \"\"\"\n    with PrefectObjectRegistry(\n        block_code_execution=True,\n        capture_failures=True,\n    ):\n        # split by the last colon once to handle Windows paths with drive letters i.e C:\\path\\to\\file.py:do_stuff\n        path, func_name = entrypoint.rsplit(\":\", maxsplit=1)\n        try:\n            flow = import_object(entrypoint)\n        except AttributeError as exc:\n            raise MissingFlowError(\n                f\"Flow function with name {func_name!r} not found in {path!r}. \"\n            ) from exc\n\n        if not isinstance(flow, Flow):\n            raise MissingFlowError(\n                f\"Function with name {func_name!r} is not a flow. Make sure that it is \"\n                \"decorated with '@flow'.\"\n            )\n\n        return flow\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.load_flow_from_script","title":"load_flow_from_script","text":"

    Extract a flow object from a script by running all of the code in the file.

    If the script has multiple flows in it, a flow name must be provided to specify the flow to return.

    Parameters:

    Name Type Description Default path str

    A path to a Python script containing flows

    required flow_name str

    An optional flow name to look for in the script

    None

    Returns:

    Type Description Flow

    The flow object from the script

    Raises:

    Type Description FlowScriptError

    If an exception is encountered while running the script

    MissingFlowError

    If no flows exist in the iterable

    MissingFlowError

    If a flow name is provided and that flow does not exist

    UnspecifiedFlowError

    If multiple flows exist but no flow name was provided

    Source code in prefect/flows.py
    def load_flow_from_script(path: str, flow_name: str = None) -> Flow:\n    \"\"\"\n    Extract a flow object from a script by running all of the code in the file.\n\n    If the script has multiple flows in it, a flow name must be provided to specify\n    the flow to return.\n\n    Args:\n        path: A path to a Python script containing flows\n        flow_name: An optional flow name to look for in the script\n\n    Returns:\n        The flow object from the script\n\n    Raises:\n        FlowScriptError: If an exception is encountered while running the script\n        MissingFlowError: If no flows exist in the iterable\n        MissingFlowError: If a flow name is provided and that flow does not exist\n        UnspecifiedFlowError: If multiple flows exist but no flow name was provided\n    \"\"\"\n    return select_flow(\n        load_flows_from_script(path),\n        flow_name=flow_name,\n        from_message=f\"in script '{path}'\",\n    )\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.load_flow_from_text","title":"load_flow_from_text","text":"

    Load a flow from a text script.

    The script will be written to a temporary local file path so errors can refer to line numbers and contextual tracebacks can be provided.

    Source code in prefect/flows.py
    def load_flow_from_text(script_contents: AnyStr, flow_name: str):\n    \"\"\"\n    Load a flow from a text script.\n\n    The script will be written to a temporary local file path so errors can refer\n    to line numbers and contextual tracebacks can be provided.\n    \"\"\"\n    with NamedTemporaryFile(\n        mode=\"wt\" if isinstance(script_contents, str) else \"wb\",\n        prefix=f\"flow-script-{flow_name}\",\n        suffix=\".py\",\n        delete=False,\n    ) as tmpfile:\n        tmpfile.write(script_contents)\n        tmpfile.flush()\n    try:\n        flow = load_flow_from_script(tmpfile.name, flow_name=flow_name)\n    finally:\n        # windows compat\n        tmpfile.close()\n        os.remove(tmpfile.name)\n    return flow\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.load_flows_from_script","title":"load_flows_from_script","text":"

    Load all flow objects from the given python script. All of the code in the file will be executed.

    Returns:

    Type Description List[Flow]

    A list of flows

    Raises:

    Type Description FlowScriptError

    If an exception is encountered while running the script

    Source code in prefect/flows.py
    def load_flows_from_script(path: str) -> List[Flow]:\n    \"\"\"\n    Load all flow objects from the given python script. All of the code in the file\n    will be executed.\n\n    Returns:\n        A list of flows\n\n    Raises:\n        FlowScriptError: If an exception is encountered while running the script\n    \"\"\"\n    return registry_from_script(path).get_instances(Flow)\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/flows/#prefect.flows.select_flow","title":"select_flow","text":"

    Select the only flow in an iterable or a flow specified by name.

    Returns A single flow object

    Raises:

    Type Description MissingFlowError

    If no flows exist in the iterable

    MissingFlowError

    If a flow name is provided and that flow does not exist

    UnspecifiedFlowError

    If multiple flows exist but no flow name was provided

    Source code in prefect/flows.py
    def select_flow(\n    flows: Iterable[Flow], flow_name: str = None, from_message: str = None\n) -> Flow:\n    \"\"\"\n    Select the only flow in an iterable or a flow specified by name.\n\n    Returns\n        A single flow object\n\n    Raises:\n        MissingFlowError: If no flows exist in the iterable\n        MissingFlowError: If a flow name is provided and that flow does not exist\n        UnspecifiedFlowError: If multiple flows exist but no flow name was provided\n    \"\"\"\n    # Convert to flows by name\n    flows = {f.name: f for f in flows}\n\n    # Add a leading space if given, otherwise use an empty string\n    from_message = (\" \" + from_message) if from_message else \"\"\n    if not flows:\n        raise MissingFlowError(f\"No flows found{from_message}.\")\n\n    elif flow_name and flow_name not in flows:\n        raise MissingFlowError(\n            f\"Flow {flow_name!r} not found{from_message}. \"\n            f\"Found the following flows: {listrepr(flows.keys())}. \"\n            \"Check to make sure that your flow function is decorated with `@flow`.\"\n        )\n\n    elif not flow_name and len(flows) > 1:\n        raise UnspecifiedFlowError(\n            (\n                f\"Found {len(flows)} flows{from_message}:\"\n                f\" {listrepr(sorted(flows.keys()))}. Specify a flow name to select a\"\n                \" flow.\"\n            ),\n        )\n\n    if flow_name:\n        return flows[flow_name]\n    else:\n        return list(flows.values())[0]\n
    ","tags":["Python API","flows","parameters"]},{"location":"api-ref/prefect/futures/","title":"prefect.futures","text":"","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures","title":"prefect.futures","text":"

    Futures represent the execution of a task and allow retrieval of the task run's state.

    This module contains the definition for futures as well as utilities for resolving futures in nested data structures.

    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.PrefectFuture","title":"PrefectFuture","text":"

    Bases: Generic[R, A]

    Represents the result of a computation happening in a task runner.

    When tasks are called, they are submitted to a task runner which creates a future for access to the state and result of the task.

    Examples:

    Define a task that returns a string

    >>> from prefect import flow, task\n>>> @task\n>>> def my_task() -> str:\n>>>     return \"hello\"\n

    Calls of this task in a flow will return a future

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()  # PrefectFuture[str, Sync] includes result type\n>>>     future.task_run.id  # UUID for the task run\n

    Wait for the task to complete

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     final_state = future.wait()\n

    Wait N seconds for the task to complete

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     final_state = future.wait(0.1)\n>>>     if final_state:\n>>>         ... # Task done\n>>>     else:\n>>>         ... # Task not done yet\n

    Wait for a task to complete and retrieve its result

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     result = future.result()\n>>>     assert result == \"hello\"\n

    Wait N seconds for a task to complete and retrieve its result

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     result = future.result(timeout=5)\n>>>     assert result == \"hello\"\n

    Retrieve the state of a task without waiting for completion

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task.submit()\n>>>     state = future.get_state()\n
    Source code in prefect/futures.py
    class PrefectFuture(Generic[R, A]):\n    \"\"\"\n    Represents the result of a computation happening in a task runner.\n\n    When tasks are called, they are submitted to a task runner which creates a future\n    for access to the state and result of the task.\n\n    Examples:\n        Define a task that returns a string\n\n        >>> from prefect import flow, task\n        >>> @task\n        >>> def my_task() -> str:\n        >>>     return \"hello\"\n\n        Calls of this task in a flow will return a future\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()  # PrefectFuture[str, Sync] includes result type\n        >>>     future.task_run.id  # UUID for the task run\n\n        Wait for the task to complete\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     final_state = future.wait()\n\n        Wait N seconds for the task to complete\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     final_state = future.wait(0.1)\n        >>>     if final_state:\n        >>>         ... # Task done\n        >>>     else:\n        >>>         ... # Task not done yet\n\n        Wait for a task to complete and retrieve its result\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     result = future.result()\n        >>>     assert result == \"hello\"\n\n        Wait N seconds for a task to complete and retrieve its result\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     result = future.result(timeout=5)\n        >>>     assert result == \"hello\"\n\n        Retrieve the state of a task without waiting for completion\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task.submit()\n        >>>     state = future.get_state()\n    \"\"\"\n\n    def __init__(\n        self,\n        name: str,\n        key: UUID,\n        task_runner: \"BaseTaskRunner\",\n        asynchronous: A = True,\n        _final_state: State[R] = None,  # Exposed for testing\n    ) -> None:\n        self.key = key\n        self.name = name\n        self.asynchronous = asynchronous\n        self.task_run = None\n        self._final_state = _final_state\n        self._exception: Optional[Exception] = None\n        self._task_runner = task_runner\n        self._submitted = anyio.Event()\n\n        self._loop = asyncio.get_running_loop()\n\n    @overload\n    def wait(\n        self: \"PrefectFuture[R, Async]\", timeout: None = None\n    ) -> Awaitable[State[R]]:\n        ...\n\n    @overload\n    def wait(self: \"PrefectFuture[R, Sync]\", timeout: None = None) -> State[R]:\n        ...\n\n    @overload\n    def wait(\n        self: \"PrefectFuture[R, Async]\", timeout: float\n    ) -> Awaitable[Optional[State[R]]]:\n        ...\n\n    @overload\n    def wait(self: \"PrefectFuture[R, Sync]\", timeout: float) -> Optional[State[R]]:\n        ...\n\n    def wait(self, timeout=None):\n        \"\"\"\n        Wait for the run to finish and return the final state\n\n        If the timeout is reached before the run reaches a final state,\n        `None` is returned.\n        \"\"\"\n        wait = create_call(self._wait, timeout=timeout)\n        if self.asynchronous:\n            return from_async.call_soon_in_loop_thread(wait).aresult()\n        else:\n            # type checking cannot handle the overloaded timeout passing\n            return from_sync.call_soon_in_loop_thread(wait).result()  # type: ignore\n\n    @overload\n    async def _wait(self, timeout: None = None) -> State[R]:\n        ...\n\n    @overload\n    async def _wait(self, timeout: float) -> Optional[State[R]]:\n        ...\n\n    async def _wait(self, timeout=None):\n        \"\"\"\n        Async implementation for `wait`\n        \"\"\"\n        await self._wait_for_submission()\n\n        if self._final_state:\n            return self._final_state\n\n        self._final_state = await self._task_runner.wait(self.key, timeout)\n        return self._final_state\n\n    @overload\n    def result(\n        self: \"PrefectFuture[R, Sync]\",\n        timeout: float = None,\n        raise_on_failure: bool = True,\n    ) -> R:\n        ...\n\n    @overload\n    def result(\n        self: \"PrefectFuture[R, Sync]\",\n        timeout: float = None,\n        raise_on_failure: bool = False,\n    ) -> Union[R, Exception]:\n        ...\n\n    @overload\n    def result(\n        self: \"PrefectFuture[R, Async]\",\n        timeout: float = None,\n        raise_on_failure: bool = True,\n    ) -> Awaitable[R]:\n        ...\n\n    @overload\n    def result(\n        self: \"PrefectFuture[R, Async]\",\n        timeout: float = None,\n        raise_on_failure: bool = False,\n    ) -> Awaitable[Union[R, Exception]]:\n        ...\n\n    def result(self, timeout: float = None, raise_on_failure: bool = True):\n        \"\"\"\n        Wait for the run to finish and return the final state.\n\n        If the timeout is reached before the run reaches a final state, a `TimeoutError`\n        will be raised.\n\n        If `raise_on_failure` is `True` and the task run failed, the task run's\n        exception will be raised.\n        \"\"\"\n        result = create_call(\n            self._result, timeout=timeout, raise_on_failure=raise_on_failure\n        )\n        if self.asynchronous:\n            return from_async.call_soon_in_loop_thread(result).aresult()\n        else:\n            return from_sync.call_soon_in_loop_thread(result).result()\n\n    async def _result(self, timeout: float = None, raise_on_failure: bool = True):\n        \"\"\"\n        Async implementation of `result`\n        \"\"\"\n        final_state = await self._wait(timeout=timeout)\n        if not final_state:\n            raise TimeoutError(\"Call timed out before task finished.\")\n        return await final_state.result(raise_on_failure=raise_on_failure, fetch=True)\n\n    @overload\n    def get_state(\n        self: \"PrefectFuture[R, Async]\", client: PrefectClient = None\n    ) -> Awaitable[State[R]]:\n        ...\n\n    @overload\n    def get_state(\n        self: \"PrefectFuture[R, Sync]\", client: PrefectClient = None\n    ) -> State[R]:\n        ...\n\n    def get_state(self, client: PrefectClient = None):\n        \"\"\"\n        Get the current state of the task run.\n        \"\"\"\n        if self.asynchronous:\n            return cast(Awaitable[State[R]], self._get_state(client=client))\n        else:\n            return cast(State[R], sync(self._get_state, client=client))\n\n    @inject_client\n    async def _get_state(self, client: PrefectClient = None) -> State[R]:\n        assert client is not None  # always injected\n\n        # We must wait for the task run id to be populated\n        await self._wait_for_submission()\n\n        task_run = await client.read_task_run(self.task_run.id)\n\n        if not task_run:\n            raise RuntimeError(\"Future has no associated task run in the server.\")\n\n        # Update the task run reference\n        self.task_run = task_run\n        return task_run.state\n\n    async def _wait_for_submission(self):\n        await run_coroutine_in_loop_from_async(self._loop, self._submitted.wait())\n\n    def __hash__(self) -> int:\n        return hash(self.key)\n\n    def __repr__(self) -> str:\n        return f\"PrefectFuture({self.name!r})\"\n\n    def __bool__(self) -> bool:\n        warnings.warn(\n            (\n                \"A 'PrefectFuture' from a task call was cast to a boolean; \"\n                \"did you mean to check the result of the task instead? \"\n                \"e.g. `if my_task().result(): ...`\"\n            ),\n            stacklevel=2,\n        )\n        return True\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.PrefectFuture.get_state","title":"get_state","text":"

    Get the current state of the task run.

    Source code in prefect/futures.py
    def get_state(self, client: PrefectClient = None):\n    \"\"\"\n    Get the current state of the task run.\n    \"\"\"\n    if self.asynchronous:\n        return cast(Awaitable[State[R]], self._get_state(client=client))\n    else:\n        return cast(State[R], sync(self._get_state, client=client))\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.PrefectFuture.result","title":"result","text":"

    Wait for the run to finish and return the final state.

    If the timeout is reached before the run reaches a final state, a TimeoutError will be raised.

    If raise_on_failure is True and the task run failed, the task run's exception will be raised.

    Source code in prefect/futures.py
    def result(self, timeout: float = None, raise_on_failure: bool = True):\n    \"\"\"\n    Wait for the run to finish and return the final state.\n\n    If the timeout is reached before the run reaches a final state, a `TimeoutError`\n    will be raised.\n\n    If `raise_on_failure` is `True` and the task run failed, the task run's\n    exception will be raised.\n    \"\"\"\n    result = create_call(\n        self._result, timeout=timeout, raise_on_failure=raise_on_failure\n    )\n    if self.asynchronous:\n        return from_async.call_soon_in_loop_thread(result).aresult()\n    else:\n        return from_sync.call_soon_in_loop_thread(result).result()\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.PrefectFuture.wait","title":"wait","text":"

    Wait for the run to finish and return the final state

    If the timeout is reached before the run reaches a final state, None is returned.

    Source code in prefect/futures.py
    def wait(self, timeout=None):\n    \"\"\"\n    Wait for the run to finish and return the final state\n\n    If the timeout is reached before the run reaches a final state,\n    `None` is returned.\n    \"\"\"\n    wait = create_call(self._wait, timeout=timeout)\n    if self.asynchronous:\n        return from_async.call_soon_in_loop_thread(wait).aresult()\n    else:\n        # type checking cannot handle the overloaded timeout passing\n        return from_sync.call_soon_in_loop_thread(wait).result()  # type: ignore\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.call_repr","title":"call_repr","text":"

    Generate a repr for a function call as \"fn_name(arg_value, kwarg_name=kwarg_value)\"

    Source code in prefect/futures.py
    def call_repr(__fn: Callable, *args: Any, **kwargs: Any) -> str:\n    \"\"\"\n    Generate a repr for a function call as \"fn_name(arg_value, kwarg_name=kwarg_value)\"\n    \"\"\"\n\n    name = __fn.__name__\n\n    # TODO: If this computation is concerningly expensive, we can iterate checking the\n    #       length at each arg or avoid calling `repr` on args with large amounts of\n    #       data\n    call_args = \", \".join(\n        [repr(arg) for arg in args]\n        + [f\"{key}={repr(val)}\" for key, val in kwargs.items()]\n    )\n\n    # Enforce a maximum length\n    if len(call_args) > 100:\n        call_args = call_args[:100] + \"...\"\n\n    return f\"{name}({call_args})\"\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.resolve_futures_to_data","title":"resolve_futures_to_data async","text":"

    Given a Python built-in collection, recursively find PrefectFutures and build a new collection with the same structure with futures resolved to their results. Resolving futures to their results may wait for execution to complete and require communication with the API.

    Unsupported object types will be returned without modification.

    Source code in prefect/futures.py
    async def resolve_futures_to_data(\n    expr: Union[PrefectFuture[R, Any], Any],\n    raise_on_failure: bool = True,\n) -> Union[R, Any]:\n    \"\"\"\n    Given a Python built-in collection, recursively find `PrefectFutures` and build a\n    new collection with the same structure with futures resolved to their results.\n    Resolving futures to their results may wait for execution to complete and require\n    communication with the API.\n\n    Unsupported object types will be returned without modification.\n    \"\"\"\n    futures: Set[PrefectFuture] = set()\n\n    maybe_expr = visit_collection(\n        expr,\n        visit_fn=partial(_collect_futures, futures),\n        return_data=False,\n        context={},\n    )\n    if maybe_expr is not None:\n        expr = maybe_expr\n\n    # Get results\n    results = await asyncio.gather(\n        *[\n            # We must wait for the future in the thread it was created in\n            from_async.call_soon_in_loop_thread(\n                create_call(future._result, raise_on_failure=raise_on_failure)\n            ).aresult()\n            for future in futures\n        ]\n    )\n\n    results_by_future = dict(zip(futures, results))\n\n    def replace_futures_with_results(expr, context):\n        # Expressions inside quotes should not be modified\n        if isinstance(context.get(\"annotation\"), quote):\n            raise StopVisiting()\n\n        if isinstance(expr, PrefectFuture):\n            return results_by_future[expr]\n        else:\n            return expr\n\n    return visit_collection(\n        expr,\n        visit_fn=replace_futures_with_results,\n        return_data=True,\n        context={},\n    )\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/futures/#prefect.futures.resolve_futures_to_states","title":"resolve_futures_to_states async","text":"

    Given a Python built-in collection, recursively find PrefectFutures and build a new collection with the same structure with futures resolved to their final states. Resolving futures to their final states may wait for execution to complete.

    Unsupported object types will be returned without modification.

    Source code in prefect/futures.py
    async def resolve_futures_to_states(\n    expr: Union[PrefectFuture[R, Any], Any]\n) -> Union[State[R], Any]:\n    \"\"\"\n    Given a Python built-in collection, recursively find `PrefectFutures` and build a\n    new collection with the same structure with futures resolved to their final states.\n    Resolving futures to their final states may wait for execution to complete.\n\n    Unsupported object types will be returned without modification.\n    \"\"\"\n    futures: Set[PrefectFuture] = set()\n\n    visit_collection(\n        expr,\n        visit_fn=partial(_collect_futures, futures),\n        return_data=False,\n        context={},\n    )\n\n    # Get final states for each future\n    states = await asyncio.gather(\n        *[\n            # We must wait for the future in the thread it was created in\n            from_async.call_soon_in_loop_thread(create_call(future._wait)).aresult()\n            for future in futures\n        ]\n    )\n\n    states_by_future = dict(zip(futures, states))\n\n    def replace_futures_with_states(expr, context):\n        # Expressions inside quotes should not be modified\n        if isinstance(context.get(\"annotation\"), quote):\n            raise StopVisiting()\n\n        if isinstance(expr, PrefectFuture):\n            return states_by_future[expr]\n        else:\n            return expr\n\n    return visit_collection(\n        expr,\n        visit_fn=replace_futures_with_states,\n        return_data=True,\n        context={},\n    )\n
    ","tags":["Python API","tasks","futures","states"]},{"location":"api-ref/prefect/infrastructure/","title":"prefect.infrastructure","text":"","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure","title":"prefect.infrastructure","text":"","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.DockerContainer","title":"DockerContainer","text":"

    Bases: Infrastructure

    Runs a command in a container.

    Requires a Docker Engine to be connectable. Docker settings will be retrieved from the environment.

    Click here to see a tutorial.

    Attributes:

    Name Type Description auto_remove bool

    If set, the container will be removed on completion. Otherwise, the container will remain after exit for inspection.

    command bool

    A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this.

    env bool

    Environment variables to set for the container.

    image str

    An optional string specifying the tag of a Docker image to use. Defaults to the Prefect image.

    image_pull_policy Optional[ImagePullPolicy]

    Specifies if the image should be pulled. One of 'ALWAYS', 'NEVER', 'IF_NOT_PRESENT'.

    image_registry Optional[DockerRegistry]

    A DockerRegistry block containing credentials to use if image is stored in a private image registry.

    labels Optional[DockerRegistry]

    An optional dictionary of labels, mapping name to value.

    name Optional[DockerRegistry]

    An optional name for the container.

    network_mode Optional[str]

    Set the network mode for the created container. Defaults to 'host' if a local API url is detected, otherwise the Docker default of 'bridge' is used. If 'networks' is set, this cannot be set.

    networks List[str]

    An optional list of strings specifying Docker networks to connect the container to.

    stream_output bool

    If set, stream output from the container to local standard output.

    volumes List[str]

    An optional list of volume mount strings in the format of \"local_path:container_path\".

    memswap_limit Union[int, str]

    Total memory (memory + swap), -1 to disable swap. Should only be set if mem_limit is also set. If mem_limit is set, this defaults to allowing the container to use as much swap as memory. For example, if mem_limit is 300m and memswap_limit is not set, the container can use 600m in total of memory and swap.

    mem_limit Union[float, str]

    Memory limit of the created container. Accepts float values to enforce a limit in bytes or a string with a unit e.g. 100000b, 1000k, 128m, 1g. If a string is given without a unit, bytes are assumed.

    privileged bool

    Give extended privileges to this container.

    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.DockerContainer--connecting-to-a-locally-hosted-prefect-api","title":"Connecting to a locally hosted Prefect API","text":"

    If using a local API URL on Linux, we will update the network mode default to 'host' to enable connectivity. If using another OS or an alternative network mode is used, we will replace 'localhost' in the API URL with 'host.docker.internal'. Generally, this will enable connectivity, but the API URL can be provided as an environment variable to override inference in more complex use-cases.

    Note, if using 'host.docker.internal' in the API URL on Linux, the API must be bound to 0.0.0.0 or the Docker IP address to allow connectivity. On macOS, this is not necessary and the API is connectable while bound to localhost.

    Source code in prefect/infrastructure/container.py
    class DockerContainer(Infrastructure):\n    \"\"\"\n    Runs a command in a container.\n\n    Requires a Docker Engine to be connectable. Docker settings will be retrieved from\n    the environment.\n\n    Click [here](https://docs.prefect.io/guides/deployment/docker) to see a tutorial.\n\n    Attributes:\n        auto_remove: If set, the container will be removed on completion. Otherwise,\n            the container will remain after exit for inspection.\n        command: A list of strings specifying the command to run in the container to\n            start the flow run. In most cases you should not override this.\n        env: Environment variables to set for the container.\n        image: An optional string specifying the tag of a Docker image to use.\n            Defaults to the Prefect image.\n        image_pull_policy: Specifies if the image should be pulled. One of 'ALWAYS',\n            'NEVER', 'IF_NOT_PRESENT'.\n        image_registry: A `DockerRegistry` block containing credentials to use if `image` is stored in a private\n            image registry.\n        labels: An optional dictionary of labels, mapping name to value.\n        name: An optional name for the container.\n        network_mode: Set the network mode for the created container. Defaults to 'host'\n            if a local API url is detected, otherwise the Docker default of 'bridge' is\n            used. If 'networks' is set, this cannot be set.\n        networks: An optional list of strings specifying Docker networks to connect the\n            container to.\n        stream_output: If set, stream output from the container to local standard output.\n        volumes: An optional list of volume mount strings in the format of\n            \"local_path:container_path\".\n        memswap_limit: Total memory (memory + swap), -1 to disable swap. Should only be\n            set if `mem_limit` is also set. If `mem_limit` is set, this defaults to\n            allowing the container to use as much swap as memory. For example, if\n            `mem_limit` is 300m and `memswap_limit` is not set, the container can use\n            600m in total of memory and swap.\n        mem_limit: Memory limit of the created container. Accepts float values to enforce\n            a limit in bytes or a string with a unit e.g. 100000b, 1000k, 128m, 1g.\n            If a string is given without a unit, bytes are assumed.\n        privileged: Give extended privileges to this container.\n\n    ## Connecting to a locally hosted Prefect API\n\n    If using a local API URL on Linux, we will update the network mode default to 'host'\n    to enable connectivity. If using another OS or an alternative network mode is used,\n    we will replace 'localhost' in the API URL with 'host.docker.internal'. Generally,\n    this will enable connectivity, but the API URL can be provided as an environment\n    variable to override inference in more complex use-cases.\n\n    Note, if using 'host.docker.internal' in the API URL on Linux, the API must be bound\n    to 0.0.0.0 or the Docker IP address to allow connectivity. On macOS, this is not\n    necessary and the API is connectable while bound to localhost.\n    \"\"\"\n\n    type: Literal[\"docker-container\"] = Field(\n        default=\"docker-container\", description=\"The type of infrastructure.\"\n    )\n    image: str = Field(\n        default_factory=get_prefect_image_name,\n        description=\"Tag of a Docker image to use. Defaults to the Prefect image.\",\n    )\n    image_pull_policy: Optional[ImagePullPolicy] = Field(\n        default=None, description=\"Specifies if the image should be pulled.\"\n    )\n    image_registry: Optional[DockerRegistry] = None\n    networks: List[str] = Field(\n        default_factory=list,\n        description=(\n            \"A list of strings specifying Docker networks to connect the container to.\"\n        ),\n    )\n    network_mode: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The network mode for the created container (e.g. host, bridge). If\"\n            \" 'networks' is set, this cannot be set.\"\n        ),\n    )\n    auto_remove: bool = Field(\n        default=False,\n        description=\"If set, the container will be removed on completion.\",\n    )\n    volumes: List[str] = Field(\n        default_factory=list,\n        description=(\n            \"A list of volume mount strings in the format of\"\n            ' \"local_path:container_path\".'\n        ),\n    )\n    stream_output: bool = Field(\n        default=True,\n        description=(\n            \"If set, the output will be streamed from the container to local standard\"\n            \" output.\"\n        ),\n    )\n    memswap_limit: Union[int, str] = Field(\n        default=None,\n        description=(\n            \"Total memory (memory + swap), -1 to disable swap. Should only be \"\n            \"set if `mem_limit` is also set. If `mem_limit` is set, this defaults to\"\n            \"allowing the container to use as much swap as memory. For example, if \"\n            \"`mem_limit` is 300m and `memswap_limit` is not set, the container can use \"\n            \"600m in total of memory and swap.\"\n        ),\n    )\n    mem_limit: Union[float, str] = Field(\n        default=None,\n        description=(\n            \"Memory limit of the created container. Accepts float values to enforce \"\n            \"a limit in bytes or a string with a unit e.g. 100000b, 1000k, 128m, 1g. \"\n            \"If a string is given without a unit, bytes are assumed.\"\n        ),\n    )\n    privileged: bool = Field(\n        default=False,\n        description=\"Give extended privileges to this container.\",\n    )\n\n    _block_type_name = \"Docker Container\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/14a315b79990200db7341e42553e23650b34bb96-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/infrastructure/#prefect.infrastructure.DockerContainer\"\n\n    @validator(\"labels\")\n    def convert_labels_to_docker_format(cls, labels: Dict[str, str]):\n        labels = labels or {}\n        new_labels = {}\n        for name, value in labels.items():\n            if \"/\" in name:\n                namespace, key = name.split(\"/\", maxsplit=1)\n                new_namespace = \".\".join(reversed(namespace.split(\".\")))\n                new_labels[f\"{new_namespace}.{key}\"] = value\n            else:\n                new_labels[name] = value\n        return new_labels\n\n    @validator(\"volumes\")\n    def check_volume_format(cls, volumes):\n        for volume in volumes:\n            if \":\" not in volume:\n                raise ValueError(\n                    \"Invalid volume specification. \"\n                    f\"Expected format 'path:container_path', but got {volume!r}\"\n                )\n\n        return volumes\n\n    @sync_compatible\n    async def run(\n        self,\n        task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> Optional[bool]:\n        if not self.command:\n            raise ValueError(\"Docker container cannot be run with empty command.\")\n\n        # The `docker` library uses requests instead of an async http library so it must\n        # be run in a thread to avoid blocking the event loop.\n        container = await run_sync_in_worker_thread(self._create_and_start_container)\n        container_pid = self._get_infrastructure_pid(container_id=container.id)\n\n        # Mark as started and return the infrastructure id\n        if task_status:\n            task_status.started(container_pid)\n\n        # Monitor the container\n        container = await run_sync_in_worker_thread(\n            self._watch_container_safe, container\n        )\n\n        exit_code = container.attrs[\"State\"].get(\"ExitCode\")\n        return DockerContainerResult(\n            status_code=exit_code if exit_code is not None else -1,\n            identifier=container_pid,\n        )\n\n    async def kill(self, infrastructure_pid: str, grace_seconds: int = 30):\n        docker_client = self._get_client()\n        base_url, container_id = self._parse_infrastructure_pid(infrastructure_pid)\n\n        if docker_client.api.base_url != base_url:\n            raise InfrastructureNotAvailable(\n                \"\".join(\n                    [\n                        (\n                            f\"Unable to stop container {container_id!r}: the current\"\n                            \" Docker API \"\n                        ),\n                        (\n                            f\"URL {docker_client.api.base_url!r} does not match the\"\n                            \" expected \"\n                        ),\n                        f\"API base URL {base_url}.\",\n                    ]\n                )\n            )\n        try:\n            container = docker_client.containers.get(container_id=container_id)\n        except docker.errors.NotFound:\n            raise InfrastructureNotFound(\n                f\"Unable to stop container {container_id!r}: The container was not\"\n                \" found.\"\n            )\n\n        try:\n            container.stop(timeout=grace_seconds)\n        except Exception:\n            raise\n\n    def preview(self):\n        # TODO: build and document a more sophisticated preview\n        docker_client = self._get_client()\n        try:\n            return json.dumps(self._build_container_settings(docker_client))\n        finally:\n            docker_client.close()\n\n    async def generate_work_pool_base_job_template(self):\n        from prefect.workers.utilities import (\n            get_default_base_job_template_for_infrastructure_type,\n        )\n\n        base_job_template = await get_default_base_job_template_for_infrastructure_type(\n            self.get_corresponding_worker_type()\n        )\n        if base_job_template is None:\n            return await super().generate_work_pool_base_job_template()\n        for key, value in self.dict(exclude_unset=True, exclude_defaults=True).items():\n            if key == \"command\":\n                base_job_template[\"variables\"][\"properties\"][\"command\"][\"default\"] = (\n                    shlex.join(value)\n                )\n            elif key == \"image_registry\":\n                self.logger.warning(\n                    \"Image registry blocks are not supported by Docker\"\n                    \" work pools. Please authenticate to your registry using\"\n                    \" the `docker login` command on your worker instances.\"\n                )\n            elif key in [\n                \"type\",\n                \"block_type_slug\",\n                \"_block_document_id\",\n                \"_block_document_name\",\n                \"_is_anonymous\",\n            ]:\n                continue\n            elif key == \"image_pull_policy\":\n                new_value = None\n                if value == ImagePullPolicy.ALWAYS:\n                    new_value = \"Always\"\n                elif value == ImagePullPolicy.NEVER:\n                    new_value = \"Never\"\n                elif value == ImagePullPolicy.IF_NOT_PRESENT:\n                    new_value = \"IfNotPresent\"\n\n                base_job_template[\"variables\"][\"properties\"][key][\"default\"] = new_value\n            elif key in base_job_template[\"variables\"][\"properties\"]:\n                base_job_template[\"variables\"][\"properties\"][key][\"default\"] = value\n            else:\n                self.logger.warning(\n                    f\"Variable {key!r} is not supported by Docker work pools. Skipping.\"\n                )\n\n        return base_job_template\n\n    def get_corresponding_worker_type(self):\n        return \"docker\"\n\n    def _get_infrastructure_pid(self, container_id: str) -> str:\n        \"\"\"Generates a Docker infrastructure_pid string in the form of\n        `<docker_host_base_url>:<container_id>`.\n        \"\"\"\n        docker_client = self._get_client()\n        base_url = docker_client.api.base_url\n        docker_client.close()\n        return f\"{base_url}:{container_id}\"\n\n    def _parse_infrastructure_pid(self, infrastructure_pid: str) -> Tuple[str, str]:\n        \"\"\"Splits a Docker infrastructure_pid into its component parts\"\"\"\n\n        # base_url can contain `:` so we only want the last item of the split\n        base_url, container_id = infrastructure_pid.rsplit(\":\", 1)\n        return base_url, str(container_id)\n\n    def _build_container_settings(\n        self,\n        docker_client: \"DockerClient\",\n    ) -> Dict:\n        network_mode = self._get_network_mode()\n        return dict(\n            image=self.image,\n            network=self.networks[0] if self.networks else None,\n            network_mode=network_mode,\n            command=self.command,\n            environment=self._get_environment_variables(network_mode),\n            auto_remove=self.auto_remove,\n            labels={**CONTAINER_LABELS, **self.labels},\n            extra_hosts=self._get_extra_hosts(docker_client),\n            name=self._get_container_name(),\n            volumes=self.volumes,\n            mem_limit=self.mem_limit,\n            memswap_limit=self.memswap_limit,\n            privileged=self.privileged,\n        )\n\n    def _create_and_start_container(self) -> \"Container\":\n        if self.image_registry:\n            # If an image registry block was supplied, load an authenticated Docker\n            # client from the block. Otherwise, use an unauthenticated client to\n            # pull images from public registries.\n            docker_client = self.image_registry.get_docker_client()\n        else:\n            docker_client = self._get_client()\n        container_settings = self._build_container_settings(docker_client)\n\n        if self._should_pull_image(docker_client):\n            self.logger.info(f\"Pulling image {self.image!r}...\")\n            self._pull_image(docker_client)\n\n        container = self._create_container(docker_client, **container_settings)\n\n        # Add additional networks after the container is created; only one network can\n        # be attached at creation time\n        if len(self.networks) > 1:\n            for network_name in self.networks[1:]:\n                network = docker_client.networks.get(network_name)\n                network.connect(container)\n\n        # Start the container\n        container.start()\n\n        docker_client.close()\n\n        return container\n\n    def _get_image_and_tag(self) -> Tuple[str, Optional[str]]:\n        return parse_image_tag(self.image)\n\n    def _determine_image_pull_policy(self) -> ImagePullPolicy:\n        \"\"\"\n        Determine the appropriate image pull policy.\n\n        1. If they specified an image pull policy, use that.\n\n        2. If they did not specify an image pull policy and gave us\n           the \"latest\" tag, use ImagePullPolicy.always.\n\n        3. If they did not specify an image pull policy and did not\n           specify a tag, use ImagePullPolicy.always.\n\n        4. If they did not specify an image pull policy and gave us\n           a tag other than \"latest\", use ImagePullPolicy.if_not_present.\n\n        This logic matches the behavior of Kubernetes.\n        See:https://kubernetes.io/docs/concepts/containers/images/#imagepullpolicy-defaulting\n        \"\"\"\n        if not self.image_pull_policy:\n            _, tag = self._get_image_and_tag()\n            if tag == \"latest\" or not tag:\n                return ImagePullPolicy.ALWAYS\n            return ImagePullPolicy.IF_NOT_PRESENT\n        return self.image_pull_policy\n\n    def _get_network_mode(self) -> Optional[str]:\n        # User's value takes precedence; this may collide with the incompatible options\n        # mentioned below.\n        if self.network_mode:\n            if sys.platform != \"linux\" and self.network_mode == \"host\":\n                warnings.warn(\n                    f\"{self.network_mode!r} network mode is not supported on platform \"\n                    f\"{sys.platform!r} and may not work as intended.\"\n                )\n            return self.network_mode\n\n        # Network mode is not compatible with networks or ports (we do not support ports\n        # yet though)\n        if self.networks:\n            return None\n\n        # Check for a local API connection\n        api_url = self.env.get(\"PREFECT_API_URL\", PREFECT_API_URL.value())\n\n        if api_url:\n            try:\n                _, netloc, _, _, _, _ = urllib.parse.urlparse(api_url)\n            except Exception as exc:\n                warnings.warn(\n                    f\"Failed to parse host from API URL {api_url!r} with exception: \"\n                    f\"{exc}\\nThe network mode will not be inferred.\"\n                )\n                return None\n\n            host = netloc.split(\":\")[0]\n\n            # If using a locally hosted API, use a host network on linux\n            if sys.platform == \"linux\" and (host == \"127.0.0.1\" or host == \"localhost\"):\n                return \"host\"\n\n        # Default to unset\n        return None\n\n    def _should_pull_image(self, docker_client: \"DockerClient\") -> bool:\n        \"\"\"\n        Decide whether we need to pull the Docker image.\n        \"\"\"\n        image_pull_policy = self._determine_image_pull_policy()\n\n        if image_pull_policy is ImagePullPolicy.ALWAYS:\n            return True\n        elif image_pull_policy is ImagePullPolicy.NEVER:\n            return False\n        elif image_pull_policy is ImagePullPolicy.IF_NOT_PRESENT:\n            try:\n                # NOTE: images.get() wants the tag included with the image\n                # name, while images.pull() wants them split.\n                docker_client.images.get(self.image)\n            except docker.errors.ImageNotFound:\n                self.logger.debug(f\"Could not find Docker image locally: {self.image}\")\n                return True\n        return False\n\n    def _pull_image(self, docker_client: \"DockerClient\"):\n        \"\"\"\n        Pull the image we're going to use to create the container.\n        \"\"\"\n        image, tag = self._get_image_and_tag()\n\n        return docker_client.images.pull(image, tag)\n\n    def _create_container(self, docker_client: \"DockerClient\", **kwargs) -> \"Container\":\n        \"\"\"\n        Create a docker container with retries on name conflicts.\n\n        If the container already exists with the given name, an incremented index is\n        added.\n        \"\"\"\n        # Create the container with retries on name conflicts (with an incremented idx)\n        index = 0\n        container = None\n        name = original_name = kwargs.pop(\"name\")\n\n        while not container:\n            from docker.errors import APIError\n\n            try:\n                display_name = repr(name) if name else \"with auto-generated name\"\n                self.logger.info(f\"Creating Docker container {display_name}...\")\n                container = docker_client.containers.create(name=name, **kwargs)\n            except APIError as exc:\n                if \"Conflict\" in str(exc) and \"container name\" in str(exc):\n                    self.logger.info(\n                        f\"Docker container name {display_name} already exists; \"\n                        \"retrying...\"\n                    )\n                    index += 1\n                    name = f\"{original_name}-{index}\"\n                else:\n                    raise\n\n        self.logger.info(\n            f\"Docker container {container.name!r} has status {container.status!r}\"\n        )\n        return container\n\n    def _watch_container_safe(self, container: \"Container\") -> \"Container\":\n        # Monitor the container capturing the latest snapshot while capturing\n        # not found errors\n        docker_client = self._get_client()\n\n        try:\n            for latest_container in self._watch_container(docker_client, container.id):\n                container = latest_container\n        except docker.errors.NotFound:\n            # The container was removed during watching\n            self.logger.warning(\n                f\"Docker container {container.name} was removed before we could wait \"\n                \"for its completion.\"\n            )\n        finally:\n            docker_client.close()\n\n        return container\n\n    def _watch_container(\n        self, docker_client: \"DockerClient\", container_id: str\n    ) -> Generator[None, None, \"Container\"]:\n        container: \"Container\" = docker_client.containers.get(container_id)\n\n        status = container.status\n        self.logger.info(\n            f\"Docker container {container.name!r} has status {container.status!r}\"\n        )\n        yield container\n\n        if self.stream_output:\n            try:\n                for log in container.logs(stream=True):\n                    log: bytes\n                    print(log.decode().rstrip())\n            except docker.errors.APIError as exc:\n                if \"marked for removal\" in str(exc):\n                    self.logger.warning(\n                        f\"Docker container {container.name} was marked for removal\"\n                        \" before logs could be retrieved. Output will not be\"\n                        \" streamed. \"\n                    )\n                else:\n                    self.logger.exception(\n                        \"An unexpected Docker API error occurred while streaming\"\n                        f\" output from container {container.name}.\"\n                    )\n\n            container.reload()\n            if container.status != status:\n                self.logger.info(\n                    f\"Docker container {container.name!r} has status\"\n                    f\" {container.status!r}\"\n                )\n            yield container\n\n        container.wait()\n        self.logger.info(\n            f\"Docker container {container.name!r} has status {container.status!r}\"\n        )\n        yield container\n\n    def _get_client(self):\n        try:\n            with warnings.catch_warnings():\n                # Silence warnings due to use of deprecated methods within dockerpy\n                # See https://github.com/docker/docker-py/pull/2931\n                warnings.filterwarnings(\n                    \"ignore\",\n                    message=\"distutils Version classes are deprecated.*\",\n                    category=DeprecationWarning,\n                )\n\n                docker_client = docker.from_env()\n\n        except docker.errors.DockerException as exc:\n            raise RuntimeError(\"Could not connect to Docker.\") from exc\n\n        return docker_client\n\n    def _get_container_name(self) -> Optional[str]:\n        \"\"\"\n        Generates a container name to match the configured name, ensuring it is Docker\n        compatible.\n        \"\"\"\n        # Must match `/?[a-zA-Z0-9][a-zA-Z0-9_.-]+` in the end\n        if not self.name:\n            return None\n\n        return (\n            slugify(\n                self.name,\n                lowercase=False,\n                # Docker does not limit length but URL limits apply eventually so\n                # limit the length for safety\n                max_length=250,\n                # Docker allows these characters for container names\n                regex_pattern=r\"[^a-zA-Z0-9_.-]+\",\n            ).lstrip(\n                # Docker does not allow leading underscore, dash, or period\n                \"_-.\"\n            )\n            # Docker does not allow 0 character names so cast to null if the name is\n            # empty after slufification\n            or None\n        )\n\n    def _get_extra_hosts(self, docker_client) -> Dict[str, str]:\n        \"\"\"\n        A host.docker.internal -> host-gateway mapping is necessary for communicating\n        with the API on Linux machines. Docker Desktop on macOS will automatically\n        already have this mapping.\n        \"\"\"\n        if sys.platform == \"linux\" and (\n            # Do not warn if the user has specified a host manually that does not use\n            # a local address\n            \"PREFECT_API_URL\" not in self.env\n            or re.search(\n                \".*(localhost)|(127.0.0.1)|(host.docker.internal).*\",\n                self.env[\"PREFECT_API_URL\"],\n            )\n        ):\n            user_version = packaging.version.parse(\n                format_outlier_version_name(docker_client.version()[\"Version\"])\n            )\n            required_version = packaging.version.parse(\"20.10.0\")\n\n            if user_version < required_version:\n                warnings.warn(\n                    \"`host.docker.internal` could not be automatically resolved to\"\n                    \" your local ip address. This feature is not supported on Docker\"\n                    f\" Engine v{user_version}, upgrade to v{required_version}+ if you\"\n                    \" encounter issues.\"\n                )\n                return {}\n            else:\n                # Compatibility for linux -- https://github.com/docker/cli/issues/2290\n                # Only supported by Docker v20.10.0+ which is our minimum recommend version\n                return {\"host.docker.internal\": \"host-gateway\"}\n\n    def _get_environment_variables(self, network_mode):\n        # If the API URL has been set by the base environment rather than the by the\n        # user, update the value to ensure connectivity when using a bridge network by\n        # updating local connections to use the docker internal host unless the\n        # network mode is \"host\" where localhost is available already.\n        env = {**self._base_environment(), **self.env}\n\n        if (\n            \"PREFECT_API_URL\" in env\n            and \"PREFECT_API_URL\" not in self.env\n            and network_mode != \"host\"\n        ):\n            env[\"PREFECT_API_URL\"] = (\n                env[\"PREFECT_API_URL\"]\n                .replace(\"localhost\", \"host.docker.internal\")\n                .replace(\"127.0.0.1\", \"host.docker.internal\")\n            )\n\n        # Drop null values allowing users to \"unset\" variables\n        return {key: value for key, value in env.items() if value is not None}\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.DockerContainerResult","title":"DockerContainerResult","text":"

    Bases: InfrastructureResult

    Contains information about a completed Docker container

    Source code in prefect/infrastructure/container.py
    class DockerContainerResult(InfrastructureResult):\n    \"\"\"Contains information about a completed Docker container\"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure","title":"Infrastructure","text":"

    Bases: Block, ABC

    Source code in prefect/infrastructure/base.py
    class Infrastructure(Block, abc.ABC):\n    _block_schema_capabilities = [\"run-infrastructure\"]\n\n    type: str\n\n    env: Dict[str, Optional[str]] = pydantic.Field(\n        default_factory=dict,\n        title=\"Environment\",\n        description=\"Environment variables to set in the configured infrastructure.\",\n    )\n    labels: Dict[str, str] = pydantic.Field(\n        default_factory=dict,\n        description=\"Labels applied to the infrastructure for metadata purposes.\",\n    )\n    name: Optional[str] = pydantic.Field(\n        default=None,\n        description=\"Name applied to the infrastructure for identification.\",\n    )\n    command: Optional[List[str]] = pydantic.Field(\n        default=None,\n        description=\"The command to run in the infrastructure.\",\n    )\n\n    async def generate_work_pool_base_job_template(self):\n        if self._block_document_id is None:\n            raise BlockNotSavedError(\n                \"Cannot publish as work pool, block has not been saved. Please call\"\n                \" `.save()` on your block before publishing.\"\n            )\n\n        block_schema = self.__class__.schema()\n        return {\n            \"job_configuration\": {\"block\": \"{{ block }}\"},\n            \"variables\": {\n                \"type\": \"object\",\n                \"properties\": {\n                    \"block\": {\n                        \"title\": \"Block\",\n                        \"description\": (\n                            \"The infrastructure block to use for job creation.\"\n                        ),\n                        \"allOf\": [{\"$ref\": f\"#/definitions/{self.__class__.__name__}\"}],\n                        \"default\": {\n                            \"$ref\": {\"block_document_id\": str(self._block_document_id)}\n                        },\n                    }\n                },\n                \"required\": [\"block\"],\n                \"definitions\": {self.__class__.__name__: block_schema},\n            },\n        }\n\n    def get_corresponding_worker_type(self):\n        return \"block\"\n\n    @sync_compatible\n    async def publish_as_work_pool(self, work_pool_name: Optional[str] = None):\n        \"\"\"\n        Creates a work pool configured to use the given block as the job creator.\n\n        Used to migrate from a agents setup to a worker setup.\n\n        Args:\n            work_pool_name: The name to give to the created work pool. If not provided, the name of the current\n                block will be used.\n        \"\"\"\n\n        base_job_template = await self.generate_work_pool_base_job_template()\n        work_pool_name = work_pool_name or self._block_document_name\n\n        if work_pool_name is None:\n            raise ValueError(\n                \"`work_pool_name` must be provided if the block has not been saved.\"\n            )\n\n        console = Console()\n\n        try:\n            async with prefect.get_client() as client:\n                work_pool = await client.create_work_pool(\n                    work_pool=WorkPoolCreate(\n                        name=work_pool_name,\n                        type=self.get_corresponding_worker_type(),\n                        base_job_template=base_job_template,\n                    )\n                )\n        except ObjectAlreadyExists:\n            console.print(\n                (\n                    f\"Work pool with name {work_pool_name!r} already exists, please use\"\n                    \" a different name.\"\n                ),\n                style=\"red\",\n            )\n            return\n\n        console.print(\n            f\"Work pool {work_pool.name} created!\",\n            style=\"green\",\n        )\n        if PREFECT_UI_URL:\n            console.print(\n                \"You see your new work pool in the UI at\"\n                f\" {PREFECT_UI_URL.value()}/work-pools/work-pool/{work_pool.name}\"\n            )\n\n        deploy_script = (\n            \"my_flow.deploy(work_pool_name='{work_pool.name}', image='my_image:tag')\"\n        )\n        if not hasattr(self, \"image\"):\n            deploy_script = (\n                \"my_flow.from_source(source='https://github.com/org/repo.git',\"\n                f\" entrypoint='flow.py:my_flow').deploy(work_pool_name='{work_pool.name}')\"\n            )\n        console.print(\n            \"\\nYou can deploy a flow to this work pool by calling\"\n            f\" [blue].deploy[/]:\\n\\n\\t{deploy_script}\\n\"\n        )\n        console.print(\n            \"\\nTo start a worker to execute flow runs in this work pool run:\\n\"\n        )\n        console.print(f\"\\t[blue]prefect worker start --pool {work_pool.name}[/]\\n\")\n\n    @abc.abstractmethod\n    async def run(\n        self,\n        task_status: anyio.abc.TaskStatus = None,\n    ) -> InfrastructureResult:\n        \"\"\"\n        Run the infrastructure.\n\n        If provided a `task_status`, the status will be reported as started when the\n        infrastructure is successfully created. The status return value will be an\n        identifier for the infrastructure.\n\n        The call will then monitor the created infrastructure, returning a result at\n        the end containing a status code indicating if the infrastructure exited cleanly\n        or encountered an error.\n        \"\"\"\n        # Note: implementations should include `sync_compatible`\n\n    @abc.abstractmethod\n    def preview(self) -> str:\n        \"\"\"\n        View a preview of the infrastructure that would be run.\n        \"\"\"\n\n    @property\n    def logger(self):\n        return get_logger(f\"prefect.infrastructure.{self.type}\")\n\n    @property\n    def is_using_a_runner(self):\n        return self.command is not None and \"prefect flow-run execute\" in shlex.join(\n            self.command\n        )\n\n    @classmethod\n    def _base_environment(cls) -> Dict[str, str]:\n        \"\"\"\n        Environment variables that should be passed to all created infrastructure.\n\n        These values should be overridable with the `env` field.\n        \"\"\"\n        return get_current_settings().to_environment_variables(exclude_unset=True)\n\n    def prepare_for_flow_run(\n        self: Self,\n        flow_run: \"FlowRun\",\n        deployment: Optional[\"Deployment\"] = None,\n        flow: Optional[\"Flow\"] = None,\n    ) -> Self:\n        \"\"\"\n        Return an infrastructure block that is prepared to execute a flow run.\n        \"\"\"\n        if deployment is not None:\n            deployment_labels = self._base_deployment_labels(deployment)\n        else:\n            deployment_labels = {}\n\n        if flow is not None:\n            flow_labels = self._base_flow_labels(flow)\n        else:\n            flow_labels = {}\n\n        return self.copy(\n            update={\n                \"env\": {**self._base_flow_run_environment(flow_run), **self.env},\n                \"labels\": {\n                    **self._base_flow_run_labels(flow_run),\n                    **deployment_labels,\n                    **flow_labels,\n                    **self.labels,\n                },\n                \"name\": self.name or flow_run.name,\n                \"command\": self.command or self._base_flow_run_command(),\n            }\n        )\n\n    @staticmethod\n    def _base_flow_run_command() -> List[str]:\n        \"\"\"\n        Generate a command for a flow run job.\n        \"\"\"\n        if experiment_enabled(\"enhanced_cancellation\"):\n            if (\n                PREFECT_EXPERIMENTAL_WARN\n                and PREFECT_EXPERIMENTAL_WARN_ENHANCED_CANCELLATION\n            ):\n                warnings.warn(\n                    EXPERIMENTAL_WARNING.format(\n                        feature=\"Enhanced flow run cancellation\",\n                        group=\"enhanced_cancellation\",\n                        help=\"\",\n                    ),\n                    ExperimentalFeature,\n                    stacklevel=3,\n                )\n            return [\"prefect\", \"flow-run\", \"execute\"]\n\n        return [\"python\", \"-m\", \"prefect.engine\"]\n\n    @staticmethod\n    def _base_flow_run_labels(flow_run: \"FlowRun\") -> Dict[str, str]:\n        \"\"\"\n        Generate a dictionary of labels for a flow run job.\n        \"\"\"\n        return {\n            \"prefect.io/flow-run-id\": str(flow_run.id),\n            \"prefect.io/flow-run-name\": flow_run.name,\n            \"prefect.io/version\": prefect.__version__,\n        }\n\n    @staticmethod\n    def _base_flow_run_environment(flow_run: \"FlowRun\") -> Dict[str, str]:\n        \"\"\"\n        Generate a dictionary of environment variables for a flow run job.\n        \"\"\"\n        environment = {}\n        environment[\"PREFECT__FLOW_RUN_ID\"] = str(flow_run.id)\n        return environment\n\n    @staticmethod\n    def _base_deployment_labels(deployment: \"Deployment\") -> Dict[str, str]:\n        labels = {\n            \"prefect.io/deployment-name\": deployment.name,\n        }\n        if deployment.updated is not None:\n            labels[\"prefect.io/deployment-updated\"] = deployment.updated.in_timezone(\n                \"utc\"\n            ).to_iso8601_string()\n        return labels\n\n    @staticmethod\n    def _base_flow_labels(flow: \"Flow\") -> Dict[str, str]:\n        return {\n            \"prefect.io/flow-name\": flow.name,\n        }\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure.prepare_for_flow_run","title":"prepare_for_flow_run","text":"

    Return an infrastructure block that is prepared to execute a flow run.

    Source code in prefect/infrastructure/base.py
    def prepare_for_flow_run(\n    self: Self,\n    flow_run: \"FlowRun\",\n    deployment: Optional[\"Deployment\"] = None,\n    flow: Optional[\"Flow\"] = None,\n) -> Self:\n    \"\"\"\n    Return an infrastructure block that is prepared to execute a flow run.\n    \"\"\"\n    if deployment is not None:\n        deployment_labels = self._base_deployment_labels(deployment)\n    else:\n        deployment_labels = {}\n\n    if flow is not None:\n        flow_labels = self._base_flow_labels(flow)\n    else:\n        flow_labels = {}\n\n    return self.copy(\n        update={\n            \"env\": {**self._base_flow_run_environment(flow_run), **self.env},\n            \"labels\": {\n                **self._base_flow_run_labels(flow_run),\n                **deployment_labels,\n                **flow_labels,\n                **self.labels,\n            },\n            \"name\": self.name or flow_run.name,\n            \"command\": self.command or self._base_flow_run_command(),\n        }\n    )\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure.preview","title":"preview abstractmethod","text":"

    View a preview of the infrastructure that would be run.

    Source code in prefect/infrastructure/base.py
    @abc.abstractmethod\ndef preview(self) -> str:\n    \"\"\"\n    View a preview of the infrastructure that would be run.\n    \"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure.publish_as_work_pool","title":"publish_as_work_pool async","text":"

    Creates a work pool configured to use the given block as the job creator.

    Used to migrate from a agents setup to a worker setup.

    Parameters:

    Name Type Description Default work_pool_name Optional[str]

    The name to give to the created work pool. If not provided, the name of the current block will be used.

    None Source code in prefect/infrastructure/base.py
    @sync_compatible\nasync def publish_as_work_pool(self, work_pool_name: Optional[str] = None):\n    \"\"\"\n    Creates a work pool configured to use the given block as the job creator.\n\n    Used to migrate from a agents setup to a worker setup.\n\n    Args:\n        work_pool_name: The name to give to the created work pool. If not provided, the name of the current\n            block will be used.\n    \"\"\"\n\n    base_job_template = await self.generate_work_pool_base_job_template()\n    work_pool_name = work_pool_name or self._block_document_name\n\n    if work_pool_name is None:\n        raise ValueError(\n            \"`work_pool_name` must be provided if the block has not been saved.\"\n        )\n\n    console = Console()\n\n    try:\n        async with prefect.get_client() as client:\n            work_pool = await client.create_work_pool(\n                work_pool=WorkPoolCreate(\n                    name=work_pool_name,\n                    type=self.get_corresponding_worker_type(),\n                    base_job_template=base_job_template,\n                )\n            )\n    except ObjectAlreadyExists:\n        console.print(\n            (\n                f\"Work pool with name {work_pool_name!r} already exists, please use\"\n                \" a different name.\"\n            ),\n            style=\"red\",\n        )\n        return\n\n    console.print(\n        f\"Work pool {work_pool.name} created!\",\n        style=\"green\",\n    )\n    if PREFECT_UI_URL:\n        console.print(\n            \"You see your new work pool in the UI at\"\n            f\" {PREFECT_UI_URL.value()}/work-pools/work-pool/{work_pool.name}\"\n        )\n\n    deploy_script = (\n        \"my_flow.deploy(work_pool_name='{work_pool.name}', image='my_image:tag')\"\n    )\n    if not hasattr(self, \"image\"):\n        deploy_script = (\n            \"my_flow.from_source(source='https://github.com/org/repo.git',\"\n            f\" entrypoint='flow.py:my_flow').deploy(work_pool_name='{work_pool.name}')\"\n        )\n    console.print(\n        \"\\nYou can deploy a flow to this work pool by calling\"\n        f\" [blue].deploy[/]:\\n\\n\\t{deploy_script}\\n\"\n    )\n    console.print(\n        \"\\nTo start a worker to execute flow runs in this work pool run:\\n\"\n    )\n    console.print(f\"\\t[blue]prefect worker start --pool {work_pool.name}[/]\\n\")\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Infrastructure.run","title":"run abstractmethod async","text":"

    Run the infrastructure.

    If provided a task_status, the status will be reported as started when the infrastructure is successfully created. The status return value will be an identifier for the infrastructure.

    The call will then monitor the created infrastructure, returning a result at the end containing a status code indicating if the infrastructure exited cleanly or encountered an error.

    Source code in prefect/infrastructure/base.py
    @abc.abstractmethod\nasync def run(\n    self,\n    task_status: anyio.abc.TaskStatus = None,\n) -> InfrastructureResult:\n    \"\"\"\n    Run the infrastructure.\n\n    If provided a `task_status`, the status will be reported as started when the\n    infrastructure is successfully created. The status return value will be an\n    identifier for the infrastructure.\n\n    The call will then monitor the created infrastructure, returning a result at\n    the end containing a status code indicating if the infrastructure exited cleanly\n    or encountered an error.\n    \"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesClusterConfig","title":"KubernetesClusterConfig","text":"

    Bases: Block

    Stores configuration for interaction with Kubernetes clusters.

    See from_file for creation.

    Attributes:

    Name Type Description config Dict

    The entire loaded YAML contents of a kubectl config file

    context_name str

    The name of the kubectl context to use

    Example

    Load a saved Kubernetes cluster config:

    from prefect.blocks.kubernetes import KubernetesClusterConfig\n\ncluster_config_block = KubernetesClusterConfig.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/kubernetes.py
    class KubernetesClusterConfig(Block):\n    \"\"\"\n    Stores configuration for interaction with Kubernetes clusters.\n\n    See `from_file` for creation.\n\n    Attributes:\n        config: The entire loaded YAML contents of a kubectl config file\n        context_name: The name of the kubectl context to use\n\n    Example:\n        Load a saved Kubernetes cluster config:\n        ```python\n        from prefect.blocks.kubernetes import KubernetesClusterConfig\n\n        cluster_config_block = KubernetesClusterConfig.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Kubernetes Cluster Config\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/2d0b896006ad463b49c28aaac14f31e00e32cfab-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig\"\n\n    config: Dict = Field(\n        default=..., description=\"The entire contents of a kubectl config file.\"\n    )\n    context_name: str = Field(\n        default=..., description=\"The name of the kubectl context to use.\"\n    )\n\n    @validator(\"config\", pre=True)\n    def parse_yaml_config(cls, value):\n        if isinstance(value, str):\n            return yaml.safe_load(value)\n        return value\n\n    @classmethod\n    def from_file(cls: Type[Self], path: Path = None, context_name: str = None) -> Self:\n        \"\"\"\n        Create a cluster config from the a Kubernetes config file.\n\n        By default, the current context in the default Kubernetes config file will be\n        used.\n\n        An alternative file or context may be specified.\n\n        The entire config file will be loaded and stored.\n        \"\"\"\n        kube_config = kubernetes.config.kube_config\n\n        path = Path(path or kube_config.KUBE_CONFIG_DEFAULT_LOCATION)\n        path = path.expanduser().resolve()\n\n        # Determine the context\n        existing_contexts, current_context = kube_config.list_kube_config_contexts(\n            config_file=str(path)\n        )\n        context_names = {ctx[\"name\"] for ctx in existing_contexts}\n        if context_name:\n            if context_name not in context_names:\n                raise ValueError(\n                    f\"Context {context_name!r} not found. \"\n                    f\"Specify one of: {listrepr(context_names, sep=', ')}.\"\n                )\n        else:\n            context_name = current_context[\"name\"]\n\n        # Load the entire config file\n        config_file_contents = path.read_text()\n        config_dict = yaml.safe_load(config_file_contents)\n\n        return cls(config=config_dict, context_name=context_name)\n\n    def get_api_client(self) -> \"ApiClient\":\n        \"\"\"\n        Returns a Kubernetes API client for this cluster config.\n        \"\"\"\n        return kubernetes.config.kube_config.new_client_from_config_dict(\n            config_dict=self.config, context=self.context_name\n        )\n\n    def configure_client(self) -> None:\n        \"\"\"\n        Activates this cluster configuration by loading the configuration into the\n        Kubernetes Python client. After calling this, Kubernetes API clients can use\n        this config's context.\n        \"\"\"\n        kubernetes.config.kube_config.load_kube_config_from_dict(\n            config_dict=self.config, context=self.context_name\n        )\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesClusterConfig.configure_client","title":"configure_client","text":"

    Activates this cluster configuration by loading the configuration into the Kubernetes Python client. After calling this, Kubernetes API clients can use this config's context.

    Source code in prefect/blocks/kubernetes.py
    def configure_client(self) -> None:\n    \"\"\"\n    Activates this cluster configuration by loading the configuration into the\n    Kubernetes Python client. After calling this, Kubernetes API clients can use\n    this config's context.\n    \"\"\"\n    kubernetes.config.kube_config.load_kube_config_from_dict(\n        config_dict=self.config, context=self.context_name\n    )\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesClusterConfig.from_file","title":"from_file classmethod","text":"

    Create a cluster config from the a Kubernetes config file.

    By default, the current context in the default Kubernetes config file will be used.

    An alternative file or context may be specified.

    The entire config file will be loaded and stored.

    Source code in prefect/blocks/kubernetes.py
    @classmethod\ndef from_file(cls: Type[Self], path: Path = None, context_name: str = None) -> Self:\n    \"\"\"\n    Create a cluster config from the a Kubernetes config file.\n\n    By default, the current context in the default Kubernetes config file will be\n    used.\n\n    An alternative file or context may be specified.\n\n    The entire config file will be loaded and stored.\n    \"\"\"\n    kube_config = kubernetes.config.kube_config\n\n    path = Path(path or kube_config.KUBE_CONFIG_DEFAULT_LOCATION)\n    path = path.expanduser().resolve()\n\n    # Determine the context\n    existing_contexts, current_context = kube_config.list_kube_config_contexts(\n        config_file=str(path)\n    )\n    context_names = {ctx[\"name\"] for ctx in existing_contexts}\n    if context_name:\n        if context_name not in context_names:\n            raise ValueError(\n                f\"Context {context_name!r} not found. \"\n                f\"Specify one of: {listrepr(context_names, sep=', ')}.\"\n            )\n    else:\n        context_name = current_context[\"name\"]\n\n    # Load the entire config file\n    config_file_contents = path.read_text()\n    config_dict = yaml.safe_load(config_file_contents)\n\n    return cls(config=config_dict, context_name=context_name)\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesClusterConfig.get_api_client","title":"get_api_client","text":"

    Returns a Kubernetes API client for this cluster config.

    Source code in prefect/blocks/kubernetes.py
    def get_api_client(self) -> \"ApiClient\":\n    \"\"\"\n    Returns a Kubernetes API client for this cluster config.\n    \"\"\"\n    return kubernetes.config.kube_config.new_client_from_config_dict(\n        config_dict=self.config, context=self.context_name\n    )\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob","title":"KubernetesJob","text":"

    Bases: Infrastructure

    Runs a command as a Kubernetes Job.

    For a guided tutorial, see How to use Kubernetes with Prefect. For more information, including examples for customizing the resulting manifest, see KubernetesJob infrastructure concepts.

    Attributes:

    Name Type Description cluster_config Optional[KubernetesClusterConfig]

    An optional Kubernetes cluster config to use for this job.

    command Optional[KubernetesClusterConfig]

    A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this.

    customizations JsonPatch

    A list of JSON 6902 patches to apply to the base Job manifest.

    env JsonPatch

    Environment variables to set for the container.

    finished_job_ttl Optional[int]

    The number of seconds to retain jobs after completion. If set, finished jobs will be cleaned up by Kubernetes after the given delay. If None (default), jobs will need to be manually removed.

    image Optional[str]

    An optional string specifying the image reference of a container image to use for the job, for example, docker.io/prefecthq/prefect:2-latest. The behavior is as described in https://kubernetes.io/docs/concepts/containers/images/#image-names. Defaults to the Prefect image.

    image_pull_policy Optional[KubernetesImagePullPolicy]

    The Kubernetes image pull policy to use for job containers.

    job KubernetesManifest

    The base manifest for the Kubernetes Job.

    job_watch_timeout_seconds Optional[int]

    Number of seconds to wait for the job to complete before marking it as crashed. Defaults to None, which means no timeout will be enforced.

    labels Optional[int]

    An optional dictionary of labels to add to the job.

    name Optional[int]

    An optional name for the job.

    namespace Optional[str]

    An optional string signifying the Kubernetes namespace to use.

    pod_watch_timeout_seconds int

    Number of seconds to watch for pod creation before timing out (default 60).

    service_account_name Optional[str]

    An optional string specifying which Kubernetes service account to use.

    stream_output bool

    If set, stream output from the job to local standard output.

    Source code in prefect/infrastructure/kubernetes.py
    class KubernetesJob(Infrastructure):\n    \"\"\"\n    Runs a command as a Kubernetes Job.\n\n    For a guided tutorial, see [How to use Kubernetes with Prefect](https://medium.com/the-prefect-blog/how-to-use-kubernetes-with-prefect-419b2e8b8cb2/).\n    For more information, including examples for customizing the resulting manifest, see [`KubernetesJob` infrastructure concepts](https://docs.prefect.io/concepts/infrastructure/#kubernetesjob).\n\n    Attributes:\n        cluster_config: An optional Kubernetes cluster config to use for this job.\n        command: A list of strings specifying the command to run in the container to\n            start the flow run. In most cases you should not override this.\n        customizations: A list of JSON 6902 patches to apply to the base Job manifest.\n        env: Environment variables to set for the container.\n        finished_job_ttl: The number of seconds to retain jobs after completion. If set, finished jobs will\n            be cleaned up by Kubernetes after the given delay. If None (default), jobs will need to be\n            manually removed.\n        image: An optional string specifying the image reference of a container image\n            to use for the job, for example, docker.io/prefecthq/prefect:2-latest. The\n            behavior is as described in https://kubernetes.io/docs/concepts/containers/images/#image-names.\n            Defaults to the Prefect image.\n        image_pull_policy: The Kubernetes image pull policy to use for job containers.\n        job: The base manifest for the Kubernetes Job.\n        job_watch_timeout_seconds: Number of seconds to wait for the job to complete\n            before marking it as crashed. Defaults to `None`, which means no timeout will be enforced.\n        labels: An optional dictionary of labels to add to the job.\n        name: An optional name for the job.\n        namespace: An optional string signifying the Kubernetes namespace to use.\n        pod_watch_timeout_seconds: Number of seconds to watch for pod creation before timing out (default 60).\n        service_account_name: An optional string specifying which Kubernetes service account to use.\n        stream_output: If set, stream output from the job to local standard output.\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/2d0b896006ad463b49c28aaac14f31e00e32cfab-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob\"\n\n    type: Literal[\"kubernetes-job\"] = Field(\n        default=\"kubernetes-job\", description=\"The type of infrastructure.\"\n    )\n    # shortcuts for the most common user-serviceable settings\n    image: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The image reference of a container image to use for the job, for example,\"\n            \" `docker.io/prefecthq/prefect:2-latest`.The behavior is as described in\"\n            \" the Kubernetes documentation and uses the latest version of Prefect by\"\n            \" default, unless an image is already present in a provided job manifest.\"\n        ),\n    )\n    namespace: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The Kubernetes namespace to use for this job. Defaults to 'default' \"\n            \"unless a namespace is already present in a provided job manifest.\"\n        ),\n    )\n    service_account_name: Optional[str] = Field(\n        default=None, description=\"The Kubernetes service account to use for this job.\"\n    )\n    image_pull_policy: Optional[KubernetesImagePullPolicy] = Field(\n        default=None,\n        description=\"The Kubernetes image pull policy to use for job containers.\",\n    )\n\n    # connection to a cluster\n    cluster_config: Optional[KubernetesClusterConfig] = Field(\n        default=None, description=\"The Kubernetes cluster config to use for this job.\"\n    )\n\n    # settings allowing full customization of the Job\n    job: KubernetesManifest = Field(\n        default_factory=lambda: KubernetesJob.base_job_manifest(),\n        description=\"The base manifest for the Kubernetes Job.\",\n        title=\"Base Job Manifest\",\n    )\n    customizations: JsonPatch = Field(\n        default_factory=lambda: JsonPatch([]),\n        description=\"A list of JSON 6902 patches to apply to the base Job manifest.\",\n    )\n\n    # controls the behavior of execution\n    job_watch_timeout_seconds: Optional[int] = Field(\n        default=None,\n        description=(\n            \"Number of seconds to wait for the job to complete before marking it as\"\n            \" crashed. Defaults to `None`, which means no timeout will be enforced.\"\n        ),\n    )\n    pod_watch_timeout_seconds: int = Field(\n        default=60,\n        description=\"Number of seconds to watch for pod creation before timing out.\",\n    )\n    stream_output: bool = Field(\n        default=True,\n        description=(\n            \"If set, output will be streamed from the job to local standard output.\"\n        ),\n    )\n    finished_job_ttl: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The number of seconds to retain jobs after completion. If set, finished\"\n            \" jobs will be cleaned up by Kubernetes after the given delay. If None\"\n            \" (default), jobs will need to be manually removed.\"\n        ),\n    )\n\n    # internal-use only right now\n    _api_dns_name: Optional[str] = None  # Replaces 'localhost' in API URL\n\n    _block_type_name = \"Kubernetes Job\"\n\n    @validator(\"job\")\n    def ensure_job_includes_all_required_components(cls, value: KubernetesManifest):\n        patch = JsonPatch.from_diff(value, cls.base_job_manifest())\n        missing_paths = sorted([op[\"path\"] for op in patch if op[\"op\"] == \"add\"])\n        if missing_paths:\n            raise ValueError(\n                \"Job is missing required attributes at the following paths: \"\n                f\"{', '.join(missing_paths)}\"\n            )\n        return value\n\n    @validator(\"job\")\n    def ensure_job_has_compatible_values(cls, value: KubernetesManifest):\n        patch = JsonPatch.from_diff(value, cls.base_job_manifest())\n        incompatible = sorted(\n            [\n                f\"{op['path']} must have value {op['value']!r}\"\n                for op in patch\n                if op[\"op\"] == \"replace\"\n            ]\n        )\n        if incompatible:\n            raise ValueError(\n                \"Job has incompatible values for the following attributes: \"\n                f\"{', '.join(incompatible)}\"\n            )\n        return value\n\n    @validator(\"customizations\", pre=True)\n    def cast_customizations_to_a_json_patch(\n        cls, value: Union[List[Dict], JsonPatch, str]\n    ) -> JsonPatch:\n        if isinstance(value, list):\n            return JsonPatch(value)\n        elif isinstance(value, str):\n            try:\n                return JsonPatch(json.loads(value))\n            except json.JSONDecodeError as exc:\n                raise ValueError(\n                    f\"Unable to parse customizations as JSON: {value}. Please make sure\"\n                    \" that the provided value is a valid JSON string.\"\n                ) from exc\n        return value\n\n    @root_validator\n    def default_namespace(cls, values):\n        job = values.get(\"job\")\n\n        namespace = values.get(\"namespace\")\n        job_namespace = job[\"metadata\"].get(\"namespace\") if job else None\n\n        if not namespace and not job_namespace:\n            values[\"namespace\"] = \"default\"\n\n        return values\n\n    @root_validator\n    def default_image(cls, values):\n        job = values.get(\"job\")\n        image = values.get(\"image\")\n        job_image = (\n            job[\"spec\"][\"template\"][\"spec\"][\"containers\"][0].get(\"image\")\n            if job\n            else None\n        )\n\n        if not image and not job_image:\n            values[\"image\"] = get_prefect_image_name()\n\n        return values\n\n    # Support serialization of the 'JsonPatch' type\n    class Config:\n        arbitrary_types_allowed = True\n        json_encoders = {JsonPatch: lambda p: p.patch}\n\n    def dict(self, *args, **kwargs) -> Dict:\n        d = super().dict(*args, **kwargs)\n        d[\"customizations\"] = self.customizations.patch\n        return d\n\n    @classmethod\n    def base_job_manifest(cls) -> KubernetesManifest:\n        \"\"\"Produces the bare minimum allowed Job manifest\"\"\"\n        return {\n            \"apiVersion\": \"batch/v1\",\n            \"kind\": \"Job\",\n            \"metadata\": {\"labels\": {}},\n            \"spec\": {\n                \"template\": {\n                    \"spec\": {\n                        \"parallelism\": 1,\n                        \"completions\": 1,\n                        \"restartPolicy\": \"Never\",\n                        \"containers\": [\n                            {\n                                \"name\": \"prefect-job\",\n                                \"env\": [],\n                            }\n                        ],\n                    }\n                }\n            },\n        }\n\n    # Note that we're using the yaml package to load both YAML and JSON files below.\n    # This works because YAML is a strict superset of JSON:\n    #\n    #   > The YAML 1.23 specification was published in 2009. Its primary focus was\n    #   > making YAML a strict superset of JSON. It also removed many of the problematic\n    #   > implicit typing recommendations.\n    #\n    #   https://yaml.org/spec/1.2.2/#12-yaml-history\n\n    @classmethod\n    def job_from_file(cls, filename: str) -> KubernetesManifest:\n        \"\"\"Load a Kubernetes Job manifest from a YAML or JSON file.\"\"\"\n        with open(filename, \"r\", encoding=\"utf-8\") as f:\n            return yaml.load(f, yaml.SafeLoader)\n\n    @classmethod\n    def customize_from_file(cls, filename: str) -> JsonPatch:\n        \"\"\"Load an RFC 6902 JSON patch from a YAML or JSON file.\"\"\"\n        with open(filename, \"r\", encoding=\"utf-8\") as f:\n            return JsonPatch(yaml.load(f, yaml.SafeLoader))\n\n    @sync_compatible\n    async def run(\n        self,\n        task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> KubernetesJobResult:\n        if not self.command:\n            raise ValueError(\"Kubernetes job cannot be run with empty command.\")\n\n        self._configure_kubernetes_library_client()\n        manifest = self.build_job()\n        job = await run_sync_in_worker_thread(self._create_job, manifest)\n\n        pid = await run_sync_in_worker_thread(self._get_infrastructure_pid, job)\n        # Indicate that the job has started\n        if task_status is not None:\n            task_status.started(pid)\n\n        # Monitor the job until completion\n        status_code = await run_sync_in_worker_thread(\n            self._watch_job, job.metadata.name\n        )\n        return KubernetesJobResult(identifier=pid, status_code=status_code)\n\n    async def kill(self, infrastructure_pid: str, grace_seconds: int = 30):\n        self._configure_kubernetes_library_client()\n        job_cluster_uid, job_namespace, job_name = self._parse_infrastructure_pid(\n            infrastructure_pid\n        )\n\n        if not job_namespace == self.namespace:\n            raise InfrastructureNotAvailable(\n                f\"Unable to kill job {job_name!r}: The job is running in namespace \"\n                f\"{job_namespace!r} but this block is configured to use \"\n                f\"{self.namespace!r}.\"\n            )\n\n        current_cluster_uid = self._get_cluster_uid()\n        if job_cluster_uid != current_cluster_uid:\n            raise InfrastructureNotAvailable(\n                f\"Unable to kill job {job_name!r}: The job is running on another \"\n                \"cluster.\"\n            )\n\n        with self.get_batch_client() as batch_client:\n            try:\n                batch_client.delete_namespaced_job(\n                    name=job_name,\n                    namespace=job_namespace,\n                    grace_period_seconds=grace_seconds,\n                    # Foreground propagation deletes dependent objects before deleting owner objects.\n                    # This ensures that the pods are cleaned up before the job is marked as deleted.\n                    # See: https://kubernetes.io/docs/concepts/architecture/garbage-collection/#foreground-deletion\n                    propagation_policy=\"Foreground\",\n                )\n            except kubernetes.client.exceptions.ApiException as exc:\n                if exc.status == 404:\n                    raise InfrastructureNotFound(\n                        f\"Unable to kill job {job_name!r}: The job was not found.\"\n                    ) from exc\n                else:\n                    raise\n\n    def preview(self):\n        return yaml.dump(self.build_job())\n\n    def get_corresponding_worker_type(self):\n        return \"kubernetes\"\n\n    async def generate_work_pool_base_job_template(self):\n        from prefect.workers.utilities import (\n            get_default_base_job_template_for_infrastructure_type,\n        )\n\n        base_job_template = await get_default_base_job_template_for_infrastructure_type(\n            self.get_corresponding_worker_type()\n        )\n        assert (\n            base_job_template is not None\n        ), \"Failed to retrieve default base job template.\"\n        for key, value in self.dict(exclude_unset=True, exclude_defaults=True).items():\n            if key == \"command\":\n                base_job_template[\"variables\"][\"properties\"][\"command\"][\"default\"] = (\n                    shlex.join(value)\n                )\n            elif key in [\n                \"type\",\n                \"block_type_slug\",\n                \"_block_document_id\",\n                \"_block_document_name\",\n                \"_is_anonymous\",\n                \"job\",\n                \"customizations\",\n            ]:\n                continue\n            elif key == \"image_pull_policy\":\n                base_job_template[\"variables\"][\"properties\"][\"image_pull_policy\"][\n                    \"default\"\n                ] = value.value\n            elif key == \"cluster_config\":\n                base_job_template[\"variables\"][\"properties\"][\"cluster_config\"][\n                    \"default\"\n                ] = {\n                    \"$ref\": {\n                        \"block_document_id\": str(self.cluster_config._block_document_id)\n                    }\n                }\n            elif key in base_job_template[\"variables\"][\"properties\"]:\n                base_job_template[\"variables\"][\"properties\"][key][\"default\"] = value\n            else:\n                self.logger.warning(\n                    f\"Variable {key!r} is not supported by Kubernetes work pools.\"\n                    \" Skipping.\"\n                )\n\n        custom_job_manifest = self.dict(exclude_unset=True, exclude_defaults=True).get(\n            \"job\"\n        )\n        if custom_job_manifest:\n            job_manifest = self.build_job()\n        else:\n            job_manifest = copy.deepcopy(\n                base_job_template[\"job_configuration\"][\"job_manifest\"]\n            )\n            job_manifest = self.customizations.apply(job_manifest)\n        base_job_template[\"job_configuration\"][\"job_manifest\"] = job_manifest\n\n        return base_job_template\n\n    def build_job(self) -> KubernetesManifest:\n        \"\"\"Builds the Kubernetes Job Manifest\"\"\"\n        job_manifest = copy.copy(self.job)\n        job_manifest = self._shortcut_customizations().apply(job_manifest)\n        job_manifest = self.customizations.apply(job_manifest)\n        return job_manifest\n\n    @contextmanager\n    def get_batch_client(self) -> Generator[\"BatchV1Api\", None, None]:\n        with kubernetes.client.ApiClient() as client:\n            try:\n                yield kubernetes.client.BatchV1Api(api_client=client)\n            finally:\n                client.rest_client.pool_manager.clear()\n\n    @contextmanager\n    def get_client(self) -> Generator[\"CoreV1Api\", None, None]:\n        with kubernetes.client.ApiClient() as client:\n            try:\n                yield kubernetes.client.CoreV1Api(api_client=client)\n            finally:\n                client.rest_client.pool_manager.clear()\n\n    def _get_infrastructure_pid(self, job: \"V1Job\") -> str:\n        \"\"\"\n        Generates a Kubernetes infrastructure PID.\n\n        The PID is in the format: \"<cluster uid>:<namespace>:<job name>\".\n        \"\"\"\n        cluster_uid = self._get_cluster_uid()\n        pid = f\"{cluster_uid}:{self.namespace}:{job.metadata.name}\"\n        return pid\n\n    def _parse_infrastructure_pid(\n        self, infrastructure_pid: str\n    ) -> Tuple[str, str, str]:\n        \"\"\"\n        Parse a Kubernetes infrastructure PID into its component parts.\n\n        Returns a cluster UID, namespace, and job name.\n        \"\"\"\n        cluster_uid, namespace, job_name = infrastructure_pid.split(\":\", 2)\n        return cluster_uid, namespace, job_name\n\n    def _get_cluster_uid(self) -> str:\n        \"\"\"\n        Gets a unique id for the current cluster being used.\n\n        There is no real unique identifier for a cluster. However, the `kube-system`\n        namespace is immutable and has a persistence UID that we use instead.\n\n        PREFECT_KUBERNETES_CLUSTER_UID can be set in cases where the `kube-system`\n        namespace cannot be read e.g. when a cluster role cannot be created. If set,\n        this variable will be used and we will not attempt to read the `kube-system`\n        namespace.\n\n        See https://github.com/kubernetes/kubernetes/issues/44954\n        \"\"\"\n        # Default to an environment variable\n        env_cluster_uid = os.environ.get(\"PREFECT_KUBERNETES_CLUSTER_UID\")\n        if env_cluster_uid:\n            return env_cluster_uid\n\n        # Read the UID from the cluster namespace\n        with self.get_client() as client:\n            namespace = client.read_namespace(\"kube-system\")\n        cluster_uid = namespace.metadata.uid\n\n        return cluster_uid\n\n    def _configure_kubernetes_library_client(self) -> None:\n        \"\"\"\n        Set the correct kubernetes client configuration.\n\n        WARNING: This action is not threadsafe and may override the configuration\n                  specified by another `KubernetesJob` instance.\n        \"\"\"\n        # TODO: Investigate returning a configured client so calls on other threads\n        #       will not invalidate the config needed here\n\n        # if a k8s cluster block is provided to the flow runner, use that\n        if self.cluster_config:\n            self.cluster_config.configure_client()\n        else:\n            # If no block specified, try to load Kubernetes configuration within a cluster. If that doesn't\n            # work, try to load the configuration from the local environment, allowing\n            # any further ConfigExceptions to bubble up.\n            try:\n                kubernetes.config.load_incluster_config()\n            except kubernetes.config.ConfigException:\n                kubernetes.config.load_kube_config()\n\n    def _shortcut_customizations(self) -> JsonPatch:\n        \"\"\"Produces the JSON 6902 patch for the most commonly used customizations, like\n        image and namespace, which we offer as top-level parameters (with sensible\n        default values)\"\"\"\n        shortcuts = []\n\n        if self.namespace:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/metadata/namespace\",\n                    \"value\": self.namespace,\n                }\n            )\n\n        if self.image:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/template/spec/containers/0/image\",\n                    \"value\": self.image,\n                }\n            )\n\n        shortcuts += [\n            {\n                \"op\": \"add\",\n                \"path\": (\n                    f\"/metadata/labels/{self._slugify_label_key(key).replace('/', '~1', 1)}\"\n                ),\n                \"value\": self._slugify_label_value(value),\n            }\n            for key, value in self.labels.items()\n        ]\n\n        shortcuts += [\n            {\n                \"op\": \"add\",\n                \"path\": \"/spec/template/spec/containers/0/env/-\",\n                \"value\": {\"name\": key, \"value\": value},\n            }\n            for key, value in self._get_environment_variables().items()\n        ]\n\n        if self.image_pull_policy:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/template/spec/containers/0/imagePullPolicy\",\n                    \"value\": self.image_pull_policy.value,\n                }\n            )\n\n        if self.service_account_name:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/template/spec/serviceAccountName\",\n                    \"value\": self.service_account_name,\n                }\n            )\n\n        if self.finished_job_ttl is not None:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/ttlSecondsAfterFinished\",\n                    \"value\": self.finished_job_ttl,\n                }\n            )\n\n        if self.command:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/spec/template/spec/containers/0/args\",\n                    \"value\": self.command,\n                }\n            )\n\n        if self.name:\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/metadata/generateName\",\n                    \"value\": self._slugify_name(self.name) + \"-\",\n                }\n            )\n        else:\n            # Generate name is required\n            shortcuts.append(\n                {\n                    \"op\": \"add\",\n                    \"path\": \"/metadata/generateName\",\n                    \"value\": (\n                        \"prefect-job-\"\n                        # We generate a name using a hash of the primary job settings\n                        + stable_hash(\n                            *self.command,\n                            *self.env.keys(),\n                            *[v for v in self.env.values() if v is not None],\n                        )\n                        + \"-\"\n                    ),\n                }\n            )\n\n        return JsonPatch(shortcuts)\n\n    def _get_job(self, job_id: str) -> Optional[\"V1Job\"]:\n        with self.get_batch_client() as batch_client:\n            try:\n                job = batch_client.read_namespaced_job(job_id, self.namespace)\n            except kubernetes.client.exceptions.ApiException:\n                self.logger.error(f\"Job {job_id!r} was removed.\", exc_info=True)\n                return None\n            return job\n\n    def _get_job_pod(self, job_name: str) -> \"V1Pod\":\n        \"\"\"Get the first running pod for a job.\"\"\"\n\n        # Wait until we find a running pod for the job\n        # if `pod_watch_timeout_seconds` is None, no timeout will be enforced\n        watch = kubernetes.watch.Watch()\n        self.logger.debug(f\"Job {job_name!r}: Starting watch for pod start...\")\n        last_phase = None\n        with self.get_client() as client:\n            for event in watch.stream(\n                func=client.list_namespaced_pod,\n                namespace=self.namespace,\n                label_selector=f\"job-name={job_name}\",\n                timeout_seconds=self.pod_watch_timeout_seconds,\n            ):\n                phase = event[\"object\"].status.phase\n                if phase != last_phase:\n                    self.logger.info(f\"Job {job_name!r}: Pod has status {phase!r}.\")\n\n                if phase != \"Pending\":\n                    watch.stop()\n                    return event[\"object\"]\n\n                last_phase = phase\n\n        self.logger.error(f\"Job {job_name!r}: Pod never started.\")\n\n    def _watch_job(self, job_name: str) -> int:\n        \"\"\"\n        Watch a job.\n\n        Return the final status code of the first container.\n        \"\"\"\n        self.logger.debug(f\"Job {job_name!r}: Monitoring job...\")\n\n        job = self._get_job(job_name)\n        if not job:\n            return -1\n\n        pod = self._get_job_pod(job_name)\n        if not pod:\n            return -1\n\n        # Calculate the deadline before streaming output\n        deadline = (\n            (time.monotonic() + self.job_watch_timeout_seconds)\n            if self.job_watch_timeout_seconds is not None\n            else None\n        )\n\n        if self.stream_output:\n            with self.get_client() as client:\n                logs = client.read_namespaced_pod_log(\n                    pod.metadata.name,\n                    self.namespace,\n                    follow=True,\n                    _preload_content=False,\n                    container=\"prefect-job\",\n                )\n                try:\n                    for log in logs.stream():\n                        print(log.decode().rstrip())\n\n                        # Check if we have passed the deadline and should stop streaming\n                        # logs\n                        remaining_time = (\n                            deadline - time.monotonic() if deadline else None\n                        )\n                        if deadline and remaining_time <= 0:\n                            break\n\n                except Exception:\n                    self.logger.warning(\n                        (\n                            \"Error occurred while streaming logs - \"\n                            \"Job will continue to run but logs will \"\n                            \"no longer be streamed to stdout.\"\n                        ),\n                        exc_info=True,\n                    )\n\n        with self.get_batch_client() as batch_client:\n            # Check if the job is completed before beginning a watch\n            job = batch_client.read_namespaced_job(\n                name=job_name, namespace=self.namespace\n            )\n            completed = job.status.completion_time is not None\n\n            while not completed:\n                remaining_time = (\n                    math.ceil(deadline - time.monotonic()) if deadline else None\n                )\n                if deadline and remaining_time <= 0:\n                    self.logger.error(\n                        f\"Job {job_name!r}: Job did not complete within \"\n                        f\"timeout of {self.job_watch_timeout_seconds}s.\"\n                    )\n                    return -1\n\n                watch = kubernetes.watch.Watch()\n                # The kubernetes library will disable retries if the timeout kwarg is\n                # present regardless of the value so we do not pass it unless given\n                # https://github.com/kubernetes-client/python/blob/84f5fea2a3e4b161917aa597bf5e5a1d95e24f5a/kubernetes/base/watch/watch.py#LL160\n                timeout_seconds = (\n                    {\"timeout_seconds\": remaining_time} if deadline else {}\n                )\n\n                for event in watch.stream(\n                    func=batch_client.list_namespaced_job,\n                    field_selector=f\"metadata.name={job_name}\",\n                    namespace=self.namespace,\n                    **timeout_seconds,\n                ):\n                    if event[\"type\"] == \"DELETED\":\n                        self.logger.error(f\"Job {job_name!r}: Job has been deleted.\")\n                        completed = True\n                    elif event[\"object\"].status.completion_time:\n                        if not event[\"object\"].status.succeeded:\n                            # Job failed, exit while loop and return pod exit code\n                            self.logger.error(f\"Job {job_name!r}: Job failed.\")\n                        completed = True\n                    # Check if the job has reached its backoff limit\n                    # and stop watching if it has\n                    elif (\n                        event[\"object\"].spec.backoff_limit is not None\n                        and event[\"object\"].status.failed is not None\n                        and event[\"object\"].status.failed\n                        > event[\"object\"].spec.backoff_limit\n                    ):\n                        self.logger.error(\n                            f\"Job {job_name!r}: Job reached backoff limit.\"\n                        )\n                        completed = True\n                    # If the job has no backoff limit, check if it has failed\n                    # and stop watching if it has\n                    elif (\n                        not event[\"object\"].spec.backoff_limit\n                        and event[\"object\"].status.failed\n                    ):\n                        completed = True\n\n                    if completed:\n                        watch.stop()\n                        break\n\n        with self.get_client() as core_client:\n            # Get all pods for the job\n            pods = core_client.list_namespaced_pod(\n                namespace=self.namespace, label_selector=f\"job-name={job_name}\"\n            )\n            # Get the status for only the most recently used pod\n            pods.items.sort(\n                key=lambda pod: pod.metadata.creation_timestamp, reverse=True\n            )\n            most_recent_pod = pods.items[0] if pods.items else None\n            first_container_status = (\n                most_recent_pod.status.container_statuses[0]\n                if most_recent_pod\n                else None\n            )\n            if not first_container_status:\n                self.logger.error(f\"Job {job_name!r}: No pods found for job.\")\n                return -1\n\n            # In some cases, such as spot instance evictions, the pod will be forcibly\n            # terminated and not report a status correctly.\n            elif (\n                first_container_status.state is None\n                or first_container_status.state.terminated is None\n                or first_container_status.state.terminated.exit_code is None\n            ):\n                self.logger.error(\n                    f\"Could not determine exit code for {job_name!r}.\"\n                    \"Exit code will be reported as -1.\"\n                    \"First container status info did not report an exit code.\"\n                    f\"First container info: {first_container_status}.\"\n                )\n                return -1\n\n        return first_container_status.state.terminated.exit_code\n\n    def _create_job(self, job_manifest: KubernetesManifest) -> \"V1Job\":\n        \"\"\"\n        Given a Kubernetes Job Manifest, create the Job on the configured Kubernetes\n        cluster and return its name.\n        \"\"\"\n        with self.get_batch_client() as batch_client:\n            job = batch_client.create_namespaced_job(self.namespace, job_manifest)\n        return job\n\n    def _slugify_name(self, name: str) -> str:\n        \"\"\"\n        Slugify text for use as a name.\n\n        Keeps only alphanumeric characters and dashes, and caps the length\n        of the slug at 45 chars.\n\n        The 45 character length allows room for the k8s utility\n        \"generateName\" to generate a unique name from the slug while\n        keeping the total length of a name below 63 characters, which is\n        the limit for e.g. label names that follow RFC 1123 (hostnames) and\n        RFC 1035 (domain names).\n\n        Args:\n            name: The name of the job\n\n        Returns:\n            the slugified job name\n        \"\"\"\n        slug = slugify(\n            name,\n            max_length=45,  # Leave enough space for generateName\n            regex_pattern=r\"[^a-zA-Z0-9-]+\",\n        )\n\n        # TODO: Handle the case that the name is an empty string after being\n        # slugified.\n\n        return slug\n\n    def _slugify_label_key(self, key: str) -> str:\n        \"\"\"\n        Slugify text for use as a label key.\n\n        Keys are composed of an optional prefix and name, separated by a slash (/).\n\n        Keeps only alphanumeric characters, dashes, underscores, and periods.\n        Limits the length of the label prefix to 253 characters.\n        Limits the length of the label name to 63 characters.\n\n        See https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#syntax-and-character-set\n\n        Args:\n            key: The label key\n\n        Returns:\n            The slugified label key\n        \"\"\"\n        if \"/\" in key:\n            prefix, name = key.split(\"/\", maxsplit=1)\n        else:\n            prefix = None\n            name = key\n\n        name_slug = (\n            slugify(name, max_length=63, regex_pattern=r\"[^a-zA-Z0-9-_.]+\").strip(\n                \"_-.\"  # Must start or end with alphanumeric characters\n            )\n            or name\n        )\n        # Fallback to the original if we end up with an empty slug, this will allow\n        # Kubernetes to throw the validation error\n\n        if prefix:\n            prefix_slug = (\n                slugify(\n                    prefix,\n                    max_length=253,\n                    regex_pattern=r\"[^a-zA-Z0-9-\\.]+\",\n                ).strip(\n                    \"_-.\"\n                )  # Must start or end with alphanumeric characters\n                or prefix\n            )\n\n            return f\"{prefix_slug}/{name_slug}\"\n\n        return name_slug\n\n    def _slugify_label_value(self, value: str) -> str:\n        \"\"\"\n        Slugify text for use as a label value.\n\n        Keeps only alphanumeric characters, dashes, underscores, and periods.\n        Limits the total length of label text to below 63 characters.\n\n        See https://kubernetes.io/docs/concepts/overview/working-with-objects/labels/#syntax-and-character-set\n\n        Args:\n            value: The text for the label\n\n        Returns:\n            The slugified value\n        \"\"\"\n        slug = (\n            slugify(value, max_length=63, regex_pattern=r\"[^a-zA-Z0-9-_\\.]+\").strip(\n                \"_-.\"  # Must start or end with alphanumeric characters\n            )\n            or value\n        )\n        # Fallback to the original if we end up with an empty slug, this will allow\n        # Kubernetes to throw the validation error\n\n        return slug\n\n    def _get_environment_variables(self):\n        # If the API URL has been set by the base environment rather than the by the\n        # user, update the value to ensure connectivity when using a bridge network by\n        # updating local connections to use the internal host\n        env = {**self._base_environment(), **self.env}\n\n        if (\n            \"PREFECT_API_URL\" in env\n            and \"PREFECT_API_URL\" not in self.env\n            and self._api_dns_name\n        ):\n            env[\"PREFECT_API_URL\"] = (\n                env[\"PREFECT_API_URL\"]\n                .replace(\"localhost\", self._api_dns_name)\n                .replace(\"127.0.0.1\", self._api_dns_name)\n            )\n\n        # Drop null values allowing users to \"unset\" variables\n        return {key: value for key, value in env.items() if value is not None}\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob.base_job_manifest","title":"base_job_manifest classmethod","text":"

    Produces the bare minimum allowed Job manifest

    Source code in prefect/infrastructure/kubernetes.py
    @classmethod\ndef base_job_manifest(cls) -> KubernetesManifest:\n    \"\"\"Produces the bare minimum allowed Job manifest\"\"\"\n    return {\n        \"apiVersion\": \"batch/v1\",\n        \"kind\": \"Job\",\n        \"metadata\": {\"labels\": {}},\n        \"spec\": {\n            \"template\": {\n                \"spec\": {\n                    \"parallelism\": 1,\n                    \"completions\": 1,\n                    \"restartPolicy\": \"Never\",\n                    \"containers\": [\n                        {\n                            \"name\": \"prefect-job\",\n                            \"env\": [],\n                        }\n                    ],\n                }\n            }\n        },\n    }\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob.build_job","title":"build_job","text":"

    Builds the Kubernetes Job Manifest

    Source code in prefect/infrastructure/kubernetes.py
    def build_job(self) -> KubernetesManifest:\n    \"\"\"Builds the Kubernetes Job Manifest\"\"\"\n    job_manifest = copy.copy(self.job)\n    job_manifest = self._shortcut_customizations().apply(job_manifest)\n    job_manifest = self.customizations.apply(job_manifest)\n    return job_manifest\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob.customize_from_file","title":"customize_from_file classmethod","text":"

    Load an RFC 6902 JSON patch from a YAML or JSON file.

    Source code in prefect/infrastructure/kubernetes.py
    @classmethod\ndef customize_from_file(cls, filename: str) -> JsonPatch:\n    \"\"\"Load an RFC 6902 JSON patch from a YAML or JSON file.\"\"\"\n    with open(filename, \"r\", encoding=\"utf-8\") as f:\n        return JsonPatch(yaml.load(f, yaml.SafeLoader))\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJob.job_from_file","title":"job_from_file classmethod","text":"

    Load a Kubernetes Job manifest from a YAML or JSON file.

    Source code in prefect/infrastructure/kubernetes.py
    @classmethod\ndef job_from_file(cls, filename: str) -> KubernetesManifest:\n    \"\"\"Load a Kubernetes Job manifest from a YAML or JSON file.\"\"\"\n    with open(filename, \"r\", encoding=\"utf-8\") as f:\n        return yaml.load(f, yaml.SafeLoader)\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.KubernetesJobResult","title":"KubernetesJobResult","text":"

    Bases: InfrastructureResult

    Contains information about the final state of a completed Kubernetes Job

    Source code in prefect/infrastructure/kubernetes.py
    class KubernetesJobResult(InfrastructureResult):\n    \"\"\"Contains information about the final state of a completed Kubernetes Job\"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.Process","title":"Process","text":"

    Bases: Infrastructure

    Run a command in a new process.

    Current environment variables and Prefect settings will be included in the created process. Configured environment variables will override any current environment variables.

    Attributes:

    Name Type Description command

    A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this.

    env

    Environment variables to set for the new process.

    labels

    Labels for the process. Labels are for metadata purposes only and cannot be attached to the process itself.

    name

    A name for the process. For display purposes only.

    stream_output bool

    Whether to stream output to local stdout.

    working_dir Union[str, Path, None]

    Working directory where the process should be opened. If not set, a tmp directory will be used.

    Source code in prefect/infrastructure/process.py
    class Process(Infrastructure):\n    \"\"\"\n    Run a command in a new process.\n\n    Current environment variables and Prefect settings will be included in the created\n    process. Configured environment variables will override any current environment\n    variables.\n\n    Attributes:\n        command: A list of strings specifying the command to run in the container to\n            start the flow run. In most cases you should not override this.\n        env: Environment variables to set for the new process.\n        labels: Labels for the process. Labels are for metadata purposes only and\n            cannot be attached to the process itself.\n        name: A name for the process. For display purposes only.\n        stream_output: Whether to stream output to local stdout.\n        working_dir: Working directory where the process should be opened. If not set,\n            a tmp directory will be used.\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/356e6766a91baf20e1d08bbe16e8b5aaef4d8643-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/concepts/infrastructure/#process\"\n\n    type: Literal[\"process\"] = Field(\n        default=\"process\", description=\"The type of infrastructure.\"\n    )\n    stream_output: bool = Field(\n        default=True,\n        description=(\n            \"If set, output will be streamed from the process to local standard output.\"\n        ),\n    )\n    working_dir: Union[str, Path, None] = Field(\n        default=None,\n        description=(\n            \"If set, the process will open within the specified path as the working\"\n            \" directory. Otherwise, a temporary directory will be created.\"\n        ),\n    )  # Underlying accepted types are str, bytes, PathLike[str], None\n\n    @sync_compatible\n    async def run(\n        self,\n        task_status: anyio.abc.TaskStatus = None,\n    ) -> \"ProcessResult\":\n        if not self.command:\n            raise ValueError(\"Process cannot be run with empty command.\")\n\n        _use_threaded_child_watcher()\n        display_name = f\" {self.name!r}\" if self.name else \"\"\n\n        # Open a subprocess to execute the flow run\n        self.logger.info(f\"Opening process{display_name}...\")\n        working_dir_ctx = (\n            tempfile.TemporaryDirectory(suffix=\"prefect\")\n            if not self.working_dir\n            else contextlib.nullcontext(self.working_dir)\n        )\n        with working_dir_ctx as working_dir:\n            self.logger.debug(\n                f\"Process{display_name} running command: {' '.join(self.command)} in\"\n                f\" {working_dir}\"\n            )\n\n            # We must add creationflags to a dict so it is only passed as a function\n            # parameter on Windows, because the presence of creationflags causes\n            # errors on Unix even if set to None\n            kwargs: Dict[str, object] = {}\n            if sys.platform == \"win32\":\n                kwargs[\"creationflags\"] = subprocess.CREATE_NEW_PROCESS_GROUP\n\n            process = await run_process(\n                self.command,\n                stream_output=self.stream_output,\n                task_status=task_status,\n                task_status_handler=_infrastructure_pid_from_process,\n                env=self._get_environment_variables(),\n                cwd=working_dir,\n                **kwargs,\n            )\n\n        # Use the pid for display if no name was given\n        display_name = display_name or f\" {process.pid}\"\n\n        if process.returncode:\n            help_message = None\n            if process.returncode == -9:\n                help_message = (\n                    \"This indicates that the process exited due to a SIGKILL signal. \"\n                    \"Typically, this is either caused by manual cancellation or \"\n                    \"high memory usage causing the operating system to \"\n                    \"terminate the process.\"\n                )\n            if process.returncode == -15:\n                help_message = (\n                    \"This indicates that the process exited due to a SIGTERM signal. \"\n                    \"Typically, this is caused by manual cancellation.\"\n                )\n            elif process.returncode == 247:\n                help_message = (\n                    \"This indicates that the process was terminated due to high \"\n                    \"memory usage.\"\n                )\n            elif (\n                sys.platform == \"win32\" and process.returncode == STATUS_CONTROL_C_EXIT\n            ):\n                help_message = (\n                    \"Process was terminated due to a Ctrl+C or Ctrl+Break signal. \"\n                    \"Typically, this is caused by manual cancellation.\"\n                )\n\n            self.logger.error(\n                f\"Process{display_name} exited with status code: {process.returncode}\"\n                + (f\"; {help_message}\" if help_message else \"\")\n            )\n        else:\n            self.logger.info(f\"Process{display_name} exited cleanly.\")\n\n        return ProcessResult(\n            status_code=process.returncode, identifier=str(process.pid)\n        )\n\n    async def kill(self, infrastructure_pid: str, grace_seconds: int = 30):\n        hostname, pid = _parse_infrastructure_pid(infrastructure_pid)\n\n        if hostname != socket.gethostname():\n            raise InfrastructureNotAvailable(\n                f\"Unable to kill process {pid!r}: The process is running on a different\"\n                f\" host {hostname!r}.\"\n            )\n\n        # In a non-windows environment first send a SIGTERM, then, after\n        # `grace_seconds` seconds have passed subsequent send SIGKILL. In\n        # Windows we use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        if sys.platform == \"win32\":\n            try:\n                os.kill(pid, signal.CTRL_BREAK_EVENT)\n            except (ProcessLookupError, WindowsError):\n                raise InfrastructureNotFound(\n                    f\"Unable to kill process {pid!r}: The process was not found.\"\n                )\n        else:\n            try:\n                os.kill(pid, signal.SIGTERM)\n            except ProcessLookupError:\n                raise InfrastructureNotFound(\n                    f\"Unable to kill process {pid!r}: The process was not found.\"\n                )\n\n            # Throttle how often we check if the process is still alive to keep\n            # from making too many system calls in a short period of time.\n            check_interval = max(grace_seconds / 10, 1)\n\n            with anyio.move_on_after(grace_seconds):\n                while True:\n                    await anyio.sleep(check_interval)\n\n                    # Detect if the process is still alive. If not do an early\n                    # return as the process respected the SIGTERM from above.\n                    try:\n                        os.kill(pid, 0)\n                    except ProcessLookupError:\n                        return\n\n            try:\n                os.kill(pid, signal.SIGKILL)\n            except OSError:\n                # We shouldn't ever end up here, but it's possible that the\n                # process ended right after the check above.\n                return\n\n    def preview(self):\n        environment = self._get_environment_variables(include_os_environ=False)\n        return \" \\\\\\n\".join(\n            [f\"{key}={value}\" for key, value in environment.items()]\n            + [\" \".join(self.command)]\n        )\n\n    def _get_environment_variables(self, include_os_environ: bool = True):\n        os_environ = os.environ if include_os_environ else {}\n        # The base environment must override the current environment or\n        # the Prefect settings context may not be respected\n        env = {**os_environ, **self._base_environment(), **self.env}\n\n        # Drop null values allowing users to \"unset\" variables\n        return {key: value for key, value in env.items() if value is not None}\n\n    def _base_flow_run_command(self):\n        return [get_sys_executable(), \"-m\", \"prefect.engine\"]\n\n    def get_corresponding_worker_type(self):\n        return \"process\"\n\n    async def generate_work_pool_base_job_template(self):\n        from prefect.workers.utilities import (\n            get_default_base_job_template_for_infrastructure_type,\n        )\n\n        base_job_template = await get_default_base_job_template_for_infrastructure_type(\n            self.get_corresponding_worker_type(),\n        )\n        assert (\n            base_job_template is not None\n        ), \"Failed to generate default base job template for Process worker.\"\n        for key, value in self.dict(exclude_unset=True, exclude_defaults=True).items():\n            if key == \"command\":\n                base_job_template[\"variables\"][\"properties\"][\"command\"][\"default\"] = (\n                    shlex.join(value)\n                )\n            elif key in [\n                \"type\",\n                \"block_type_slug\",\n                \"_block_document_id\",\n                \"_block_document_name\",\n                \"_is_anonymous\",\n            ]:\n                continue\n            elif key in base_job_template[\"variables\"][\"properties\"]:\n                base_job_template[\"variables\"][\"properties\"][key][\"default\"] = value\n            else:\n                self.logger.warning(\n                    f\"Variable {key!r} is not supported by Process work pools.\"\n                    \" Skipping.\"\n                )\n\n        return base_job_template\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/infrastructure/#prefect.infrastructure.ProcessResult","title":"ProcessResult","text":"

    Bases: InfrastructureResult

    Contains information about the final state of a completed process

    Source code in prefect/infrastructure/process.py
    class ProcessResult(InfrastructureResult):\n    \"\"\"Contains information about the final state of a completed process\"\"\"\n
    ","tags":["Python API","infrastructure","Docker","Kubernetes","subprocess","process"]},{"location":"api-ref/prefect/logging/","title":"Logging","text":"","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/#prefect.logging","title":"prefect.logging","text":"","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/#prefect.logging.get_logger","title":"get_logger cached","text":"

    Get a prefect logger. These loggers are intended for internal use within the prefect package.

    See get_run_logger for retrieving loggers for use within task or flow runs. By default, only run-related loggers are connected to the APILogHandler.

    Source code in prefect/logging/loggers.py
    @lru_cache()\ndef get_logger(name: str = None) -> logging.Logger:\n    \"\"\"\n    Get a `prefect` logger. These loggers are intended for internal use within the\n    `prefect` package.\n\n    See `get_run_logger` for retrieving loggers for use within task or flow runs.\n    By default, only run-related loggers are connected to the `APILogHandler`.\n    \"\"\"\n\n    parent_logger = logging.getLogger(\"prefect\")\n\n    if name:\n        # Append the name if given but allow explicit full names e.g. \"prefect.test\"\n        # should not become \"prefect.prefect.test\"\n        if not name.startswith(parent_logger.name + \".\"):\n            logger = parent_logger.getChild(name)\n        else:\n            logger = logging.getLogger(name)\n    else:\n        logger = parent_logger\n\n    return logger\n
    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/#prefect.logging.get_run_logger","title":"get_run_logger","text":"

    Get a Prefect logger for the current task run or flow run.

    The logger will be named either prefect.task_runs or prefect.flow_runs. Contextual data about the run will be attached to the log records.

    These loggers are connected to the APILogHandler by default to send log records to the API.

    Parameters:

    Name Type Description Default context RunContext

    A specific context may be provided as an override. By default, the context is inferred from global state and this should not be needed.

    None **kwargs str

    Additional keyword arguments will be attached to the log records in addition to the run metadata

    {}

    Raises:

    Type Description RuntimeError

    If no context can be found

    Source code in prefect/logging/loggers.py
    def get_run_logger(\n    context: \"RunContext\" = None, **kwargs: str\n) -> Union[logging.Logger, logging.LoggerAdapter]:\n    \"\"\"\n    Get a Prefect logger for the current task run or flow run.\n\n    The logger will be named either `prefect.task_runs` or `prefect.flow_runs`.\n    Contextual data about the run will be attached to the log records.\n\n    These loggers are connected to the `APILogHandler` by default to send log records to\n    the API.\n\n    Arguments:\n        context: A specific context may be provided as an override. By default, the\n            context is inferred from global state and this should not be needed.\n        **kwargs: Additional keyword arguments will be attached to the log records in\n            addition to the run metadata\n\n    Raises:\n        RuntimeError: If no context can be found\n    \"\"\"\n    # Check for existing contexts\n    task_run_context = prefect.context.TaskRunContext.get()\n    flow_run_context = prefect.context.FlowRunContext.get()\n\n    # Apply the context override\n    if context:\n        if isinstance(context, prefect.context.FlowRunContext):\n            flow_run_context = context\n        elif isinstance(context, prefect.context.TaskRunContext):\n            task_run_context = context\n        else:\n            raise TypeError(\n                f\"Received unexpected type {type(context).__name__!r} for context. \"\n                \"Expected one of 'None', 'FlowRunContext', or 'TaskRunContext'.\"\n            )\n\n    # Determine if this is a task or flow run logger\n    if task_run_context:\n        logger = task_run_logger(\n            task_run=task_run_context.task_run,\n            task=task_run_context.task,\n            flow_run=flow_run_context.flow_run if flow_run_context else None,\n            flow=flow_run_context.flow if flow_run_context else None,\n            **kwargs,\n        )\n    elif flow_run_context:\n        logger = flow_run_logger(\n            flow_run=flow_run_context.flow_run, flow=flow_run_context.flow, **kwargs\n        )\n    elif (\n        get_logger(\"prefect.flow_run\").disabled\n        and get_logger(\"prefect.task_run\").disabled\n    ):\n        logger = logging.getLogger(\"null\")\n    else:\n        raise MissingContextError(\"There is no active flow or task run context.\")\n\n    return logger\n
    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/manifests/","title":"prefect.manifests","text":"","tags":["Python API","deployments"]},{"location":"api-ref/prefect/manifests/#prefect.manifests","title":"prefect.manifests","text":"

    Manifests are portable descriptions of one or more workflows within a given directory structure.

    They are the foundational building blocks for defining Flow Deployments.

    ","tags":["Python API","deployments"]},{"location":"api-ref/prefect/manifests/#prefect.manifests.Manifest","title":"Manifest","text":"

    Bases: BaseModel

    A JSON representation of a flow.

    Source code in prefect/manifests.py
    class Manifest(BaseModel):\n    \"\"\"A JSON representation of a flow.\"\"\"\n\n    flow_name: str = Field(default=..., description=\"The name of the flow.\")\n    import_path: str = Field(\n        default=..., description=\"The relative import path for the flow.\"\n    )\n    parameter_openapi_schema: ParameterSchema = Field(\n        default=..., description=\"The OpenAPI schema of the flow's parameters.\"\n    )\n
    ","tags":["Python API","deployments"]},{"location":"api-ref/prefect/serializers/","title":"prefect.serializers","text":"","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers","title":"prefect.serializers","text":"

    Serializer implementations for converting objects to bytes and bytes to objects.

    All serializers are based on the Serializer class and include a type string that allows them to be referenced without referencing the actual class. For example, you can get often specify the JSONSerializer with the string \"json\". Some serializers support additional settings for configuration of serialization. These are stored on the instance so the same settings can be used to load saved objects.

    All serializers must implement dumps and loads which convert objects to bytes and bytes to an object respectively.

    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.CompressedJSONSerializer","title":"CompressedJSONSerializer","text":"

    Bases: CompressedSerializer

    A compressed serializer preconfigured to use the json serializer.

    Source code in prefect/serializers.py
    class CompressedJSONSerializer(CompressedSerializer):\n    \"\"\"\n    A compressed serializer preconfigured to use the json serializer.\n    \"\"\"\n\n    type: Literal[\"compressed/json\"] = \"compressed/json\"\n    serializer: Serializer = pydantic.Field(default_factory=JSONSerializer)\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.CompressedPickleSerializer","title":"CompressedPickleSerializer","text":"

    Bases: CompressedSerializer

    A compressed serializer preconfigured to use the pickle serializer.

    Source code in prefect/serializers.py
    class CompressedPickleSerializer(CompressedSerializer):\n    \"\"\"\n    A compressed serializer preconfigured to use the pickle serializer.\n    \"\"\"\n\n    type: Literal[\"compressed/pickle\"] = \"compressed/pickle\"\n    serializer: Serializer = pydantic.Field(default_factory=PickleSerializer)\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.CompressedSerializer","title":"CompressedSerializer","text":"

    Bases: Serializer

    Wraps another serializer, compressing its output. Uses lzma by default. See compressionlib for using alternative libraries.

    Attributes:

    Name Type Description serializer Serializer

    The serializer to use before compression.

    compressionlib str

    The import path of a compression module to use. Must have methods compress(bytes) -> bytes and decompress(bytes) -> bytes.

    level str

    If not null, the level of compression to pass to compress.

    Source code in prefect/serializers.py
    class CompressedSerializer(Serializer):\n    \"\"\"\n    Wraps another serializer, compressing its output.\n    Uses `lzma` by default. See `compressionlib` for using alternative libraries.\n\n    Attributes:\n        serializer: The serializer to use before compression.\n        compressionlib: The import path of a compression module to use.\n            Must have methods `compress(bytes) -> bytes` and `decompress(bytes) -> bytes`.\n        level: If not null, the level of compression to pass to `compress`.\n    \"\"\"\n\n    type: Literal[\"compressed\"] = \"compressed\"\n\n    serializer: Serializer\n    compressionlib: str = \"lzma\"\n\n    @pydantic.validator(\"serializer\", pre=True)\n    def cast_type_names_to_serializers(cls, value):\n        if isinstance(value, str):\n            return Serializer(type=value)\n        return value\n\n    @pydantic.validator(\"compressionlib\")\n    def check_compressionlib(cls, value):\n        \"\"\"\n        Check that the given pickle library is importable and has compress/decompress\n        methods.\n        \"\"\"\n        try:\n            compressor = from_qualified_name(value)\n        except (ImportError, AttributeError) as exc:\n            raise ValueError(\n                f\"Failed to import requested compression library: {value!r}.\"\n            ) from exc\n\n        if not callable(getattr(compressor, \"compress\", None)):\n            raise ValueError(\n                f\"Compression library at {value!r} does not have a 'compress' method.\"\n            )\n\n        if not callable(getattr(compressor, \"decompress\", None)):\n            raise ValueError(\n                f\"Compression library at {value!r} does not have a 'decompress' method.\"\n            )\n\n        return value\n\n    def dumps(self, obj: Any) -> bytes:\n        blob = self.serializer.dumps(obj)\n        compressor = from_qualified_name(self.compressionlib)\n        return base64.encodebytes(compressor.compress(blob))\n\n    def loads(self, blob: bytes) -> Any:\n        compressor = from_qualified_name(self.compressionlib)\n        uncompressed = compressor.decompress(base64.decodebytes(blob))\n        return self.serializer.loads(uncompressed)\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.CompressedSerializer.check_compressionlib","title":"check_compressionlib","text":"

    Check that the given pickle library is importable and has compress/decompress methods.

    Source code in prefect/serializers.py
    @pydantic.validator(\"compressionlib\")\ndef check_compressionlib(cls, value):\n    \"\"\"\n    Check that the given pickle library is importable and has compress/decompress\n    methods.\n    \"\"\"\n    try:\n        compressor = from_qualified_name(value)\n    except (ImportError, AttributeError) as exc:\n        raise ValueError(\n            f\"Failed to import requested compression library: {value!r}.\"\n        ) from exc\n\n    if not callable(getattr(compressor, \"compress\", None)):\n        raise ValueError(\n            f\"Compression library at {value!r} does not have a 'compress' method.\"\n        )\n\n    if not callable(getattr(compressor, \"decompress\", None)):\n        raise ValueError(\n            f\"Compression library at {value!r} does not have a 'decompress' method.\"\n        )\n\n    return value\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.JSONSerializer","title":"JSONSerializer","text":"

    Bases: Serializer

    Serializes data to JSON.

    Input types must be compatible with the stdlib json library.

    Wraps the json library to serialize to UTF-8 bytes instead of string types.

    Source code in prefect/serializers.py
    class JSONSerializer(Serializer):\n    \"\"\"\n    Serializes data to JSON.\n\n    Input types must be compatible with the stdlib json library.\n\n    Wraps the `json` library to serialize to UTF-8 bytes instead of string types.\n    \"\"\"\n\n    type: Literal[\"json\"] = \"json\"\n    jsonlib: str = \"json\"\n    object_encoder: Optional[str] = pydantic.Field(\n        default=\"prefect.serializers.prefect_json_object_encoder\",\n        description=(\n            \"An optional callable to use when serializing objects that are not \"\n            \"supported by the JSON encoder. By default, this is set to a callable that \"\n            \"adds support for all types supported by Pydantic.\"\n        ),\n    )\n    object_decoder: Optional[str] = pydantic.Field(\n        default=\"prefect.serializers.prefect_json_object_decoder\",\n        description=(\n            \"An optional callable to use when deserializing objects. This callable \"\n            \"is passed each dictionary encountered during JSON deserialization. \"\n            \"By default, this is set to a callable that deserializes content created \"\n            \"by our default `object_encoder`.\"\n        ),\n    )\n    dumps_kwargs: dict = pydantic.Field(default_factory=dict)\n    loads_kwargs: dict = pydantic.Field(default_factory=dict)\n\n    @pydantic.validator(\"dumps_kwargs\")\n    def dumps_kwargs_cannot_contain_default(cls, value):\n        # `default` is set by `object_encoder`. A user provided callable would make this\n        # class unserializable anyway.\n        if \"default\" in value:\n            raise ValueError(\n                \"`default` cannot be provided. Use `object_encoder` instead.\"\n            )\n        return value\n\n    @pydantic.validator(\"loads_kwargs\")\n    def loads_kwargs_cannot_contain_object_hook(cls, value):\n        # `object_hook` is set by `object_decoder`. A user provided callable would make\n        # this class unserializable anyway.\n        if \"object_hook\" in value:\n            raise ValueError(\n                \"`object_hook` cannot be provided. Use `object_decoder` instead.\"\n            )\n        return value\n\n    def dumps(self, data: Any) -> bytes:\n        json = from_qualified_name(self.jsonlib)\n        kwargs = self.dumps_kwargs.copy()\n        if self.object_encoder:\n            kwargs[\"default\"] = from_qualified_name(self.object_encoder)\n        result = json.dumps(data, **kwargs)\n        if isinstance(result, str):\n            # The standard library returns str but others may return bytes directly\n            result = result.encode()\n        return result\n\n    def loads(self, blob: bytes) -> Any:\n        json = from_qualified_name(self.jsonlib)\n        kwargs = self.loads_kwargs.copy()\n        if self.object_decoder:\n            kwargs[\"object_hook\"] = from_qualified_name(self.object_decoder)\n        return json.loads(blob.decode(), **kwargs)\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.PickleSerializer","title":"PickleSerializer","text":"

    Bases: Serializer

    Serializes objects using the pickle protocol.

    • Uses cloudpickle by default. See picklelib for using alternative libraries.
    • Stores the version of the pickle library to check for compatibility during deserialization.
    • Wraps pickles in base64 for safe transmission.
    Source code in prefect/serializers.py
    class PickleSerializer(Serializer):\n    \"\"\"\n    Serializes objects using the pickle protocol.\n\n    - Uses `cloudpickle` by default. See `picklelib` for using alternative libraries.\n    - Stores the version of the pickle library to check for compatibility during\n        deserialization.\n    - Wraps pickles in base64 for safe transmission.\n    \"\"\"\n\n    type: Literal[\"pickle\"] = \"pickle\"\n\n    picklelib: str = \"cloudpickle\"\n    picklelib_version: str = None\n\n    @pydantic.validator(\"picklelib\")\n    def check_picklelib(cls, value):\n        \"\"\"\n        Check that the given pickle library is importable and has dumps/loads methods.\n        \"\"\"\n        try:\n            pickler = from_qualified_name(value)\n        except (ImportError, AttributeError) as exc:\n            raise ValueError(\n                f\"Failed to import requested pickle library: {value!r}.\"\n            ) from exc\n\n        if not callable(getattr(pickler, \"dumps\", None)):\n            raise ValueError(\n                f\"Pickle library at {value!r} does not have a 'dumps' method.\"\n            )\n\n        if not callable(getattr(pickler, \"loads\", None)):\n            raise ValueError(\n                f\"Pickle library at {value!r} does not have a 'loads' method.\"\n            )\n\n        return value\n\n    @pydantic.root_validator\n    def check_picklelib_version(cls, values):\n        \"\"\"\n        Infers a default value for `picklelib_version` if null or ensures it matches\n        the version retrieved from the `pickelib`.\n        \"\"\"\n        picklelib = values.get(\"picklelib\")\n        picklelib_version = values.get(\"picklelib_version\")\n\n        if not picklelib:\n            raise ValueError(\"Unable to check version of unrecognized picklelib module\")\n\n        pickler = from_qualified_name(picklelib)\n        pickler_version = getattr(pickler, \"__version__\", None)\n\n        if not picklelib_version:\n            values[\"picklelib_version\"] = pickler_version\n        elif picklelib_version != pickler_version:\n            warnings.warn(\n                (\n                    f\"Mismatched {picklelib!r} versions. Found {pickler_version} in the\"\n                    f\" environment but {picklelib_version} was requested. This may\"\n                    \" cause the serializer to fail.\"\n                ),\n                RuntimeWarning,\n                stacklevel=3,\n            )\n\n        return values\n\n    def dumps(self, obj: Any) -> bytes:\n        pickler = from_qualified_name(self.picklelib)\n        blob = pickler.dumps(obj)\n        return base64.encodebytes(blob)\n\n    def loads(self, blob: bytes) -> Any:\n        pickler = from_qualified_name(self.picklelib)\n        return pickler.loads(base64.decodebytes(blob))\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.PickleSerializer.check_picklelib","title":"check_picklelib","text":"

    Check that the given pickle library is importable and has dumps/loads methods.

    Source code in prefect/serializers.py
    @pydantic.validator(\"picklelib\")\ndef check_picklelib(cls, value):\n    \"\"\"\n    Check that the given pickle library is importable and has dumps/loads methods.\n    \"\"\"\n    try:\n        pickler = from_qualified_name(value)\n    except (ImportError, AttributeError) as exc:\n        raise ValueError(\n            f\"Failed to import requested pickle library: {value!r}.\"\n        ) from exc\n\n    if not callable(getattr(pickler, \"dumps\", None)):\n        raise ValueError(\n            f\"Pickle library at {value!r} does not have a 'dumps' method.\"\n        )\n\n    if not callable(getattr(pickler, \"loads\", None)):\n        raise ValueError(\n            f\"Pickle library at {value!r} does not have a 'loads' method.\"\n        )\n\n    return value\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.PickleSerializer.check_picklelib_version","title":"check_picklelib_version","text":"

    Infers a default value for picklelib_version if null or ensures it matches the version retrieved from the pickelib.

    Source code in prefect/serializers.py
    @pydantic.root_validator\ndef check_picklelib_version(cls, values):\n    \"\"\"\n    Infers a default value for `picklelib_version` if null or ensures it matches\n    the version retrieved from the `pickelib`.\n    \"\"\"\n    picklelib = values.get(\"picklelib\")\n    picklelib_version = values.get(\"picklelib_version\")\n\n    if not picklelib:\n        raise ValueError(\"Unable to check version of unrecognized picklelib module\")\n\n    pickler = from_qualified_name(picklelib)\n    pickler_version = getattr(pickler, \"__version__\", None)\n\n    if not picklelib_version:\n        values[\"picklelib_version\"] = pickler_version\n    elif picklelib_version != pickler_version:\n        warnings.warn(\n            (\n                f\"Mismatched {picklelib!r} versions. Found {pickler_version} in the\"\n                f\" environment but {picklelib_version} was requested. This may\"\n                \" cause the serializer to fail.\"\n            ),\n            RuntimeWarning,\n            stacklevel=3,\n        )\n\n    return values\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.Serializer","title":"Serializer","text":"

    Bases: BaseModel, Generic[D], ABC

    A serializer that can encode objects of type 'D' into bytes.

    Source code in prefect/serializers.py
    @add_type_dispatch\nclass Serializer(BaseModel, Generic[D], abc.ABC):\n    \"\"\"\n    A serializer that can encode objects of type 'D' into bytes.\n    \"\"\"\n\n    type: str\n\n    @abc.abstractmethod\n    def dumps(self, obj: D) -> bytes:\n        \"\"\"Encode the object into a blob of bytes.\"\"\"\n\n    @abc.abstractmethod\n    def loads(self, blob: bytes) -> D:\n        \"\"\"Decode the blob of bytes into an object.\"\"\"\n\n    class Config:\n        extra = \"forbid\"\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.Serializer.dumps","title":"dumps abstractmethod","text":"

    Encode the object into a blob of bytes.

    Source code in prefect/serializers.py
    @abc.abstractmethod\ndef dumps(self, obj: D) -> bytes:\n    \"\"\"Encode the object into a blob of bytes.\"\"\"\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.Serializer.loads","title":"loads abstractmethod","text":"

    Decode the blob of bytes into an object.

    Source code in prefect/serializers.py
    @abc.abstractmethod\ndef loads(self, blob: bytes) -> D:\n    \"\"\"Decode the blob of bytes into an object.\"\"\"\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.prefect_json_object_decoder","title":"prefect_json_object_decoder","text":"

    JSONDecoder.object_hook for decoding objects from JSON when previously encoded with prefect_json_object_encoder

    Source code in prefect/serializers.py
    def prefect_json_object_decoder(result: dict):\n    \"\"\"\n    `JSONDecoder.object_hook` for decoding objects from JSON when previously encoded\n    with `prefect_json_object_encoder`\n    \"\"\"\n    if \"__class__\" in result:\n        return pydantic.parse_obj_as(\n            from_qualified_name(result[\"__class__\"]), result[\"data\"]\n        )\n    elif \"__exc_type__\" in result:\n        return from_qualified_name(result[\"__exc_type__\"])(result[\"message\"])\n    else:\n        return result\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/serializers/#prefect.serializers.prefect_json_object_encoder","title":"prefect_json_object_encoder","text":"

    JSONEncoder.default for encoding objects into JSON with extended type support.

    Raises a TypeError to fallback on other encoders on failure.

    Source code in prefect/serializers.py
    def prefect_json_object_encoder(obj: Any) -> Any:\n    \"\"\"\n    `JSONEncoder.default` for encoding objects into JSON with extended type support.\n\n    Raises a `TypeError` to fallback on other encoders on failure.\n    \"\"\"\n    if isinstance(obj, BaseException):\n        return {\"__exc_type__\": to_qualified_name(obj.__class__), \"message\": str(obj)}\n    else:\n        return {\n            \"__class__\": to_qualified_name(obj.__class__),\n            \"data\": pydantic_encoder(obj),\n        }\n
    ","tags":["Python API","serializers","JSON","pickle"]},{"location":"api-ref/prefect/settings/","title":"prefect.settings","text":"","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings","title":"prefect.settings","text":"

    Prefect settings management.

    Each setting is defined as a Setting type. The name of each setting is stylized in all caps, matching the environment variable that can be used to change the setting.

    All settings defined in this file are used to generate a dynamic Pydantic settings class called Settings. When instantiated, this class will load settings from environment variables and pull default values from the setting definitions.

    The current instance of Settings being used by the application is stored in a SettingsContext model which allows each instance of the Settings class to be accessed in an async-safe manner.

    Aside from environment variables, we allow settings to be changed during the runtime of the process using profiles. Profiles contain setting overrides that the user may persist without setting environment variables. Profiles are also used internally for managing settings during task run execution where differing settings may be used concurrently in the same process and during testing where we need to override settings to ensure their value is respected as intended.

    The SettingsContext is set when the prefect module is imported. This context is referred to as the \"root\" settings context for clarity. Generally, this is the only settings context that will be used. When this context is entered, we will instantiate a Settings object, loading settings from environment variables and defaults, then we will load the active profile and use it to override settings. See enter_root_settings_context for details on determining the active profile.

    Another SettingsContext may be entered at any time to change the settings being used by the code within the context. Generally, users should not use this. Settings management should be left to Prefect application internals.

    Generally, settings should be accessed with SETTING_VARIABLE.value() which will pull the current Settings instance from the current SettingsContext and retrieve the value of the relevant setting.

    Accessing a setting's value will also call the Setting.value_callback which allows settings to be dynamically modified on retrieval. This allows us to make settings dependent on the value of other settings or perform other dynamic effects.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_HOME","title":"PREFECT_HOME = Setting(Path, default=Path('~') / '.prefect', value_callback=expanduser_in_path) module-attribute","text":"

    Prefect's home directory. Defaults to ~/.prefect. This directory may be created automatically when required.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXTRA_ENTRYPOINTS","title":"PREFECT_EXTRA_ENTRYPOINTS = Setting(str, default='') module-attribute","text":"

    Modules for Prefect to import when Prefect is imported.

    Values should be separated by commas, e.g. my_module,my_other_module. Objects within modules may be specified by a ':' partition, e.g. my_module:my_object. If a callable object is provided, it will be called with no arguments on import.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_DEBUG_MODE","title":"PREFECT_DEBUG_MODE = Setting(bool, default=False) module-attribute","text":"

    If True, places the API in debug mode. This may modify behavior to facilitate debugging, including extra logs and other verbose assistance. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLI_COLORS","title":"PREFECT_CLI_COLORS = Setting(bool, default=True) module-attribute","text":"

    If True, use colors in CLI output. If False, output will not include colors codes. Defaults to True.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLI_PROMPT","title":"PREFECT_CLI_PROMPT = Setting(Optional[bool], default=None) module-attribute","text":"

    If True, use interactive prompts in CLI commands. If False, no interactive prompts will be used. If None, the value will be dynamically determined based on the presence of an interactive-enabled terminal.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLI_WRAP_LINES","title":"PREFECT_CLI_WRAP_LINES = Setting(bool, default=True) module-attribute","text":"

    If True, wrap text by inserting new lines in long lines in CLI output. If False, output will not be wrapped. Defaults to True.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TEST_MODE","title":"PREFECT_TEST_MODE = Setting(bool, default=False) module-attribute","text":"

    If True, places the API in test mode. This may modify behavior to facilitate testing. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_UNIT_TEST_MODE","title":"PREFECT_UNIT_TEST_MODE = Setting(bool, default=False) module-attribute","text":"

    This variable only exists to facilitate unit testing. If True, code is executing in a unit test context. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TEST_SETTING","title":"PREFECT_TEST_SETTING = Setting(Any, default=None, value_callback=only_return_value_in_test_mode) module-attribute","text":"

    This variable only exists to facilitate testing of settings. If accessed when PREFECT_TEST_MODE is not set, None is returned.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_TLS_INSECURE_SKIP_VERIFY","title":"PREFECT_API_TLS_INSECURE_SKIP_VERIFY = Setting(bool, default=False) module-attribute","text":"

    If True, disables SSL checking to allow insecure requests. This is recommended only during development, e.g. when using self-signed certificates.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_URL","title":"PREFECT_API_URL = Setting(str, default=None) module-attribute","text":"

    If provided, the URL of a hosted Prefect API. Defaults to None.

    When using Prefect Cloud, this will include an account and workspace.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_KEY","title":"PREFECT_API_KEY = Setting(str, default=None, is_secret=True) module-attribute","text":"

    API key used to authenticate with a the Prefect API. Defaults to None.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_ENABLE_HTTP2","title":"PREFECT_API_ENABLE_HTTP2 = Setting(bool, default=True) module-attribute","text":"

    If true, enable support for HTTP/2 for communicating with an API.

    If the API does not support HTTP/2, this will have no effect and connections will be made via HTTP/1.1.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLIENT_MAX_RETRIES","title":"PREFECT_CLIENT_MAX_RETRIES = Setting(int, default=5) module-attribute","text":"

    The maximum number of retries to perform on failed HTTP requests.

    Defaults to 5. Set to 0 to disable retries.

    See PREFECT_CLIENT_RETRY_EXTRA_CODES for details on which HTTP status codes are retried.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLIENT_RETRY_JITTER_FACTOR","title":"PREFECT_CLIENT_RETRY_JITTER_FACTOR = Setting(float, default=0.2) module-attribute","text":"

    A value greater than or equal to zero to control the amount of jitter added to retried client requests. Higher values introduce larger amounts of jitter.

    Set to 0 to disable jitter. See clamped_poisson_interval for details on the how jitter can affect retry lengths.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLIENT_RETRY_EXTRA_CODES","title":"PREFECT_CLIENT_RETRY_EXTRA_CODES = Setting(str, default='', value_callback=status_codes_as_integers_in_range) module-attribute","text":"

    A comma-separated list of extra HTTP status codes to retry on. Defaults to an empty string. 429, 502 and 503 are always retried. Please note that not all routes are idempotent and retrying may result in unexpected behavior.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLOUD_API_URL","title":"PREFECT_CLOUD_API_URL = Setting(str, default='https://api.prefect.cloud/api', value_callback=check_for_deprecated_cloud_url) module-attribute","text":"

    API URL for Prefect Cloud. Used for authentication.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLOUD_URL","title":"PREFECT_CLOUD_URL = Setting(str, default=None, deprecated=True, deprecated_start_date='Dec 2022', deprecated_help='Use `PREFECT_CLOUD_API_URL` instead.') module-attribute","text":"","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_UI_URL","title":"PREFECT_UI_URL = Setting(Optional[str], default=None, value_callback=default_ui_url) module-attribute","text":"

    The URL for the UI. By default, this is inferred from the PREFECT_API_URL.

    When using Prefect Cloud, this will include the account and workspace. When using an ephemeral server, this will be None.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_CLOUD_UI_URL","title":"PREFECT_CLOUD_UI_URL = Setting(str, default=None, value_callback=default_cloud_ui_url) module-attribute","text":"

    The URL for the Cloud UI. By default, this is inferred from the PREFECT_CLOUD_API_URL.

    PREFECT_UI_URL will be workspace specific and will be usable in the open source too.

    In contrast, this value is only valid for Cloud and will not include the workspace.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_REQUEST_TIMEOUT","title":"PREFECT_API_REQUEST_TIMEOUT = Setting(float, default=60.0) module-attribute","text":"

    The default timeout for requests to the API

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN","title":"PREFECT_EXPERIMENTAL_WARN = Setting(bool, default=True) module-attribute","text":"

    If enabled, warn on usage of experimental features.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_PROFILES_PATH","title":"PREFECT_PROFILES_PATH = Setting(Path, default=Path('${PREFECT_HOME}') / 'profiles.toml', value_callback=template_with_settings(PREFECT_HOME)) module-attribute","text":"

    The path to a profiles configuration files.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RESULTS_DEFAULT_SERIALIZER","title":"PREFECT_RESULTS_DEFAULT_SERIALIZER = Setting(str, default='pickle') module-attribute","text":"

    The default serializer to use when not otherwise specified.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RESULTS_PERSIST_BY_DEFAULT","title":"PREFECT_RESULTS_PERSIST_BY_DEFAULT = Setting(bool, default=False) module-attribute","text":"

    The default setting for persisting results when not otherwise specified. If enabled, flow and task results will be persisted unless they opt out.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASKS_REFRESH_CACHE","title":"PREFECT_TASKS_REFRESH_CACHE = Setting(bool, default=False) module-attribute","text":"

    If True, enables a refresh of cached results: re-executing the task will refresh the cached results. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASK_DEFAULT_RETRIES","title":"PREFECT_TASK_DEFAULT_RETRIES = Setting(int, default=0) module-attribute","text":"

    This value sets the default number of retries for all tasks. This value does not overwrite individually set retries values on tasks

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_FLOW_DEFAULT_RETRIES","title":"PREFECT_FLOW_DEFAULT_RETRIES = Setting(int, default=0) module-attribute","text":"

    This value sets the default number of retries for all flows. This value does not overwrite individually set retries values on a flow

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_FLOW_DEFAULT_RETRY_DELAY_SECONDS","title":"PREFECT_FLOW_DEFAULT_RETRY_DELAY_SECONDS = Setting(Union[int, float], default=0) module-attribute","text":"

    This value sets the retry delay seconds for all flows. This value does not overwrite individually set retry delay seconds

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASK_DEFAULT_RETRY_DELAY_SECONDS","title":"PREFECT_TASK_DEFAULT_RETRY_DELAY_SECONDS = Setting(Union[float, int, List[float]], default=0) module-attribute","text":"

    This value sets the default retry delay seconds for all tasks. This value does not overwrite individually set retry delay seconds

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS","title":"PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS = Setting(int, default=30) module-attribute","text":"

    The number of seconds to wait before retrying when a task run cannot secure a concurrency slot from the server.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOCAL_STORAGE_PATH","title":"PREFECT_LOCAL_STORAGE_PATH = Setting(Path, default=Path('${PREFECT_HOME}') / 'storage', value_callback=template_with_settings(PREFECT_HOME)) module-attribute","text":"

    The path to a block storage directory to store things in.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_MEMO_STORE_PATH","title":"PREFECT_MEMO_STORE_PATH = Setting(Path, default=Path('${PREFECT_HOME}') / 'memo_store.toml', value_callback=template_with_settings(PREFECT_HOME)) module-attribute","text":"

    The path to the memo store file.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_MEMOIZE_BLOCK_AUTO_REGISTRATION","title":"PREFECT_MEMOIZE_BLOCK_AUTO_REGISTRATION = Setting(bool, default=True) module-attribute","text":"

    Controls whether or not block auto-registration on start up should be memoized. Setting to False may result in slower server start up times.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_LEVEL","title":"PREFECT_LOGGING_LEVEL = Setting(str, default='INFO', value_callback=debug_mode_log_level) module-attribute","text":"

    The default logging level for Prefect loggers. Defaults to \"INFO\" during normal operation. Is forced to \"DEBUG\" during debug mode.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_INTERNAL_LEVEL","title":"PREFECT_LOGGING_INTERNAL_LEVEL = Setting(str, default='ERROR', value_callback=debug_mode_log_level) module-attribute","text":"

    The default logging level for Prefect's internal machinery loggers. Defaults to \"ERROR\" during normal operation. Is forced to \"DEBUG\" during debug mode.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_SERVER_LEVEL","title":"PREFECT_LOGGING_SERVER_LEVEL = Setting(str, default='WARNING') module-attribute","text":"

    The default logging level for the Prefect API server.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_SETTINGS_PATH","title":"PREFECT_LOGGING_SETTINGS_PATH = Setting(Path, default=Path('${PREFECT_HOME}') / 'logging.yml', value_callback=template_with_settings(PREFECT_HOME)) module-attribute","text":"

    The path to a custom YAML logging configuration file. If no file is found, the default logging.yml is used. Defaults to a logging.yml in the Prefect home directory.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_EXTRA_LOGGERS","title":"PREFECT_LOGGING_EXTRA_LOGGERS = Setting(str, default='', value_callback=get_extra_loggers) module-attribute","text":"

    Additional loggers to attach to Prefect logging at runtime. Values should be comma separated. The handlers attached to the 'prefect' logger will be added to these loggers. Additionally, if the level is not set, it will be set to the same level as the 'prefect' logger.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_LOG_PRINTS","title":"PREFECT_LOGGING_LOG_PRINTS = Setting(bool, default=False) module-attribute","text":"

    If set, print statements in flows and tasks will be redirected to the Prefect logger for the given run. This setting can be overridden by individual tasks and flows.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_ENABLED","title":"PREFECT_LOGGING_TO_API_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Toggles sending logs to the API. If False, logs sent to the API log handler will not be sent to the API.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_BATCH_INTERVAL","title":"PREFECT_LOGGING_TO_API_BATCH_INTERVAL = Setting(float, default=2.0) module-attribute","text":"

    The number of seconds between batched writes of logs to the API.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_BATCH_SIZE","title":"PREFECT_LOGGING_TO_API_BATCH_SIZE = Setting(int, default=4000000) module-attribute","text":"

    The maximum size in bytes for a batch of logs.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_MAX_LOG_SIZE","title":"PREFECT_LOGGING_TO_API_MAX_LOG_SIZE = Setting(int, default=1000000) module-attribute","text":"

    The maximum size in bytes for a single log.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_TO_API_WHEN_MISSING_FLOW","title":"PREFECT_LOGGING_TO_API_WHEN_MISSING_FLOW = Setting(Literal['warn', 'error', 'ignore'], default='warn') module-attribute","text":"

    Controls the behavior when loggers attempt to send logs to the API handler from outside of a flow.

    All logs sent to the API must be associated with a flow run. The API log handler can only be used outside of a flow by manually providing a flow run identifier. Logs that are not associated with a flow run will not be sent to the API. This setting can be used to determine if a warning or error is displayed when the identifier is missing.

    The following options are available:

    • \"warn\": Log a warning message.
    • \"error\": Raise an error.
    • \"ignore\": Do not log a warning message or raise an error.
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SQLALCHEMY_POOL_SIZE","title":"PREFECT_SQLALCHEMY_POOL_SIZE = Setting(int, default=None) module-attribute","text":"

    Controls connection pool size when using a PostgreSQL database with the Prefect API. If not set, the default SQLAlchemy pool size will be used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SQLALCHEMY_MAX_OVERFLOW","title":"PREFECT_SQLALCHEMY_MAX_OVERFLOW = Setting(int, default=None) module-attribute","text":"

    Controls maximum overflow of the connection pool when using a PostgreSQL database with the Prefect API. If not set, the default SQLAlchemy maximum overflow value will be used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_COLORS","title":"PREFECT_LOGGING_COLORS = Setting(bool, default=True) module-attribute","text":"

    Whether to style console logs with color.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_LOGGING_MARKUP","title":"PREFECT_LOGGING_MARKUP = Setting(bool, default=False) module-attribute","text":"

    Whether to interpret strings wrapped in square brackets as a style. This allows styles to be conveniently added to log messages, e.g. [red]This is a red message.[/red]. However, the downside is, if enabled, strings that contain square brackets may be inaccurately interpreted and lead to incomplete output, e.g. DROP TABLE [dbo].[SomeTable];\" outputs DROP TABLE .[SomeTable];.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD","title":"PREFECT_TASK_INTROSPECTION_WARN_THRESHOLD = Setting(float, default=10.0) module-attribute","text":"

    Threshold time in seconds for logging a warning if task parameter introspection exceeds this duration. Parameter introspection can be a significant performance hit when the parameter is a large collection object, e.g. a large dictionary or DataFrame, and each element needs to be inspected. See prefect.utilities.annotations.quote for more details. Defaults to 10.0. Set to 0 to disable logging the warning.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_AGENT_QUERY_INTERVAL","title":"PREFECT_AGENT_QUERY_INTERVAL = Setting(float, default=15) module-attribute","text":"

    The agent loop interval, in seconds. Agents will check for new runs this often. Defaults to 15.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_AGENT_PREFETCH_SECONDS","title":"PREFECT_AGENT_PREFETCH_SECONDS = Setting(int, default=15) module-attribute","text":"

    Agents will look for scheduled runs this many seconds in the future and attempt to run them. This accounts for any additional infrastructure spin-up time or latency in preparing a flow run. Note flow runs will not start before their scheduled time, even if they are prefetched. Defaults to 15.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_ASYNC_FETCH_STATE_RESULT","title":"PREFECT_ASYNC_FETCH_STATE_RESULT = Setting(bool, default=False) module-attribute","text":"

    Determines whether State.result() fetches results automatically or not. In Prefect 2.6.0, the State.result() method was updated to be async to facilitate automatic retrieval of results from storage which means when writing async code you must await the call. For backwards compatibility, the result is not retrieved by default for async users. You may opt into this per call by passing fetch=True or toggle this setting to change the behavior globally. This setting does not affect users writing synchronous tasks and flows. This setting does not affect retrieval of results when using Future.result().

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_BLOCKS_REGISTER_ON_START","title":"PREFECT_API_BLOCKS_REGISTER_ON_START = Setting(bool, default=True) module-attribute","text":"

    If set, any block types that have been imported will be registered with the backend on application startup. If not set, block types must be manually registered.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_PASSWORD","title":"PREFECT_API_DATABASE_PASSWORD = Setting(str, default=None, is_secret=True) module-attribute","text":"

    Password to template into the PREFECT_API_DATABASE_CONNECTION_URL. This is useful if the password must be provided separately from the connection URL. To use this setting, you must include it in your connection URL.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_CONNECTION_URL","title":"PREFECT_API_DATABASE_CONNECTION_URL = Setting(str, default=None, value_callback=default_database_connection_url, is_secret=True) module-attribute","text":"

    A database connection URL in a SQLAlchemy-compatible format. Prefect currently supports SQLite and Postgres. Note that all Prefect database engines must use an async driver - for SQLite, use sqlite+aiosqlite and for Postgres use postgresql+asyncpg.

    SQLite in-memory databases can be used by providing the url sqlite+aiosqlite:///file::memory:?cache=shared&uri=true&check_same_thread=false, which will allow the database to be accessed by multiple threads. Note that in-memory databases can not be accessed from multiple processes and should only be used for simple tests.

    Defaults to a sqlite database stored in the Prefect home directory.

    If you need to provide password via a different environment variable, you use the PREFECT_API_DATABASE_PASSWORD setting. For example:

    PREFECT_API_DATABASE_PASSWORD='mypassword'\nPREFECT_API_DATABASE_CONNECTION_URL='postgresql+asyncpg://postgres:${PREFECT_API_DATABASE_PASSWORD}@localhost/prefect'\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_ECHO","title":"PREFECT_API_DATABASE_ECHO = Setting(bool, default=False) module-attribute","text":"

    If True, SQLAlchemy will log all SQL issued to the database. Defaults to False.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_MIGRATE_ON_START","title":"PREFECT_API_DATABASE_MIGRATE_ON_START = Setting(bool, default=True) module-attribute","text":"

    If True, the database will be upgraded on application creation. If False, the database will need to be upgraded manually.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_TIMEOUT","title":"PREFECT_API_DATABASE_TIMEOUT = Setting(Optional[float], default=10.0) module-attribute","text":"

    A statement timeout, in seconds, applied to all database interactions made by the API. Defaults to 10 seconds.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DATABASE_CONNECTION_TIMEOUT","title":"PREFECT_API_DATABASE_CONNECTION_TIMEOUT = Setting(Optional[float], default=5) module-attribute","text":"

    A connection timeout, in seconds, applied to database connections. Defaults to 5.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS","title":"PREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS = Setting(float, default=60) module-attribute","text":"

    The scheduler loop interval, in seconds. This determines how often the scheduler will attempt to schedule new flow runs, but has no impact on how quickly either flow runs or task runs are actually executed. Defaults to 60.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_DEPLOYMENT_BATCH_SIZE","title":"PREFECT_API_SERVICES_SCHEDULER_DEPLOYMENT_BATCH_SIZE = Setting(int, default=100) module-attribute","text":"

    The number of deployments the scheduler will attempt to schedule in a single batch. If there are more deployments than the batch size, the scheduler immediately attempts to schedule the next batch; it does not sleep for scheduler_loop_seconds until it has visited every deployment once. Defaults to 100.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_MAX_RUNS","title":"PREFECT_API_SERVICES_SCHEDULER_MAX_RUNS = Setting(int, default=100) module-attribute","text":"

    The scheduler will attempt to schedule up to this many auto-scheduled runs in the future. Note that runs may have fewer than this many scheduled runs, depending on the value of scheduler_max_scheduled_time. Defaults to 100.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_MIN_RUNS","title":"PREFECT_API_SERVICES_SCHEDULER_MIN_RUNS = Setting(int, default=3) module-attribute","text":"

    The scheduler will attempt to schedule at least this many auto-scheduled runs in the future. Note that runs may have more than this many scheduled runs, depending on the value of scheduler_min_scheduled_time. Defaults to 3.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME","title":"PREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME = Setting(timedelta, default=timedelta(days=100)) module-attribute","text":"

    The scheduler will create new runs up to this far in the future. Note that this setting will take precedence over scheduler_max_runs: if a flow runs once a month and scheduler_max_scheduled_time is three months, then only three runs will be scheduled. Defaults to 100 days (8640000 seconds).

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME","title":"PREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME = Setting(timedelta, default=timedelta(hours=1)) module-attribute","text":"

    The scheduler will create new runs at least this far in the future. Note that this setting will take precedence over scheduler_min_runs: if a flow runs every hour and scheduler_min_scheduled_time is three hours, then three runs will be scheduled even if scheduler_min_runs is 1. Defaults to 1 hour (3600 seconds).

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_INSERT_BATCH_SIZE","title":"PREFECT_API_SERVICES_SCHEDULER_INSERT_BATCH_SIZE = Setting(int, default=500) module-attribute","text":"

    The number of flow runs the scheduler will attempt to insert in one batch across all deployments. If the number of flow runs to schedule exceeds this amount, the runs will be inserted in batches of this size. Defaults to 500.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_LATE_RUNS_LOOP_SECONDS","title":"PREFECT_API_SERVICES_LATE_RUNS_LOOP_SECONDS = Setting(float, default=5) module-attribute","text":"

    The late runs service will look for runs to mark as late this often. Defaults to 5.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_LATE_RUNS_AFTER_SECONDS","title":"PREFECT_API_SERVICES_LATE_RUNS_AFTER_SECONDS = Setting(timedelta, default=timedelta(seconds=5)) module-attribute","text":"

    The late runs service will mark runs as late after they have exceeded their scheduled start time by this many seconds. Defaults to 5 seconds.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_LOOP_SECONDS","title":"PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_LOOP_SECONDS = Setting(float, default=5) module-attribute","text":"

    The pause expiration service will look for runs to mark as failed this often. Defaults to 5.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_CANCELLATION_CLEANUP_LOOP_SECONDS","title":"PREFECT_API_SERVICES_CANCELLATION_CLEANUP_LOOP_SECONDS = Setting(float, default=20) module-attribute","text":"

    The cancellation cleanup service will look non-terminal tasks and subflows this often. Defaults to 20.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_DEFAULT_LIMIT","title":"PREFECT_API_DEFAULT_LIMIT = Setting(int, default=200) module-attribute","text":"

    The default limit applied to queries that can return multiple objects, such as POST /flow_runs/filter.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SERVER_API_HOST","title":"PREFECT_SERVER_API_HOST = Setting(str, default='127.0.0.1') module-attribute","text":"

    The API's host address (defaults to 127.0.0.1).

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SERVER_API_PORT","title":"PREFECT_SERVER_API_PORT = Setting(int, default=4200) module-attribute","text":"

    The API's port address (defaults to 4200).

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SERVER_API_KEEPALIVE_TIMEOUT","title":"PREFECT_SERVER_API_KEEPALIVE_TIMEOUT = Setting(int, default=5) module-attribute","text":"

    The API's keep alive timeout (defaults to 5). Refer to https://www.uvicorn.org/settings/#timeouts for details.

    When the API is hosted behind a load balancer, you may want to set this to a value greater than the load balancer's idle timeout.

    Note this setting only applies when calling prefect server start; if hosting the API with another tool you will need to configure this there instead.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_UI_ENABLED","title":"PREFECT_UI_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to serve the Prefect UI.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_UI_API_URL","title":"PREFECT_UI_API_URL = Setting(str, default=None, value_callback=default_ui_api_url) module-attribute","text":"

    The connection url for communication from the UI to the API. Defaults to PREFECT_API_URL if set. Otherwise, the default URL is generated from PREFECT_SERVER_API_HOST and PREFECT_SERVER_API_PORT. If providing a custom value, the aforementioned settings may be templated into the given string.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_SERVER_ANALYTICS_ENABLED","title":"PREFECT_SERVER_ANALYTICS_ENABLED = Setting(bool, default=True) module-attribute","text":"

    When enabled, Prefect sends anonymous data (e.g. count of flow runs, package version) on server startup to help us improve our product.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_SCHEDULER_ENABLED","title":"PREFECT_API_SERVICES_SCHEDULER_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the scheduling service in the server application. If disabled, you will need to run this service separately to schedule runs for deployments.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_LATE_RUNS_ENABLED","title":"PREFECT_API_SERVICES_LATE_RUNS_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the late runs service in the server application. If disabled, you will need to run this service separately to have runs past their scheduled start time marked as late.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_FLOW_RUN_NOTIFICATIONS_ENABLED","title":"PREFECT_API_SERVICES_FLOW_RUN_NOTIFICATIONS_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the flow run notifications service in the server application. If disabled, you will need to run this service separately to send flow run notifications.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_ENABLED","title":"PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the paused flow run expiration service in the server application. If disabled, paused flows that have timed out will remain in a Paused state until a resume attempt.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH","title":"PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH = Setting(int, default=2000) module-attribute","text":"

    The maximum number of characters allowed for a task run cache key. This setting cannot be changed client-side, it must be set on the server.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_SERVICES_CANCELLATION_CLEANUP_ENABLED","title":"PREFECT_API_SERVICES_CANCELLATION_CLEANUP_ENABLED = Setting(bool, default=True) module-attribute","text":"

    Whether or not to start the cancellation cleanup service in the server application. If disabled, task runs and subflow runs belonging to cancelled flows may remain in non-terminal states.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_API_MAX_FLOW_RUN_GRAPH_NODES","title":"PREFECT_API_MAX_FLOW_RUN_GRAPH_NODES = Setting(int, default=10000) module-attribute","text":"

    The maximum size of a flow run graph on the v2 API

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_EVENTS_CLIENT","title":"PREFECT_EXPERIMENTAL_ENABLE_EVENTS_CLIENT = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental Prefect work pools.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_EVENTS_CLIENT","title":"PREFECT_EXPERIMENTAL_WARN_EVENTS_CLIENT = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect work pools are used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_WORK_POOLS","title":"PREFECT_EXPERIMENTAL_ENABLE_WORK_POOLS = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental Prefect work pools.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_WORK_POOLS","title":"PREFECT_EXPERIMENTAL_WARN_WORK_POOLS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect work pools are used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_WORKERS","title":"PREFECT_EXPERIMENTAL_ENABLE_WORKERS = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental Prefect workers.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_WORKERS","title":"PREFECT_EXPERIMENTAL_WARN_WORKERS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect workers are used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_VISUALIZE","title":"PREFECT_EXPERIMENTAL_WARN_VISUALIZE = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect visualize is used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_ENHANCED_CANCELLATION","title":"PREFECT_EXPERIMENTAL_ENABLE_ENHANCED_CANCELLATION = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental enhanced flow run cancellation.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_ENHANCED_CANCELLATION","title":"PREFECT_EXPERIMENTAL_WARN_ENHANCED_CANCELLATION = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental enhanced flow run cancellation is used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_DEPLOYMENT_STATUS","title":"PREFECT_EXPERIMENTAL_ENABLE_DEPLOYMENT_STATUS = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable deployment status in the UI

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_DEPLOYMENT_STATUS","title":"PREFECT_EXPERIMENTAL_WARN_DEPLOYMENT_STATUS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when deployment status is used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_FLOW_RUN_INPUT","title":"PREFECT_EXPERIMENTAL_FLOW_RUN_INPUT = Setting(bool, default=False) module-attribute","text":"

    Whether or not to enable flow run input.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_FLOW_RUN_INPUT","title":"PREFECT_EXPERIMENTAL_WARN_FLOW_RUN_INPUT = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable flow run input.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_PROCESS_LIMIT","title":"PREFECT_RUNNER_PROCESS_LIMIT = Setting(int, default=5) module-attribute","text":"

    Maximum number of processes a runner will execute in parallel.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_POLL_FREQUENCY","title":"PREFECT_RUNNER_POLL_FREQUENCY = Setting(int, default=10) module-attribute","text":"

    Number of seconds a runner should wait between queries for scheduled work.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_SERVER_MISSED_POLLS_TOLERANCE","title":"PREFECT_RUNNER_SERVER_MISSED_POLLS_TOLERANCE = Setting(int, default=2) module-attribute","text":"

    Number of missed polls before a runner is considered unhealthy by its webserver.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_SERVER_HOST","title":"PREFECT_RUNNER_SERVER_HOST = Setting(str, default='0.0.0.0') module-attribute","text":"

    The host address the runner's webserver should bind to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_SERVER_PORT","title":"PREFECT_RUNNER_SERVER_PORT = Setting(int, default=8080) module-attribute","text":"

    The port the runner's webserver should bind to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_RUNNER_SERVER_LOG_LEVEL","title":"PREFECT_RUNNER_SERVER_LOG_LEVEL = Setting(str, default='error') module-attribute","text":"

    The log level of the runner's webserver.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_HEARTBEAT_SECONDS","title":"PREFECT_WORKER_HEARTBEAT_SECONDS = Setting(float, default=30) module-attribute","text":"

    Number of seconds a worker should wait between sending a heartbeat.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_QUERY_SECONDS","title":"PREFECT_WORKER_QUERY_SECONDS = Setting(float, default=10) module-attribute","text":"

    Number of seconds a worker should wait between queries for scheduled flow runs.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_PREFETCH_SECONDS","title":"PREFECT_WORKER_PREFETCH_SECONDS = Setting(float, default=10) module-attribute","text":"

    The number of seconds into the future a worker should query for scheduled flow runs. Can be used to compensate for infrastructure start up time for a worker.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_WEBSERVER_HOST","title":"PREFECT_WORKER_WEBSERVER_HOST = Setting(str, default='0.0.0.0') module-attribute","text":"

    The host address the worker's webserver should bind to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_WORKER_WEBSERVER_PORT","title":"PREFECT_WORKER_WEBSERVER_PORT = Setting(int, default=8080) module-attribute","text":"

    The port the worker's webserver should bind to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_EXTRA_RUNNER_ENDPOINTS","title":"PREFECT_EXPERIMENTAL_ENABLE_EXTRA_RUNNER_ENDPOINTS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to enable experimental worker webserver endpoints.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_ARTIFACTS","title":"PREFECT_EXPERIMENTAL_ENABLE_ARTIFACTS = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable experimental Prefect artifacts.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_ARTIFACTS","title":"PREFECT_EXPERIMENTAL_WARN_ARTIFACTS = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when experimental Prefect artifacts are used.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_ENABLE_WORKSPACE_DASHBOARD","title":"PREFECT_EXPERIMENTAL_ENABLE_WORKSPACE_DASHBOARD = Setting(bool, default=True) module-attribute","text":"

    Whether or not to enable the experimental workspace dashboard.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_EXPERIMENTAL_WARN_WORKSPACE_DASHBOARD","title":"PREFECT_EXPERIMENTAL_WARN_WORKSPACE_DASHBOARD = Setting(bool, default=False) module-attribute","text":"

    Whether or not to warn when the experimental workspace dashboard is enabled.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_DEFAULT_RESULT_STORAGE_BLOCK","title":"PREFECT_DEFAULT_RESULT_STORAGE_BLOCK = Setting(str, default=None) module-attribute","text":"

    The block-type/block-document slug of a block to use as the default result storage.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_DEFAULT_WORK_POOL_NAME","title":"PREFECT_DEFAULT_WORK_POOL_NAME = Setting(str, default=None) module-attribute","text":"

    The default work pool to deploy to.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE","title":"PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE = Setting(str, default=None) module-attribute","text":"

    The default Docker namespace to use when building images.

    Can be either an organization/username or a registry URL with an organization/username.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Setting","title":"Setting","text":"

    Bases: Generic[T]

    Setting definition type.

    Source code in prefect/settings.py
    class Setting(Generic[T]):\n    \"\"\"\n    Setting definition type.\n    \"\"\"\n\n    def __init__(\n        self,\n        type: Type[T],\n        *,\n        deprecated: bool = False,\n        deprecated_start_date: Optional[str] = None,\n        deprecated_end_date: Optional[str] = None,\n        deprecated_help: str = \"\",\n        deprecated_when_message: str = \"\",\n        deprecated_when: Optional[Callable[[Any], bool]] = None,\n        deprecated_renamed_to: Optional[\"Setting[T]\"] = None,\n        value_callback: Optional[Callable[[\"Settings\", T], T]] = None,\n        is_secret: bool = False,\n        **kwargs: Any,\n    ) -> None:\n        self.field: fields.FieldInfo = Field(**kwargs)\n        self.type = type\n        self.value_callback = value_callback\n        self._name = None\n        self.is_secret = is_secret\n        self.deprecated = deprecated\n        self.deprecated_start_date = deprecated_start_date\n        self.deprecated_end_date = deprecated_end_date\n        self.deprecated_help = deprecated_help\n        self.deprecated_when = deprecated_when or (lambda _: True)\n        self.deprecated_when_message = deprecated_when_message\n        self.deprecated_renamed_to = deprecated_renamed_to\n        self.deprecated_renamed_from = None\n        self.__doc__ = self.field.description\n\n        # Validate the deprecation settings, will throw an error at setting definition\n        # time if the developer has not configured it correctly\n        if deprecated:\n            generate_deprecation_message(\n                name=\"...\",  # setting names not populated until after init\n                start_date=self.deprecated_start_date,\n                end_date=self.deprecated_end_date,\n                help=self.deprecated_help,\n                when=self.deprecated_when_message,\n            )\n\n        if deprecated_renamed_to is not None:\n            # Track the deprecation both ways\n            deprecated_renamed_to.deprecated_renamed_from = self\n\n    def value(self, bypass_callback: bool = False) -> T:\n        \"\"\"\n        Get the current value of a setting.\n\n        Example:\n        ```python\n        from prefect.settings import PREFECT_API_URL\n        PREFECT_API_URL.value()\n        ```\n        \"\"\"\n        return self.value_from(get_current_settings(), bypass_callback=bypass_callback)\n\n    def value_from(self, settings: \"Settings\", bypass_callback: bool = False) -> T:\n        \"\"\"\n        Get the value of a setting from a settings object\n\n        Example:\n        ```python\n        from prefect.settings import get_default_settings\n        PREFECT_API_URL.value_from(get_default_settings())\n        ```\n        \"\"\"\n        value = settings.value_of(self, bypass_callback=bypass_callback)\n\n        if not bypass_callback and self.deprecated and self.deprecated_when(value):\n            # Check if this setting is deprecated and someone is accessing the value\n            # via the old name\n            warnings.warn(self.deprecated_message, DeprecationWarning, stacklevel=3)\n\n            # If the the value is empty, return the new setting's value for compat\n            if value is None and self.deprecated_renamed_to is not None:\n                return self.deprecated_renamed_to.value_from(settings)\n\n        if not bypass_callback and self.deprecated_renamed_from is not None:\n            # Check if this setting is a rename of a deprecated setting and the\n            # deprecated setting is set and should be used for compatibility\n            deprecated_value = self.deprecated_renamed_from.value_from(\n                settings, bypass_callback=True\n            )\n            if deprecated_value is not None:\n                warnings.warn(\n                    (\n                        f\"{self.deprecated_renamed_from.deprecated_message} Because\"\n                        f\" {self.deprecated_renamed_from.name!r} is set it will be used\"\n                        f\" instead of {self.name!r} for backwards compatibility.\"\n                    ),\n                    DeprecationWarning,\n                    stacklevel=3,\n                )\n            return deprecated_value or value\n\n        return value\n\n    @property\n    def name(self):\n        if self._name:\n            return self._name\n\n        # Lookup the name on first access\n        for name, val in tuple(globals().items()):\n            if val == self:\n                self._name = name\n                return name\n\n        raise ValueError(\"Setting not found in `prefect.settings` module.\")\n\n    @name.setter\n    def name(self, value: str):\n        self._name = value\n\n    @property\n    def deprecated_message(self):\n        return generate_deprecation_message(\n            name=f\"Setting {self.name!r}\",\n            start_date=self.deprecated_start_date,\n            end_date=self.deprecated_end_date,\n            help=self.deprecated_help,\n            when=self.deprecated_when_message,\n        )\n\n    def __repr__(self) -> str:\n        return f\"<{self.name}: {self.type.__name__}>\"\n\n    def __bool__(self) -> bool:\n        \"\"\"\n        Returns a truthy check of the current value.\n        \"\"\"\n        return bool(self.value())\n\n    def __eq__(self, __o: object) -> bool:\n        return __o.__eq__(self.value())\n\n    def __hash__(self) -> int:\n        return hash((type(self), self.name))\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Setting.value","title":"value","text":"

    Get the current value of a setting.

    Example:

    from prefect.settings import PREFECT_API_URL\nPREFECT_API_URL.value()\n

    Source code in prefect/settings.py
    def value(self, bypass_callback: bool = False) -> T:\n    \"\"\"\n    Get the current value of a setting.\n\n    Example:\n    ```python\n    from prefect.settings import PREFECT_API_URL\n    PREFECT_API_URL.value()\n    ```\n    \"\"\"\n    return self.value_from(get_current_settings(), bypass_callback=bypass_callback)\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Setting.value_from","title":"value_from","text":"

    Get the value of a setting from a settings object

    Example:

    from prefect.settings import get_default_settings\nPREFECT_API_URL.value_from(get_default_settings())\n

    Source code in prefect/settings.py
    def value_from(self, settings: \"Settings\", bypass_callback: bool = False) -> T:\n    \"\"\"\n    Get the value of a setting from a settings object\n\n    Example:\n    ```python\n    from prefect.settings import get_default_settings\n    PREFECT_API_URL.value_from(get_default_settings())\n    ```\n    \"\"\"\n    value = settings.value_of(self, bypass_callback=bypass_callback)\n\n    if not bypass_callback and self.deprecated and self.deprecated_when(value):\n        # Check if this setting is deprecated and someone is accessing the value\n        # via the old name\n        warnings.warn(self.deprecated_message, DeprecationWarning, stacklevel=3)\n\n        # If the the value is empty, return the new setting's value for compat\n        if value is None and self.deprecated_renamed_to is not None:\n            return self.deprecated_renamed_to.value_from(settings)\n\n    if not bypass_callback and self.deprecated_renamed_from is not None:\n        # Check if this setting is a rename of a deprecated setting and the\n        # deprecated setting is set and should be used for compatibility\n        deprecated_value = self.deprecated_renamed_from.value_from(\n            settings, bypass_callback=True\n        )\n        if deprecated_value is not None:\n            warnings.warn(\n                (\n                    f\"{self.deprecated_renamed_from.deprecated_message} Because\"\n                    f\" {self.deprecated_renamed_from.name!r} is set it will be used\"\n                    f\" instead of {self.name!r} for backwards compatibility.\"\n                ),\n                DeprecationWarning,\n                stacklevel=3,\n            )\n        return deprecated_value or value\n\n    return value\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings","title":"Settings","text":"

    Bases: SettingsFieldsMixin

    Contains validated Prefect settings.

    Settings should be accessed using the relevant Setting object. For example:

    from prefect.settings import PREFECT_HOME\nPREFECT_HOME.value()\n

    Accessing a setting attribute directly will ignore any value_callback mutations. This is not recommended:

    from prefect.settings import Settings\nSettings().PREFECT_PROFILES_PATH  # PosixPath('${PREFECT_HOME}/profiles.toml')\n

    Source code in prefect/settings.py
    @add_cloudpickle_reduction\nclass Settings(SettingsFieldsMixin):\n    \"\"\"\n    Contains validated Prefect settings.\n\n    Settings should be accessed using the relevant `Setting` object. For example:\n    ```python\n    from prefect.settings import PREFECT_HOME\n    PREFECT_HOME.value()\n    ```\n\n    Accessing a setting attribute directly will ignore any `value_callback` mutations.\n    This is not recommended:\n    ```python\n    from prefect.settings import Settings\n    Settings().PREFECT_PROFILES_PATH  # PosixPath('${PREFECT_HOME}/profiles.toml')\n    ```\n    \"\"\"\n\n    def value_of(self, setting: Setting[T], bypass_callback: bool = False) -> T:\n        \"\"\"\n        Retrieve a setting's value.\n        \"\"\"\n        value = getattr(self, setting.name)\n        if setting.value_callback and not bypass_callback:\n            value = setting.value_callback(self, value)\n        return value\n\n    @validator(PREFECT_LOGGING_LEVEL.name, PREFECT_LOGGING_SERVER_LEVEL.name)\n    def check_valid_log_level(cls, value):\n        if isinstance(value, str):\n            value = value.upper()\n        logging._checkLevel(value)\n        return value\n\n    @root_validator\n    def post_root_validators(cls, values):\n        \"\"\"\n        Add root validation functions for settings here.\n        \"\"\"\n        # TODO: We could probably register these dynamically but this is the simpler\n        #       approach for now. We can explore more interesting validation features\n        #       in the future.\n        values = max_log_size_smaller_than_batch_size(values)\n        values = warn_on_database_password_value_without_usage(values)\n        values = warn_on_misconfigured_api_url(values)\n        return values\n\n    def copy_with_update(\n        self,\n        updates: Mapping[Setting, Any] = None,\n        set_defaults: Mapping[Setting, Any] = None,\n        restore_defaults: Iterable[Setting] = None,\n    ) -> \"Settings\":\n        \"\"\"\n        Create a new `Settings` object with validation.\n\n        Arguments:\n            updates: A mapping of settings to new values. Existing values for the\n                given settings will be overridden.\n            set_defaults: A mapping of settings to new default values. Existing values for\n                the given settings will only be overridden if they were not set.\n            restore_defaults: An iterable of settings to restore to their default values.\n\n        Returns:\n            A new `Settings` object.\n        \"\"\"\n        updates = updates or {}\n        set_defaults = set_defaults or {}\n        restore_defaults = restore_defaults or set()\n        restore_defaults_names = {setting.name for setting in restore_defaults}\n\n        return self.__class__(\n            **{\n                **{setting.name: value for setting, value in set_defaults.items()},\n                **self.dict(exclude_unset=True, exclude=restore_defaults_names),\n                **{setting.name: value for setting, value in updates.items()},\n            }\n        )\n\n    def with_obfuscated_secrets(self):\n        \"\"\"\n        Returns a copy of this settings object with secret setting values obfuscated.\n        \"\"\"\n        settings = self.copy(\n            update={\n                setting.name: obfuscate(self.value_of(setting))\n                for setting in SETTING_VARIABLES.values()\n                if setting.is_secret\n                # Exclude deprecated settings with null values to avoid warnings\n                and not (setting.deprecated and self.value_of(setting) is None)\n            }\n        )\n        # Ensure that settings that have not been marked as \"set\" before are still so\n        # after we have updated their value above\n        settings.__fields_set__.intersection_update(self.__fields_set__)\n        return settings\n\n    def to_environment_variables(\n        self, include: Iterable[Setting] = None, exclude_unset: bool = False\n    ) -> Dict[str, str]:\n        \"\"\"\n        Convert the settings object to environment variables.\n\n        Note that setting values will not be run through their `value_callback` allowing\n        dynamic resolution to occur when loaded from the returned environment.\n\n        Args:\n            include_keys: An iterable of settings to include in the return value.\n                If not set, all settings are used.\n            exclude_unset: Only include settings that have been set (i.e. the value is\n                not from the default). If set, unset keys will be dropped even if they\n                are set in `include_keys`.\n\n        Returns:\n            A dictionary of settings with values cast to strings\n        \"\"\"\n        include = set(include or SETTING_VARIABLES.values())\n\n        if exclude_unset:\n            set_keys = {\n                # Collect all of the \"set\" keys and cast to `Setting` objects\n                SETTING_VARIABLES[key]\n                for key in self.dict(exclude_unset=True)\n            }\n            include.intersection_update(set_keys)\n\n        # Validate the types of items in `include` to prevent exclusion bugs\n        for key in include:\n            if not isinstance(key, Setting):\n                raise TypeError(\n                    \"Invalid type {type(key).__name__!r} for key in `include`.\"\n                )\n\n        env = {\n            # Use `getattr` instead of `value_of` to avoid value callback resolution\n            key: getattr(self, key)\n            for key, setting in SETTING_VARIABLES.items()\n            if setting in include\n        }\n\n        # Cast to strings and drop null values\n        return {key: str(value) for key, value in env.items() if value is not None}\n\n    class Config:\n        frozen = True\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings.value_of","title":"value_of","text":"

    Retrieve a setting's value.

    Source code in prefect/settings.py
    def value_of(self, setting: Setting[T], bypass_callback: bool = False) -> T:\n    \"\"\"\n    Retrieve a setting's value.\n    \"\"\"\n    value = getattr(self, setting.name)\n    if setting.value_callback and not bypass_callback:\n        value = setting.value_callback(self, value)\n    return value\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings.post_root_validators","title":"post_root_validators","text":"

    Add root validation functions for settings here.

    Source code in prefect/settings.py
    @root_validator\ndef post_root_validators(cls, values):\n    \"\"\"\n    Add root validation functions for settings here.\n    \"\"\"\n    # TODO: We could probably register these dynamically but this is the simpler\n    #       approach for now. We can explore more interesting validation features\n    #       in the future.\n    values = max_log_size_smaller_than_batch_size(values)\n    values = warn_on_database_password_value_without_usage(values)\n    values = warn_on_misconfigured_api_url(values)\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings.with_obfuscated_secrets","title":"with_obfuscated_secrets","text":"

    Returns a copy of this settings object with secret setting values obfuscated.

    Source code in prefect/settings.py
    def with_obfuscated_secrets(self):\n    \"\"\"\n    Returns a copy of this settings object with secret setting values obfuscated.\n    \"\"\"\n    settings = self.copy(\n        update={\n            setting.name: obfuscate(self.value_of(setting))\n            for setting in SETTING_VARIABLES.values()\n            if setting.is_secret\n            # Exclude deprecated settings with null values to avoid warnings\n            and not (setting.deprecated and self.value_of(setting) is None)\n        }\n    )\n    # Ensure that settings that have not been marked as \"set\" before are still so\n    # after we have updated their value above\n    settings.__fields_set__.intersection_update(self.__fields_set__)\n    return settings\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Settings.to_environment_variables","title":"to_environment_variables","text":"

    Convert the settings object to environment variables.

    Note that setting values will not be run through their value_callback allowing dynamic resolution to occur when loaded from the returned environment.

    Parameters:

    Name Type Description Default include_keys

    An iterable of settings to include in the return value. If not set, all settings are used.

    required exclude_unset bool

    Only include settings that have been set (i.e. the value is not from the default). If set, unset keys will be dropped even if they are set in include_keys.

    False

    Returns:

    Type Description Dict[str, str]

    A dictionary of settings with values cast to strings

    Source code in prefect/settings.py
    def to_environment_variables(\n    self, include: Iterable[Setting] = None, exclude_unset: bool = False\n) -> Dict[str, str]:\n    \"\"\"\n    Convert the settings object to environment variables.\n\n    Note that setting values will not be run through their `value_callback` allowing\n    dynamic resolution to occur when loaded from the returned environment.\n\n    Args:\n        include_keys: An iterable of settings to include in the return value.\n            If not set, all settings are used.\n        exclude_unset: Only include settings that have been set (i.e. the value is\n            not from the default). If set, unset keys will be dropped even if they\n            are set in `include_keys`.\n\n    Returns:\n        A dictionary of settings with values cast to strings\n    \"\"\"\n    include = set(include or SETTING_VARIABLES.values())\n\n    if exclude_unset:\n        set_keys = {\n            # Collect all of the \"set\" keys and cast to `Setting` objects\n            SETTING_VARIABLES[key]\n            for key in self.dict(exclude_unset=True)\n        }\n        include.intersection_update(set_keys)\n\n    # Validate the types of items in `include` to prevent exclusion bugs\n    for key in include:\n        if not isinstance(key, Setting):\n            raise TypeError(\n                \"Invalid type {type(key).__name__!r} for key in `include`.\"\n            )\n\n    env = {\n        # Use `getattr` instead of `value_of` to avoid value callback resolution\n        key: getattr(self, key)\n        for key, setting in SETTING_VARIABLES.items()\n        if setting in include\n    }\n\n    # Cast to strings and drop null values\n    return {key: str(value) for key, value in env.items() if value is not None}\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Profile","title":"Profile","text":"

    Bases: BaseModel

    A user profile containing settings.

    Source code in prefect/settings.py
    class Profile(BaseModel):\n    \"\"\"\n    A user profile containing settings.\n    \"\"\"\n\n    name: str\n    settings: Dict[Setting, Any] = Field(default_factory=dict)\n    source: Optional[Path]\n\n    @validator(\"settings\", pre=True)\n    def map_names_to_settings(cls, value):\n        if value is None:\n            return value\n\n        # Cast string setting names to variables\n        validated = {}\n        for setting, val in value.items():\n            if isinstance(setting, str) and setting in SETTING_VARIABLES:\n                validated[SETTING_VARIABLES[setting]] = val\n            elif isinstance(setting, Setting):\n                validated[setting] = val\n            else:\n                raise ValueError(f\"Unknown setting {setting!r}.\")\n\n        return validated\n\n    def validate_settings(self) -> None:\n        \"\"\"\n        Validate the settings contained in this profile.\n\n        Raises:\n            pydantic.ValidationError: When settings do not have valid values.\n        \"\"\"\n        # Create a new `Settings` instance with the settings from this profile relying\n        # on Pydantic validation to raise an error.\n        # We do not return the `Settings` object because this is not the recommended\n        # path for constructing settings with a profile. See `use_profile` instead.\n        Settings(**{setting.name: value for setting, value in self.settings.items()})\n\n    def convert_deprecated_renamed_settings(self) -> List[Tuple[Setting, Setting]]:\n        \"\"\"\n        Update settings in place to replace deprecated settings with new settings when\n        renamed.\n\n        Returns a list of tuples with the old and new setting.\n        \"\"\"\n        changed = []\n        for setting in tuple(self.settings):\n            if (\n                setting.deprecated\n                and setting.deprecated_renamed_to\n                and setting.deprecated_renamed_to not in self.settings\n            ):\n                self.settings[setting.deprecated_renamed_to] = self.settings.pop(\n                    setting\n                )\n                changed.append((setting, setting.deprecated_renamed_to))\n        return changed\n\n    class Config:\n        arbitrary_types_allowed = True\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Profile.validate_settings","title":"validate_settings","text":"

    Validate the settings contained in this profile.

    Raises:

    Type Description ValidationError

    When settings do not have valid values.

    Source code in prefect/settings.py
    def validate_settings(self) -> None:\n    \"\"\"\n    Validate the settings contained in this profile.\n\n    Raises:\n        pydantic.ValidationError: When settings do not have valid values.\n    \"\"\"\n    # Create a new `Settings` instance with the settings from this profile relying\n    # on Pydantic validation to raise an error.\n    # We do not return the `Settings` object because this is not the recommended\n    # path for constructing settings with a profile. See `use_profile` instead.\n    Settings(**{setting.name: value for setting, value in self.settings.items()})\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.Profile.convert_deprecated_renamed_settings","title":"convert_deprecated_renamed_settings","text":"

    Update settings in place to replace deprecated settings with new settings when renamed.

    Returns a list of tuples with the old and new setting.

    Source code in prefect/settings.py
    def convert_deprecated_renamed_settings(self) -> List[Tuple[Setting, Setting]]:\n    \"\"\"\n    Update settings in place to replace deprecated settings with new settings when\n    renamed.\n\n    Returns a list of tuples with the old and new setting.\n    \"\"\"\n    changed = []\n    for setting in tuple(self.settings):\n        if (\n            setting.deprecated\n            and setting.deprecated_renamed_to\n            and setting.deprecated_renamed_to not in self.settings\n        ):\n            self.settings[setting.deprecated_renamed_to] = self.settings.pop(\n                setting\n            )\n            changed.append((setting, setting.deprecated_renamed_to))\n    return changed\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection","title":"ProfilesCollection","text":"

    \" A utility class for working with a collection of profiles.

    Profiles in the collection must have unique names.

    The collection may store the name of the active profile.

    Source code in prefect/settings.py
    class ProfilesCollection:\n    \"\"\" \"\n    A utility class for working with a collection of profiles.\n\n    Profiles in the collection must have unique names.\n\n    The collection may store the name of the active profile.\n    \"\"\"\n\n    def __init__(\n        self, profiles: Iterable[Profile], active: Optional[str] = None\n    ) -> None:\n        self.profiles_by_name = {profile.name: profile for profile in profiles}\n        self.active_name = active\n\n    @property\n    def names(self) -> Set[str]:\n        \"\"\"\n        Return a set of profile names in this collection.\n        \"\"\"\n        return set(self.profiles_by_name.keys())\n\n    @property\n    def active_profile(self) -> Optional[Profile]:\n        \"\"\"\n        Retrieve the active profile in this collection.\n        \"\"\"\n        if self.active_name is None:\n            return None\n        return self[self.active_name]\n\n    def set_active(self, name: Optional[str], check: bool = True):\n        \"\"\"\n        Set the active profile name in the collection.\n\n        A null value may be passed to indicate that this collection does not determine\n        the active profile.\n        \"\"\"\n        if check and name is not None and name not in self.names:\n            raise ValueError(f\"Unknown profile name {name!r}.\")\n        self.active_name = name\n\n    def update_profile(\n        self, name: str, settings: Mapping[Union[Dict, str], Any], source: Path = None\n    ) -> Profile:\n        \"\"\"\n        Add a profile to the collection or update the existing on if the name is already\n        present in this collection.\n\n        If updating an existing profile, the settings will be merged. Settings can\n        be dropped from the existing profile by setting them to `None` in the new\n        profile.\n\n        Returns the new profile object.\n        \"\"\"\n        existing = self.profiles_by_name.get(name)\n\n        # Convert the input to a `Profile` to cast settings to the correct type\n        profile = Profile(name=name, settings=settings, source=source)\n\n        if existing:\n            new_settings = {**existing.settings, **profile.settings}\n\n            # Drop null keys to restore to default\n            for key, value in tuple(new_settings.items()):\n                if value is None:\n                    new_settings.pop(key)\n\n            new_profile = Profile(\n                name=profile.name,\n                settings=new_settings,\n                source=source or profile.source,\n            )\n        else:\n            new_profile = profile\n\n        self.profiles_by_name[new_profile.name] = new_profile\n\n        return new_profile\n\n    def add_profile(self, profile: Profile) -> None:\n        \"\"\"\n        Add a profile to the collection.\n\n        If the profile name already exists, an exception will be raised.\n        \"\"\"\n        if profile.name in self.profiles_by_name:\n            raise ValueError(\n                f\"Profile name {profile.name!r} already exists in collection.\"\n            )\n\n        self.profiles_by_name[profile.name] = profile\n\n    def remove_profile(self, name: str) -> None:\n        \"\"\"\n        Remove a profile from the collection.\n        \"\"\"\n        self.profiles_by_name.pop(name)\n\n    def without_profile_source(self, path: Optional[Path]) -> \"ProfilesCollection\":\n        \"\"\"\n        Remove profiles that were loaded from a given path.\n\n        Returns a new collection.\n        \"\"\"\n        return ProfilesCollection(\n            [\n                profile\n                for profile in self.profiles_by_name.values()\n                if profile.source != path\n            ],\n            active=self.active_name,\n        )\n\n    def to_dict(self):\n        \"\"\"\n        Convert to a dictionary suitable for writing to disk.\n        \"\"\"\n        return {\n            \"active\": self.active_name,\n            \"profiles\": {\n                profile.name: {\n                    setting.name: value for setting, value in profile.settings.items()\n                }\n                for profile in self.profiles_by_name.values()\n            },\n        }\n\n    def __getitem__(self, name: str) -> Profile:\n        return self.profiles_by_name[name]\n\n    def __iter__(self):\n        return self.profiles_by_name.__iter__()\n\n    def items(self):\n        return self.profiles_by_name.items()\n\n    def __eq__(self, __o: object) -> bool:\n        if not isinstance(__o, ProfilesCollection):\n            return False\n\n        return (\n            self.profiles_by_name == __o.profiles_by_name\n            and self.active_name == __o.active_name\n        )\n\n    def __repr__(self) -> str:\n        return (\n            f\"ProfilesCollection(profiles={list(self.profiles_by_name.values())!r},\"\n            f\" active={self.active_name!r})>\"\n        )\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.names","title":"names: Set[str] property","text":"

    Return a set of profile names in this collection.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.active_profile","title":"active_profile: Optional[Profile] property","text":"

    Retrieve the active profile in this collection.

    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.set_active","title":"set_active","text":"

    Set the active profile name in the collection.

    A null value may be passed to indicate that this collection does not determine the active profile.

    Source code in prefect/settings.py
    def set_active(self, name: Optional[str], check: bool = True):\n    \"\"\"\n    Set the active profile name in the collection.\n\n    A null value may be passed to indicate that this collection does not determine\n    the active profile.\n    \"\"\"\n    if check and name is not None and name not in self.names:\n        raise ValueError(f\"Unknown profile name {name!r}.\")\n    self.active_name = name\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.update_profile","title":"update_profile","text":"

    Add a profile to the collection or update the existing on if the name is already present in this collection.

    If updating an existing profile, the settings will be merged. Settings can be dropped from the existing profile by setting them to None in the new profile.

    Returns the new profile object.

    Source code in prefect/settings.py
    def update_profile(\n    self, name: str, settings: Mapping[Union[Dict, str], Any], source: Path = None\n) -> Profile:\n    \"\"\"\n    Add a profile to the collection or update the existing on if the name is already\n    present in this collection.\n\n    If updating an existing profile, the settings will be merged. Settings can\n    be dropped from the existing profile by setting them to `None` in the new\n    profile.\n\n    Returns the new profile object.\n    \"\"\"\n    existing = self.profiles_by_name.get(name)\n\n    # Convert the input to a `Profile` to cast settings to the correct type\n    profile = Profile(name=name, settings=settings, source=source)\n\n    if existing:\n        new_settings = {**existing.settings, **profile.settings}\n\n        # Drop null keys to restore to default\n        for key, value in tuple(new_settings.items()):\n            if value is None:\n                new_settings.pop(key)\n\n        new_profile = Profile(\n            name=profile.name,\n            settings=new_settings,\n            source=source or profile.source,\n        )\n    else:\n        new_profile = profile\n\n    self.profiles_by_name[new_profile.name] = new_profile\n\n    return new_profile\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.add_profile","title":"add_profile","text":"

    Add a profile to the collection.

    If the profile name already exists, an exception will be raised.

    Source code in prefect/settings.py
    def add_profile(self, profile: Profile) -> None:\n    \"\"\"\n    Add a profile to the collection.\n\n    If the profile name already exists, an exception will be raised.\n    \"\"\"\n    if profile.name in self.profiles_by_name:\n        raise ValueError(\n            f\"Profile name {profile.name!r} already exists in collection.\"\n        )\n\n    self.profiles_by_name[profile.name] = profile\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.remove_profile","title":"remove_profile","text":"

    Remove a profile from the collection.

    Source code in prefect/settings.py
    def remove_profile(self, name: str) -> None:\n    \"\"\"\n    Remove a profile from the collection.\n    \"\"\"\n    self.profiles_by_name.pop(name)\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.ProfilesCollection.without_profile_source","title":"without_profile_source","text":"

    Remove profiles that were loaded from a given path.

    Returns a new collection.

    Source code in prefect/settings.py
    def without_profile_source(self, path: Optional[Path]) -> \"ProfilesCollection\":\n    \"\"\"\n    Remove profiles that were loaded from a given path.\n\n    Returns a new collection.\n    \"\"\"\n    return ProfilesCollection(\n        [\n            profile\n            for profile in self.profiles_by_name.values()\n            if profile.source != path\n        ],\n        active=self.active_name,\n    )\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.get_extra_loggers","title":"get_extra_loggers","text":"

    value_callback for PREFECT_LOGGING_EXTRA_LOGGERSthat parses the CSV string into a list and trims whitespace from logger names.

    Source code in prefect/settings.py
    def get_extra_loggers(_: \"Settings\", value: str) -> List[str]:\n    \"\"\"\n    `value_callback` for `PREFECT_LOGGING_EXTRA_LOGGERS`that parses the CSV string into a\n    list and trims whitespace from logger names.\n    \"\"\"\n    return [name.strip() for name in value.split(\",\")] if value else []\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.debug_mode_log_level","title":"debug_mode_log_level","text":"

    value_callback for PREFECT_LOGGING_LEVEL that overrides the log level to DEBUG when debug mode is enabled.

    Source code in prefect/settings.py
    def debug_mode_log_level(settings, value):\n    \"\"\"\n    `value_callback` for `PREFECT_LOGGING_LEVEL` that overrides the log level to DEBUG\n    when debug mode is enabled.\n    \"\"\"\n    if PREFECT_DEBUG_MODE.value_from(settings):\n        return \"DEBUG\"\n    else:\n        return value\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.only_return_value_in_test_mode","title":"only_return_value_in_test_mode","text":"

    value_callback for PREFECT_TEST_SETTING that only allows access during test mode

    Source code in prefect/settings.py
    def only_return_value_in_test_mode(settings, value):\n    \"\"\"\n    `value_callback` for `PREFECT_TEST_SETTING` that only allows access during test mode\n    \"\"\"\n    if PREFECT_TEST_MODE.value_from(settings):\n        return value\n    else:\n        return None\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.default_ui_api_url","title":"default_ui_api_url","text":"

    value_callback for PREFECT_UI_API_URL that sets the default value to relative path '/api', otherwise it constructs an API URL from the API settings.

    Source code in prefect/settings.py
    def default_ui_api_url(settings, value):\n    \"\"\"\n    `value_callback` for `PREFECT_UI_API_URL` that sets the default value to\n    relative path '/api', otherwise it constructs an API URL from the API settings.\n    \"\"\"\n    if value is None:\n        # Set a default value\n        value = \"/api\"\n\n    return template_with_settings(\n        PREFECT_SERVER_API_HOST, PREFECT_SERVER_API_PORT, PREFECT_API_URL\n    )(settings, value)\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.status_codes_as_integers_in_range","title":"status_codes_as_integers_in_range","text":"

    value_callback for PREFECT_CLIENT_RETRY_EXTRA_CODES that ensures status codes are integers in the range 100-599.

    Source code in prefect/settings.py
    def status_codes_as_integers_in_range(_, value):\n    \"\"\"\n    `value_callback` for `PREFECT_CLIENT_RETRY_EXTRA_CODES` that ensures status codes\n    are integers in the range 100-599.\n    \"\"\"\n    if value == \"\":\n        return set()\n\n    values = {v.strip() for v in value.split(\",\")}\n\n    if any(not v.isdigit() or int(v) < 100 or int(v) > 599 for v in values):\n        raise ValueError(\n            \"PREFECT_CLIENT_RETRY_EXTRA_CODES must be a comma separated list of \"\n            \"integers between 100 and 599.\"\n        )\n\n    values = {int(v) for v in values}\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.template_with_settings","title":"template_with_settings","text":"

    Returns a value_callback that will template the given settings into the runtime value for the setting.

    Source code in prefect/settings.py
    def template_with_settings(*upstream_settings: Setting) -> Callable[[\"Settings\", T], T]:\n    \"\"\"\n    Returns a `value_callback` that will template the given settings into the runtime\n    value for the setting.\n    \"\"\"\n\n    def templater(settings, value):\n        if value is None:\n            return value  # Do not attempt to template a null string\n\n        original_type = type(value)\n        template_values = {\n            setting.name: setting.value_from(settings) for setting in upstream_settings\n        }\n        template = string.Template(str(value))\n        return original_type(template.substitute(template_values))\n\n    return templater\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.max_log_size_smaller_than_batch_size","title":"max_log_size_smaller_than_batch_size","text":"

    Validator for settings asserting the batch size and match log size are compatible

    Source code in prefect/settings.py
    def max_log_size_smaller_than_batch_size(values):\n    \"\"\"\n    Validator for settings asserting the batch size and match log size are compatible\n    \"\"\"\n    if (\n        values[\"PREFECT_LOGGING_TO_API_BATCH_SIZE\"]\n        < values[\"PREFECT_LOGGING_TO_API_MAX_LOG_SIZE\"]\n    ):\n        raise ValueError(\n            \"`PREFECT_LOGGING_TO_API_MAX_LOG_SIZE` cannot be larger than\"\n            \" `PREFECT_LOGGING_TO_API_BATCH_SIZE`\"\n        )\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.warn_on_database_password_value_without_usage","title":"warn_on_database_password_value_without_usage","text":"

    Validator for settings warning if the database password is set but not used.

    Source code in prefect/settings.py
    def warn_on_database_password_value_without_usage(values):\n    \"\"\"\n    Validator for settings warning if the database password is set but not used.\n    \"\"\"\n    value = values[\"PREFECT_API_DATABASE_PASSWORD\"]\n    if (\n        value\n        and not value.startswith(OBFUSCATED_PREFIX)\n        and (\n            \"PREFECT_API_DATABASE_PASSWORD\"\n            not in values[\"PREFECT_API_DATABASE_CONNECTION_URL\"]\n        )\n    ):\n        warnings.warn(\n            \"PREFECT_API_DATABASE_PASSWORD is set but not included in the \"\n            \"PREFECT_API_DATABASE_CONNECTION_URL. \"\n            \"The provided password will be ignored.\"\n        )\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.warn_on_misconfigured_api_url","title":"warn_on_misconfigured_api_url","text":"

    Validator for settings warning if the API URL is misconfigured.

    Source code in prefect/settings.py
    def warn_on_misconfigured_api_url(values):\n    \"\"\"\n    Validator for settings warning if the API URL is misconfigured.\n    \"\"\"\n    api_url = values[\"PREFECT_API_URL\"]\n    if api_url is not None:\n        misconfigured_mappings = {\n            \"app.prefect.cloud\": (\n                \"`PREFECT_API_URL` points to `app.prefect.cloud`. Did you\"\n                \" mean `api.prefect.cloud`?\"\n            ),\n            \"account/\": (\n                \"`PREFECT_API_URL` uses `/account/` but should use `/accounts/`.\"\n            ),\n            \"workspace/\": (\n                \"`PREFECT_API_URL` uses `/workspace/` but should use `/workspaces/`.\"\n            ),\n        }\n        warnings_list = []\n\n        for misconfig, warning in misconfigured_mappings.items():\n            if misconfig in api_url:\n                warnings_list.append(warning)\n\n        parsed_url = urlparse(api_url)\n        if parsed_url.path and not parsed_url.path.startswith(\"/api\"):\n            warnings_list.append(\n                \"`PREFECT_API_URL` should have `/api` after the base URL.\"\n            )\n\n        if warnings_list:\n            example = (\n                'e.g. PREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]\"'\n            )\n            warnings_list.append(example)\n\n            warnings.warn(\"\\n\".join(warnings_list), stacklevel=2)\n\n    return values\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.get_current_settings","title":"get_current_settings","text":"

    Returns a settings object populated with values from the current settings context or, if no settings context is active, the environment.

    Source code in prefect/settings.py
    def get_current_settings() -> Settings:\n    \"\"\"\n    Returns a settings object populated with values from the current settings context\n    or, if no settings context is active, the environment.\n    \"\"\"\n    from prefect.context import SettingsContext\n\n    settings_context = SettingsContext.get()\n    if settings_context is not None:\n        return settings_context.settings\n\n    return get_settings_from_env()\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.get_settings_from_env","title":"get_settings_from_env","text":"

    Returns a settings object populated with default values and overrides from environment variables, ignoring any values in profiles.

    Calls with the same environment return a cached object instead of reconstructing to avoid validation overhead.

    Source code in prefect/settings.py
    def get_settings_from_env() -> Settings:\n    \"\"\"\n    Returns a settings object populated with default values and overrides from\n    environment variables, ignoring any values in profiles.\n\n    Calls with the same environment return a cached object instead of reconstructing\n    to avoid validation overhead.\n    \"\"\"\n    # Since os.environ is a Dict[str, str] we can safely hash it by contents, but we\n    # must be careful to avoid hashing a generator instead of a tuple\n    cache_key = hash(tuple((key, value) for key, value in os.environ.items()))\n\n    if cache_key not in _FROM_ENV_CACHE:\n        _FROM_ENV_CACHE[cache_key] = Settings()\n\n    return _FROM_ENV_CACHE[cache_key]\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.get_default_settings","title":"get_default_settings","text":"

    Returns a settings object populated with default values, ignoring any overrides from environment variables or profiles.

    This is cached since the defaults should not change during the lifetime of the module.

    Source code in prefect/settings.py
    def get_default_settings() -> Settings:\n    \"\"\"\n    Returns a settings object populated with default values, ignoring any overrides\n    from environment variables or profiles.\n\n    This is cached since the defaults should not change during the lifetime of the\n    module.\n    \"\"\"\n    global _DEFAULTS_CACHE\n\n    if not _DEFAULTS_CACHE:\n        old = os.environ\n        try:\n            os.environ = {}\n            settings = get_settings_from_env()\n        finally:\n            os.environ = old\n\n        _DEFAULTS_CACHE = settings\n\n    return _DEFAULTS_CACHE\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.temporary_settings","title":"temporary_settings","text":"

    Temporarily override the current settings by entering a new profile.

    See Settings.copy_with_update for details on different argument behavior.

    Examples:

    >>> from prefect.settings import PREFECT_API_URL\n>>>\n>>> with temporary_settings(updates={PREFECT_API_URL: \"foo\"}):\n>>>    assert PREFECT_API_URL.value() == \"foo\"\n>>>\n>>>    with temporary_settings(set_defaults={PREFECT_API_URL: \"bar\"}):\n>>>         assert PREFECT_API_URL.value() == \"foo\"\n>>>\n>>>    with temporary_settings(restore_defaults={PREFECT_API_URL}):\n>>>         assert PREFECT_API_URL.value() is None\n>>>\n>>>         with temporary_settings(set_defaults={PREFECT_API_URL: \"bar\"})\n>>>             assert PREFECT_API_URL.value() == \"bar\"\n>>> assert PREFECT_API_URL.value() is None\n
    Source code in prefect/settings.py
    @contextmanager\ndef temporary_settings(\n    updates: Mapping[Setting, Any] = None,\n    set_defaults: Mapping[Setting, Any] = None,\n    restore_defaults: Iterable[Setting] = None,\n) -> Settings:\n    \"\"\"\n    Temporarily override the current settings by entering a new profile.\n\n    See `Settings.copy_with_update` for details on different argument behavior.\n\n    Examples:\n        >>> from prefect.settings import PREFECT_API_URL\n        >>>\n        >>> with temporary_settings(updates={PREFECT_API_URL: \"foo\"}):\n        >>>    assert PREFECT_API_URL.value() == \"foo\"\n        >>>\n        >>>    with temporary_settings(set_defaults={PREFECT_API_URL: \"bar\"}):\n        >>>         assert PREFECT_API_URL.value() == \"foo\"\n        >>>\n        >>>    with temporary_settings(restore_defaults={PREFECT_API_URL}):\n        >>>         assert PREFECT_API_URL.value() is None\n        >>>\n        >>>         with temporary_settings(set_defaults={PREFECT_API_URL: \"bar\"})\n        >>>             assert PREFECT_API_URL.value() == \"bar\"\n        >>> assert PREFECT_API_URL.value() is None\n    \"\"\"\n    import prefect.context\n\n    context = prefect.context.get_settings_context()\n\n    new_settings = context.settings.copy_with_update(\n        updates=updates, set_defaults=set_defaults, restore_defaults=restore_defaults\n    )\n\n    with prefect.context.SettingsContext(\n        profile=context.profile, settings=new_settings\n    ):\n        yield new_settings\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.load_profiles","title":"load_profiles","text":"

    Load all profiles from the default and current profile paths.

    Source code in prefect/settings.py
    def load_profiles() -> ProfilesCollection:\n    \"\"\"\n    Load all profiles from the default and current profile paths.\n    \"\"\"\n    profiles = _read_profiles_from(DEFAULT_PROFILES_PATH)\n\n    user_profiles_path = PREFECT_PROFILES_PATH.value()\n    if user_profiles_path.exists():\n        user_profiles = _read_profiles_from(user_profiles_path)\n\n        # Merge all of the user profiles with the defaults\n        for name in user_profiles:\n            profiles.update_profile(\n                name,\n                settings=user_profiles[name].settings,\n                source=user_profiles[name].source,\n            )\n\n        if user_profiles.active_name:\n            profiles.set_active(user_profiles.active_name, check=False)\n\n    return profiles\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.load_current_profile","title":"load_current_profile","text":"

    Load the current profile from the default and current profile paths.

    This will not include settings from the current settings context. Only settings that have been persisted to the profiles file will be saved.

    Source code in prefect/settings.py
    def load_current_profile():\n    \"\"\"\n    Load the current profile from the default and current profile paths.\n\n    This will _not_ include settings from the current settings context. Only settings\n    that have been persisted to the profiles file will be saved.\n    \"\"\"\n    from prefect.context import SettingsContext\n\n    profiles = load_profiles()\n    context = SettingsContext.get()\n\n    if context:\n        profiles.set_active(context.profile.name)\n\n    return profiles.active_profile\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.save_profiles","title":"save_profiles","text":"

    Writes all non-default profiles to the current profiles path.

    Source code in prefect/settings.py
    def save_profiles(profiles: ProfilesCollection) -> None:\n    \"\"\"\n    Writes all non-default profiles to the current profiles path.\n    \"\"\"\n    profiles_path = PREFECT_PROFILES_PATH.value()\n    profiles = profiles.without_profile_source(DEFAULT_PROFILES_PATH)\n    return _write_profiles_to(profiles_path, profiles)\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.load_profile","title":"load_profile","text":"

    Load a single profile by name.

    Source code in prefect/settings.py
    def load_profile(name: str) -> Profile:\n    \"\"\"\n    Load a single profile by name.\n    \"\"\"\n    profiles = load_profiles()\n    try:\n        return profiles[name]\n    except KeyError:\n        raise ValueError(f\"Profile {name!r} not found.\")\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/settings/#prefect.settings.update_current_profile","title":"update_current_profile","text":"

    Update the persisted data for the profile currently in-use.

    If the profile does not exist in the profiles file, it will be created.

    Given settings will be merged with the existing settings as described in ProfilesCollection.update_profile.

    Returns:

    Type Description Profile

    The new profile.

    Source code in prefect/settings.py
    def update_current_profile(settings: Dict[Union[str, Setting], Any]) -> Profile:\n    \"\"\"\n    Update the persisted data for the profile currently in-use.\n\n    If the profile does not exist in the profiles file, it will be created.\n\n    Given settings will be merged with the existing settings as described in\n    `ProfilesCollection.update_profile`.\n\n    Returns:\n        The new profile.\n    \"\"\"\n    import prefect.context\n\n    current_profile = prefect.context.get_settings_context().profile\n\n    if not current_profile:\n        raise MissingProfileError(\"No profile is currently in use.\")\n\n    profiles = load_profiles()\n\n    # Ensure the current profile's settings are present\n    profiles.update_profile(current_profile.name, current_profile.settings)\n    # Then merge the new settings in\n    new_profile = profiles.update_profile(current_profile.name, settings)\n\n    # Validate before saving\n    new_profile.validate_settings()\n\n    save_profiles(profiles)\n\n    return profiles[current_profile.name]\n
    ","tags":["Python API","settings","configuration","environment variables"]},{"location":"api-ref/prefect/software/","title":"prefect.software","text":"","tags":["Python API","software"]},{"location":"api-ref/prefect/software/#prefect.software","title":"prefect.software","text":"","tags":["Python API","software"]},{"location":"api-ref/prefect/states/","title":"prefect.states","text":"","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states","title":"prefect.states","text":"","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.AwaitingRetry","title":"AwaitingRetry","text":"

    Convenience function for creating AwaitingRetry states.

    Returns:

    Name Type Description State State

    a AwaitingRetry state

    Source code in prefect/states.py
    def AwaitingRetry(\n    cls: Type[State] = State, scheduled_time: datetime.datetime = None, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `AwaitingRetry` states.\n\n    Returns:\n        State: a AwaitingRetry state\n    \"\"\"\n    return Scheduled(\n        cls=cls, scheduled_time=scheduled_time, name=\"AwaitingRetry\", **kwargs\n    )\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Cancelled","title":"Cancelled","text":"

    Convenience function for creating Cancelled states.

    Returns:

    Name Type Description State State

    a Cancelled state

    Source code in prefect/states.py
    def Cancelled(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Cancelled` states.\n\n    Returns:\n        State: a Cancelled state\n    \"\"\"\n    return cls(type=StateType.CANCELLED, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Cancelling","title":"Cancelling","text":"

    Convenience function for creating Cancelling states.

    Returns:

    Name Type Description State State

    a Cancelling state

    Source code in prefect/states.py
    def Cancelling(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Cancelling` states.\n\n    Returns:\n        State: a Cancelling state\n    \"\"\"\n    return cls(type=StateType.CANCELLING, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Completed","title":"Completed","text":"

    Convenience function for creating Completed states.

    Returns:

    Name Type Description State State

    a Completed state

    Source code in prefect/states.py
    def Completed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Completed` states.\n\n    Returns:\n        State: a Completed state\n    \"\"\"\n    return cls(type=StateType.COMPLETED, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Crashed","title":"Crashed","text":"

    Convenience function for creating Crashed states.

    Returns:

    Name Type Description State State

    a Crashed state

    Source code in prefect/states.py
    def Crashed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Crashed` states.\n\n    Returns:\n        State: a Crashed state\n    \"\"\"\n    return cls(type=StateType.CRASHED, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Failed","title":"Failed","text":"

    Convenience function for creating Failed states.

    Returns:

    Name Type Description State State

    a Failed state

    Source code in prefect/states.py
    def Failed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Failed` states.\n\n    Returns:\n        State: a Failed state\n    \"\"\"\n    return cls(type=StateType.FAILED, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Late","title":"Late","text":"

    Convenience function for creating Late states.

    Returns:

    Name Type Description State State

    a Late state

    Source code in prefect/states.py
    def Late(\n    cls: Type[State] = State, scheduled_time: datetime.datetime = None, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `Late` states.\n\n    Returns:\n        State: a Late state\n    \"\"\"\n    return Scheduled(cls=cls, scheduled_time=scheduled_time, name=\"Late\", **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Paused","title":"Paused","text":"

    Convenience function for creating Paused states.

    Returns:

    Name Type Description State State

    a Paused state

    Source code in prefect/states.py
    def Paused(\n    cls: Type[State] = State,\n    timeout_seconds: Optional[int] = None,\n    pause_expiration_time: Optional[datetime.datetime] = None,\n    reschedule: bool = False,\n    pause_key: Optional[str] = None,\n    **kwargs,\n) -> State:\n    \"\"\"Convenience function for creating `Paused` states.\n\n    Returns:\n        State: a Paused state\n    \"\"\"\n    state_details = StateDetails.parse_obj(kwargs.pop(\"state_details\", {}))\n\n    if state_details.pause_timeout:\n        raise ValueError(\"An extra pause timeout was provided in state_details\")\n\n    if pause_expiration_time is not None and timeout_seconds is not None:\n        raise ValueError(\n            \"Cannot supply both a pause_expiration_time and timeout_seconds\"\n        )\n\n    if pause_expiration_time is None and timeout_seconds is None:\n        pass\n    else:\n        state_details.pause_timeout = pause_expiration_time or (\n            pendulum.now(\"UTC\") + pendulum.Duration(seconds=timeout_seconds)\n        )\n\n    state_details.pause_reschedule = reschedule\n    state_details.pause_key = pause_key\n\n    return cls(type=StateType.PAUSED, state_details=state_details, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Pending","title":"Pending","text":"

    Convenience function for creating Pending states.

    Returns:

    Name Type Description State State

    a Pending state

    Source code in prefect/states.py
    def Pending(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Pending` states.\n\n    Returns:\n        State: a Pending state\n    \"\"\"\n    return cls(type=StateType.PENDING, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Retrying","title":"Retrying","text":"

    Convenience function for creating Retrying states.

    Returns:

    Name Type Description State State

    a Retrying state

    Source code in prefect/states.py
    def Retrying(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Retrying` states.\n\n    Returns:\n        State: a Retrying state\n    \"\"\"\n    return cls(type=StateType.RUNNING, name=\"Retrying\", **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Running","title":"Running","text":"

    Convenience function for creating Running states.

    Returns:

    Name Type Description State State

    a Running state

    Source code in prefect/states.py
    def Running(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Running` states.\n\n    Returns:\n        State: a Running state\n    \"\"\"\n    return cls(type=StateType.RUNNING, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Scheduled","title":"Scheduled","text":"

    Convenience function for creating Scheduled states.

    Returns:

    Name Type Description State State

    a Scheduled state

    Source code in prefect/states.py
    def Scheduled(\n    cls: Type[State] = State, scheduled_time: datetime.datetime = None, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `Scheduled` states.\n\n    Returns:\n        State: a Scheduled state\n    \"\"\"\n    state_details = StateDetails.parse_obj(kwargs.pop(\"state_details\", {}))\n    if scheduled_time is None:\n        scheduled_time = pendulum.now(\"UTC\")\n    elif state_details.scheduled_time:\n        raise ValueError(\"An extra scheduled_time was provided in state_details\")\n    state_details.scheduled_time = scheduled_time\n\n    return cls(type=StateType.SCHEDULED, state_details=state_details, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.Suspended","title":"Suspended","text":"

    Convenience function for creating Suspended states.

    Returns:

    Name Type Description State

    a Suspended state

    Source code in prefect/states.py
    def Suspended(\n    cls: Type[State] = State,\n    timeout_seconds: Optional[int] = None,\n    pause_expiration_time: Optional[datetime.datetime] = None,\n    pause_key: Optional[str] = None,\n    **kwargs,\n):\n    \"\"\"Convenience function for creating `Suspended` states.\n\n    Returns:\n        State: a Suspended state\n    \"\"\"\n    return Paused(\n        cls=cls,\n        name=\"Suspended\",\n        reschedule=True,\n        timeout_seconds=timeout_seconds,\n        pause_expiration_time=pause_expiration_time,\n        pause_key=pause_key,\n        **kwargs,\n    )\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.exception_to_crashed_state","title":"exception_to_crashed_state async","text":"

    Takes an exception that occurs outside of user code and converts it to a 'Crash' exception with a 'Crashed' state.

    Source code in prefect/states.py
    async def exception_to_crashed_state(\n    exc: BaseException,\n    result_factory: Optional[ResultFactory] = None,\n) -> State:\n    \"\"\"\n    Takes an exception that occurs _outside_ of user code and converts it to a\n    'Crash' exception with a 'Crashed' state.\n    \"\"\"\n    state_message = None\n\n    if isinstance(exc, anyio.get_cancelled_exc_class()):\n        state_message = \"Execution was cancelled by the runtime environment.\"\n\n    elif isinstance(exc, KeyboardInterrupt):\n        state_message = \"Execution was aborted by an interrupt signal.\"\n\n    elif isinstance(exc, TerminationSignal):\n        state_message = \"Execution was aborted by a termination signal.\"\n\n    elif isinstance(exc, SystemExit):\n        state_message = \"Execution was aborted by Python system exit call.\"\n\n    elif isinstance(exc, (httpx.TimeoutException, httpx.ConnectError)):\n        try:\n            request: httpx.Request = exc.request\n        except RuntimeError:\n            # The request property is not set\n            state_message = (\n                \"Request failed while attempting to contact the server:\"\n                f\" {format_exception(exc)}\"\n            )\n        else:\n            # TODO: We can check if this is actually our API url\n            state_message = f\"Request to {request.url} failed: {format_exception(exc)}.\"\n\n    else:\n        state_message = (\n            \"Execution was interrupted by an unexpected exception:\"\n            f\" {format_exception(exc)}\"\n        )\n\n    if result_factory:\n        data = await result_factory.create_result(exc)\n    else:\n        # Attach the exception for local usage, will not be available when retrieved\n        # from the API\n        data = exc\n\n    return Crashed(message=state_message, data=data)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.exception_to_failed_state","title":"exception_to_failed_state async","text":"

    Convenience function for creating Failed states from exceptions

    Source code in prefect/states.py
    async def exception_to_failed_state(\n    exc: Optional[BaseException] = None,\n    result_factory: Optional[ResultFactory] = None,\n    **kwargs,\n) -> State:\n    \"\"\"\n    Convenience function for creating `Failed` states from exceptions\n    \"\"\"\n    if not exc:\n        _, exc, _ = sys.exc_info()\n        if exc is None:\n            raise ValueError(\n                \"Exception was not passed and no active exception could be found.\"\n            )\n    else:\n        pass\n\n    if result_factory:\n        data = await result_factory.create_result(exc)\n    else:\n        # Attach the exception for local usage, will not be available when retrieved\n        # from the API\n        data = exc\n\n    existing_message = kwargs.pop(\"message\", \"\")\n    if existing_message and not existing_message.endswith(\" \"):\n        existing_message += \" \"\n\n    # TODO: Consider if we want to include traceback information, it is intentionally\n    #       excluded from messages for now\n    message = existing_message + format_exception(exc)\n\n    return Failed(data=data, message=message, **kwargs)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.get_state_exception","title":"get_state_exception async","text":"

    If not given a FAILED or CRASHED state, this raise a value error.

    If the state result is a state, its exception will be returned.

    If the state result is an iterable of states, the exception of the first failure will be returned.

    If the state result is a string, a wrapper exception will be returned with the string as the message.

    If the state result is null, a wrapper exception will be returned with the state message attached.

    If the state result is not of a known type, a TypeError will be returned.

    When a wrapper exception is returned, the type will be: - FailedRun if the state type is FAILED. - CrashedRun if the state type is CRASHED. - CancelledRun if the state type is CANCELLED.

    Source code in prefect/states.py
    @sync_compatible\nasync def get_state_exception(state: State) -> BaseException:\n    \"\"\"\n    If not given a FAILED or CRASHED state, this raise a value error.\n\n    If the state result is a state, its exception will be returned.\n\n    If the state result is an iterable of states, the exception of the first failure\n    will be returned.\n\n    If the state result is a string, a wrapper exception will be returned with the\n    string as the message.\n\n    If the state result is null, a wrapper exception will be returned with the state\n    message attached.\n\n    If the state result is not of a known type, a `TypeError` will be returned.\n\n    When a wrapper exception is returned, the type will be:\n        - `FailedRun` if the state type is FAILED.\n        - `CrashedRun` if the state type is CRASHED.\n        - `CancelledRun` if the state type is CANCELLED.\n    \"\"\"\n\n    if state.is_failed():\n        wrapper = FailedRun\n        default_message = \"Run failed.\"\n    elif state.is_crashed():\n        wrapper = CrashedRun\n        default_message = \"Run crashed.\"\n    elif state.is_cancelled():\n        wrapper = CancelledRun\n        default_message = \"Run cancelled.\"\n    else:\n        raise ValueError(f\"Expected failed or crashed state got {state!r}.\")\n\n    if isinstance(state.data, BaseResult):\n        result = await state.data.get()\n    elif state.data is None:\n        result = None\n    else:\n        result = state.data\n\n    if result is None:\n        return wrapper(state.message or default_message)\n\n    if isinstance(result, Exception):\n        return result\n\n    elif isinstance(result, BaseException):\n        return result\n\n    elif isinstance(result, str):\n        return wrapper(result)\n\n    elif is_state(result):\n        # Return the exception from the inner state\n        return await get_state_exception(result)\n\n    elif is_state_iterable(result):\n        # Return the first failure\n        for state in result:\n            if state.is_failed() or state.is_crashed() or state.is_cancelled():\n                return await get_state_exception(state)\n\n        raise ValueError(\n            \"Failed state result was an iterable of states but none were failed.\"\n        )\n\n    else:\n        raise TypeError(\n            f\"Unexpected result for failed state: {result!r} \u2014\u2014 \"\n            f\"{type(result).__name__} cannot be resolved into an exception\"\n        )\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.get_state_result","title":"get_state_result","text":"

    Get the result from a state.

    See State.result()

    Source code in prefect/states.py
    def get_state_result(\n    state: State[R], raise_on_failure: bool = True, fetch: Optional[bool] = None\n) -> R:\n    \"\"\"\n    Get the result from a state.\n\n    See `State.result()`\n    \"\"\"\n\n    if fetch is None and (\n        PREFECT_ASYNC_FETCH_STATE_RESULT or not in_async_main_thread()\n    ):\n        # Fetch defaults to `True` for sync users or async users who have opted in\n        fetch = True\n\n    if not fetch:\n        if fetch is None and in_async_main_thread():\n            warnings.warn(\n                (\n                    \"State.result() was called from an async context but not awaited. \"\n                    \"This method will be updated to return a coroutine by default in \"\n                    \"the future. Pass `fetch=True` and `await` the call to get rid of \"\n                    \"this warning.\"\n                ),\n                DeprecationWarning,\n                stacklevel=2,\n            )\n        # Backwards compatibility\n        if isinstance(state.data, DataDocument):\n            return result_from_state_with_data_document(\n                state, raise_on_failure=raise_on_failure\n            )\n        else:\n            return state.data\n    else:\n        return _get_state_result(state, raise_on_failure=raise_on_failure)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.is_state","title":"is_state","text":"

    Check if the given object is a state instance

    Source code in prefect/states.py
    def is_state(obj: Any) -> TypeGuard[State]:\n    \"\"\"\n    Check if the given object is a state instance\n    \"\"\"\n    # We may want to narrow this to client-side state types but for now this provides\n    # backwards compatibility\n    try:\n        from prefect.server.schemas.states import State as State_\n\n        classes_ = (State, State_)\n    except ImportError:\n        classes_ = State\n\n    # return isinstance(obj, (State, State_))\n    return isinstance(obj, classes_)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.is_state_iterable","title":"is_state_iterable","text":"

    Check if a the given object is an iterable of states types

    Supported iterables are: - set - list - tuple

    Other iterables will return False even if they contain states.

    Source code in prefect/states.py
    def is_state_iterable(obj: Any) -> TypeGuard[Iterable[State]]:\n    \"\"\"\n    Check if a the given object is an iterable of states types\n\n    Supported iterables are:\n    - set\n    - list\n    - tuple\n\n    Other iterables will return `False` even if they contain states.\n    \"\"\"\n    # We do not check for arbitrary iterables because this is not intended to be used\n    # for things like dictionaries, dataframes, or pydantic models\n    if (\n        not isinstance(obj, BaseAnnotation)\n        and isinstance(obj, (list, set, tuple))\n        and obj\n    ):\n        return all([is_state(o) for o in obj])\n    else:\n        return False\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.raise_state_exception","title":"raise_state_exception async","text":"

    Given a FAILED or CRASHED state, raise the contained exception.

    Source code in prefect/states.py
    @sync_compatible\nasync def raise_state_exception(state: State) -> None:\n    \"\"\"\n    Given a FAILED or CRASHED state, raise the contained exception.\n    \"\"\"\n    if not (state.is_failed() or state.is_crashed() or state.is_cancelled()):\n        return None\n\n    raise await get_state_exception(state)\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/states/#prefect.states.return_value_to_state","title":"return_value_to_state async","text":"

    Given a return value from a user's function, create a State the run should be placed in.

    • If data is returned, we create a 'COMPLETED' state with the data
    • If a single, manually created state is returned, we use that state as given (manual creation is determined by the lack of ids)
    • If an upstream state or iterable of upstream states is returned, we apply the aggregate rule

    The aggregate rule says that given multiple states we will determine the final state such that:

    • If any states are not COMPLETED the final state is FAILED
    • If all of the states are COMPLETED the final state is COMPLETED
    • The states will be placed in the final state data attribute

    Callers should resolve all futures into states before passing return values to this function.

    Source code in prefect/states.py
    async def return_value_to_state(retval: R, result_factory: ResultFactory) -> State[R]:\n    \"\"\"\n    Given a return value from a user's function, create a `State` the run should\n    be placed in.\n\n    - If data is returned, we create a 'COMPLETED' state with the data\n    - If a single, manually created state is returned, we use that state as given\n        (manual creation is determined by the lack of ids)\n    - If an upstream state or iterable of upstream states is returned, we apply the\n        aggregate rule\n\n    The aggregate rule says that given multiple states we will determine the final state\n    such that:\n\n    - If any states are not COMPLETED the final state is FAILED\n    - If all of the states are COMPLETED the final state is COMPLETED\n    - The states will be placed in the final state `data` attribute\n\n    Callers should resolve all futures into states before passing return values to this\n    function.\n    \"\"\"\n\n    if (\n        is_state(retval)\n        # Check for manual creation\n        and not retval.state_details.flow_run_id\n        and not retval.state_details.task_run_id\n    ):\n        state = retval\n\n        # Do not modify states with data documents attached; backwards compatibility\n        if isinstance(state.data, DataDocument):\n            return state\n\n        # Unless the user has already constructed a result explicitly, use the factory\n        # to update the data to the correct type\n        if not isinstance(state.data, BaseResult):\n            state.data = await result_factory.create_result(state.data)\n\n        return state\n\n    # Determine a new state from the aggregate of contained states\n    if is_state(retval) or is_state_iterable(retval):\n        states = StateGroup(ensure_iterable(retval))\n\n        # Determine the new state type\n        if states.all_completed():\n            new_state_type = StateType.COMPLETED\n        elif states.any_cancelled():\n            new_state_type = StateType.CANCELLED\n        elif states.any_paused():\n            new_state_type = StateType.PAUSED\n        else:\n            new_state_type = StateType.FAILED\n\n        # Generate a nice message for the aggregate\n        if states.all_completed():\n            message = \"All states completed.\"\n        elif states.any_cancelled():\n            message = f\"{states.cancelled_count}/{states.total_count} states cancelled.\"\n        elif states.any_paused():\n            message = f\"{states.paused_count}/{states.total_count} states paused.\"\n        elif states.any_failed():\n            message = f\"{states.fail_count}/{states.total_count} states failed.\"\n        elif not states.all_final():\n            message = (\n                f\"{states.not_final_count}/{states.total_count} states are not final.\"\n            )\n        else:\n            message = \"Given states: \" + states.counts_message()\n\n        # TODO: We may actually want to set the data to a `StateGroup` object and just\n        #       allow it to be unpacked into a tuple and such so users can interact with\n        #       it\n        return State(\n            type=new_state_type,\n            message=message,\n            data=await result_factory.create_result(retval),\n        )\n\n    # Generators aren't portable, implicitly convert them to a list.\n    if isinstance(retval, GeneratorType):\n        data = list(retval)\n    else:\n        data = retval\n\n    # Otherwise, they just gave data and this is a completed retval\n    return Completed(data=await result_factory.create_result(data))\n
    ","tags":["Python API","states"]},{"location":"api-ref/prefect/task-runners/","title":"prefect.task_runners","text":"","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners","title":"prefect.task_runners","text":"

    Interface and implementations of various task runners.

    Task Runners in Prefect are responsible for managing the execution of Prefect task runs. Generally speaking, users are not expected to interact with task runners outside of configuring and initializing them for a flow.

    Example
    >>> from prefect import flow, task\n>>> from prefect.task_runners import SequentialTaskRunner\n>>> from typing import List\n>>>\n>>> @task\n>>> def say_hello(name):\n...     print(f\"hello {name}\")\n>>>\n>>> @task\n>>> def say_goodbye(name):\n...     print(f\"goodbye {name}\")\n>>>\n>>> @flow(task_runner=SequentialTaskRunner())\n>>> def greetings(names: List[str]):\n...     for name in names:\n...         say_hello(name)\n...         say_goodbye(name)\n>>>\n>>> greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\nhello arthur\ngoodbye arthur\nhello trillian\ngoodbye trillian\nhello ford\ngoodbye ford\nhello marvin\ngoodbye marvin\n

    Switching to a DaskTaskRunner:

    >>> from prefect_dask.task_runners import DaskTaskRunner\n>>> flow.task_runner = DaskTaskRunner()\n>>> greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\nhello arthur\ngoodbye arthur\nhello trillian\nhello ford\ngoodbye marvin\nhello marvin\ngoodbye ford\ngoodbye trillian\n

    For usage details, see the Task Runners documentation.

    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner","title":"BaseTaskRunner","text":"Source code in prefect/task_runners.py
    class BaseTaskRunner(metaclass=abc.ABCMeta):\n    def __init__(self) -> None:\n        self.logger = get_logger(f\"task_runner.{self.name}\")\n        self._started: bool = False\n\n    @property\n    @abc.abstractmethod\n    def concurrency_type(self) -> TaskConcurrencyType:\n        pass  # noqa\n\n    @property\n    def name(self):\n        return type(self).__name__.lower().replace(\"taskrunner\", \"\")\n\n    def duplicate(self):\n        \"\"\"\n        Return a new task runner instance with the same options.\n        \"\"\"\n        # The base class returns `NotImplemented` to indicate that this is not yet\n        # implemented by a given task runner.\n        return NotImplemented\n\n    def __eq__(self, other: object) -> bool:\n        \"\"\"\n        Returns true if the task runners use the same options.\n        \"\"\"\n        if type(other) == type(self) and (\n            # Compare public attributes for naive equality check\n            # Subclasses should implement this method with a check init option equality\n            {k: v for k, v in self.__dict__.items() if not k.startswith(\"_\")}\n            == {k: v for k, v in other.__dict__.items() if not k.startswith(\"_\")}\n        ):\n            return True\n        else:\n            return NotImplemented\n\n    @abc.abstractmethod\n    async def submit(\n        self,\n        key: UUID,\n        call: Callable[..., Awaitable[State[R]]],\n    ) -> None:\n        \"\"\"\n        Submit a call for execution and return a `PrefectFuture` that can be used to\n        get the call result.\n\n        Args:\n            task_run: The task run being submitted.\n            task_key: A unique key for this orchestration run of the task. Can be used\n                for caching.\n            call: The function to be executed\n            run_kwargs: A dict of keyword arguments to pass to `call`\n\n        Returns:\n            A future representing the result of `call` execution\n        \"\"\"\n        raise NotImplementedError()\n\n    @abc.abstractmethod\n    async def wait(self, key: UUID, timeout: float = None) -> Optional[State]:\n        \"\"\"\n        Given a `PrefectFuture`, wait for its return state up to `timeout` seconds.\n        If it is not finished after the timeout expires, `None` should be returned.\n\n        Implementers should be careful to ensure that this function never returns or\n        raises an exception.\n        \"\"\"\n        raise NotImplementedError()\n\n    @asynccontextmanager\n    async def start(\n        self: T,\n    ) -> AsyncIterator[T]:\n        \"\"\"\n        Start the task runner, preparing any resources necessary for task submission.\n\n        Children should implement `_start` to prepare and clean up resources.\n\n        Yields:\n            The prepared task runner\n        \"\"\"\n        if self._started:\n            raise RuntimeError(\"The task runner is already started!\")\n\n        async with AsyncExitStack() as exit_stack:\n            self.logger.debug(\"Starting task runner...\")\n            try:\n                await self._start(exit_stack)\n                self._started = True\n                yield self\n            finally:\n                self.logger.debug(\"Shutting down task runner...\")\n                self._started = False\n\n    async def _start(self, exit_stack: AsyncExitStack) -> None:\n        \"\"\"\n        Create any resources required for this task runner to submit work.\n\n        Cleanup of resources should be submitted to the `exit_stack`.\n        \"\"\"\n        pass  # noqa\n\n    def __str__(self) -> str:\n        return type(self).__name__\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner.duplicate","title":"duplicate","text":"

    Return a new task runner instance with the same options.

    Source code in prefect/task_runners.py
    def duplicate(self):\n    \"\"\"\n    Return a new task runner instance with the same options.\n    \"\"\"\n    # The base class returns `NotImplemented` to indicate that this is not yet\n    # implemented by a given task runner.\n    return NotImplemented\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner.start","title":"start async","text":"

    Start the task runner, preparing any resources necessary for task submission.

    Children should implement _start to prepare and clean up resources.

    Yields:

    Type Description AsyncIterator[T]

    The prepared task runner

    Source code in prefect/task_runners.py
    @asynccontextmanager\nasync def start(\n    self: T,\n) -> AsyncIterator[T]:\n    \"\"\"\n    Start the task runner, preparing any resources necessary for task submission.\n\n    Children should implement `_start` to prepare and clean up resources.\n\n    Yields:\n        The prepared task runner\n    \"\"\"\n    if self._started:\n        raise RuntimeError(\"The task runner is already started!\")\n\n    async with AsyncExitStack() as exit_stack:\n        self.logger.debug(\"Starting task runner...\")\n        try:\n            await self._start(exit_stack)\n            self._started = True\n            yield self\n        finally:\n            self.logger.debug(\"Shutting down task runner...\")\n            self._started = False\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner.submit","title":"submit abstractmethod async","text":"

    Submit a call for execution and return a PrefectFuture that can be used to get the call result.

    Parameters:

    Name Type Description Default task_run

    The task run being submitted.

    required task_key

    A unique key for this orchestration run of the task. Can be used for caching.

    required call Callable[..., Awaitable[State[R]]]

    The function to be executed

    required run_kwargs

    A dict of keyword arguments to pass to call

    required

    Returns:

    Type Description None

    A future representing the result of call execution

    Source code in prefect/task_runners.py
    @abc.abstractmethod\nasync def submit(\n    self,\n    key: UUID,\n    call: Callable[..., Awaitable[State[R]]],\n) -> None:\n    \"\"\"\n    Submit a call for execution and return a `PrefectFuture` that can be used to\n    get the call result.\n\n    Args:\n        task_run: The task run being submitted.\n        task_key: A unique key for this orchestration run of the task. Can be used\n            for caching.\n        call: The function to be executed\n        run_kwargs: A dict of keyword arguments to pass to `call`\n\n    Returns:\n        A future representing the result of `call` execution\n    \"\"\"\n    raise NotImplementedError()\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.BaseTaskRunner.wait","title":"wait abstractmethod async","text":"

    Given a PrefectFuture, wait for its return state up to timeout seconds. If it is not finished after the timeout expires, None should be returned.

    Implementers should be careful to ensure that this function never returns or raises an exception.

    Source code in prefect/task_runners.py
    @abc.abstractmethod\nasync def wait(self, key: UUID, timeout: float = None) -> Optional[State]:\n    \"\"\"\n    Given a `PrefectFuture`, wait for its return state up to `timeout` seconds.\n    If it is not finished after the timeout expires, `None` should be returned.\n\n    Implementers should be careful to ensure that this function never returns or\n    raises an exception.\n    \"\"\"\n    raise NotImplementedError()\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.ConcurrentTaskRunner","title":"ConcurrentTaskRunner","text":"

    Bases: BaseTaskRunner

    A concurrent task runner that allows tasks to switch when blocking on IO. Synchronous tasks will be submitted to a thread pool maintained by anyio.

    Example
    Using a thread for concurrency:\n>>> from prefect import flow\n>>> from prefect.task_runners import ConcurrentTaskRunner\n>>> @flow(task_runner=ConcurrentTaskRunner)\n>>> def my_flow():\n>>>     ...\n
    Source code in prefect/task_runners.py
    class ConcurrentTaskRunner(BaseTaskRunner):\n    \"\"\"\n    A concurrent task runner that allows tasks to switch when blocking on IO.\n    Synchronous tasks will be submitted to a thread pool maintained by `anyio`.\n\n    Example:\n        ```\n        Using a thread for concurrency:\n        >>> from prefect import flow\n        >>> from prefect.task_runners import ConcurrentTaskRunner\n        >>> @flow(task_runner=ConcurrentTaskRunner)\n        >>> def my_flow():\n        >>>     ...\n        ```\n    \"\"\"\n\n    def __init__(self):\n        # TODO: Consider adding `max_workers` support using anyio capacity limiters\n\n        # Runtime attributes\n        self._task_group: anyio.abc.TaskGroup = None\n        self._result_events: Dict[UUID, Event] = {}\n        self._results: Dict[UUID, Any] = {}\n        self._keys: Set[UUID] = set()\n\n        super().__init__()\n\n    @property\n    def concurrency_type(self) -> TaskConcurrencyType:\n        return TaskConcurrencyType.CONCURRENT\n\n    def duplicate(self):\n        return type(self)()\n\n    async def submit(\n        self,\n        key: UUID,\n        call: Callable[[], Awaitable[State[R]]],\n    ) -> None:\n        if not self._started:\n            raise RuntimeError(\n                \"The task runner must be started before submitting work.\"\n            )\n\n        if not self._task_group:\n            raise RuntimeError(\n                \"The concurrent task runner cannot be used to submit work after \"\n                \"serialization.\"\n            )\n\n        # Create an event to set on completion\n        self._result_events[key] = Event()\n\n        # Rely on the event loop for concurrency\n        self._task_group.start_soon(self._run_and_store_result, key, call)\n\n    async def wait(\n        self,\n        key: UUID,\n        timeout: float = None,\n    ) -> Optional[State]:\n        if not self._task_group:\n            raise RuntimeError(\n                \"The concurrent task runner cannot be used to wait for work after \"\n                \"serialization.\"\n            )\n\n        return await self._get_run_result(key, timeout)\n\n    async def _run_and_store_result(\n        self, key: UUID, call: Callable[[], Awaitable[State[R]]]\n    ):\n        \"\"\"\n        Simple utility to store the orchestration result in memory on completion\n\n        Since this run is occurring on the main thread, we capture exceptions to prevent\n        task crashes from crashing the flow run.\n        \"\"\"\n        try:\n            result = await call()\n        except BaseException as exc:\n            result = await exception_to_crashed_state(exc)\n\n        self._results[key] = result\n        self._result_events[key].set()\n\n    async def _get_run_result(\n        self, key: UUID, timeout: float = None\n    ) -> Optional[State]:\n        \"\"\"\n        Block until the run result has been populated.\n        \"\"\"\n        result = None  # retval on timeout\n\n        # Note we do not use `asyncio.wrap_future` and instead use an `Event` to avoid\n        # stdlib behavior where the wrapped future is cancelled if the parent future is\n        # cancelled (as it would be during a timeout here)\n        with anyio.move_on_after(timeout):\n            await self._result_events[key].wait()\n            result = self._results[key]\n\n        return result  # timeout reached\n\n    async def _start(self, exit_stack: AsyncExitStack):\n        \"\"\"\n        Start the process pool\n        \"\"\"\n        self._task_group = await exit_stack.enter_async_context(\n            anyio.create_task_group()\n        )\n\n    def __getstate__(self):\n        \"\"\"\n        Allow the `ConcurrentTaskRunner` to be serialized by dropping the task group.\n        \"\"\"\n        data = self.__dict__.copy()\n        data.update({k: None for k in {\"_task_group\"}})\n        return data\n\n    def __setstate__(self, data: dict):\n        \"\"\"\n        When deserialized, we will no longer have a reference to the task group.\n        \"\"\"\n        self.__dict__.update(data)\n        self._task_group = None\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/task-runners/#prefect.task_runners.SequentialTaskRunner","title":"SequentialTaskRunner","text":"

    Bases: BaseTaskRunner

    A simple task runner that executes calls as they are submitted.

    If writing synchronous tasks, this runner will always execute tasks sequentially. If writing async tasks, this runner will execute tasks sequentially unless grouped using anyio.create_task_group or asyncio.gather.

    Source code in prefect/task_runners.py
    class SequentialTaskRunner(BaseTaskRunner):\n    \"\"\"\n    A simple task runner that executes calls as they are submitted.\n\n    If writing synchronous tasks, this runner will always execute tasks sequentially.\n    If writing async tasks, this runner will execute tasks sequentially unless grouped\n    using `anyio.create_task_group` or `asyncio.gather`.\n    \"\"\"\n\n    def __init__(self) -> None:\n        super().__init__()\n        self._results: Dict[str, State] = {}\n\n    @property\n    def concurrency_type(self) -> TaskConcurrencyType:\n        return TaskConcurrencyType.SEQUENTIAL\n\n    def duplicate(self):\n        return type(self)()\n\n    async def submit(\n        self,\n        key: UUID,\n        call: Callable[..., Awaitable[State[R]]],\n    ) -> None:\n        # Run the function immediately and store the result in memory\n        try:\n            result = await call()\n        except BaseException as exc:\n            result = await exception_to_crashed_state(exc)\n\n        self._results[key] = result\n\n    async def wait(self, key: UUID, timeout: float = None) -> Optional[State]:\n        return self._results[key]\n
    ","tags":["Python API","tasks","task runners","Dask","Ray"]},{"location":"api-ref/prefect/tasks/","title":"prefect.tasks","text":"","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks","title":"prefect.tasks","text":"

    Module containing the base workflow task class and decorator - for most use cases, using the @task decorator is preferred.

    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.Task","title":"Task","text":"

    Bases: Generic[P, R]

    A Prefect task definition.

    Note

    We recommend using the @task decorator for most use-cases.

    Wraps a function with an entrypoint to the Prefect engine. Calling this class within a flow function creates a new task run.

    To preserve the input and output types, we use the generic type variables P and R for \"Parameters\" and \"Returns\" respectively.

    Parameters:

    Name Type Description Default fn Callable[P, R]

    The function defining the task.

    required name str

    An optional name for the task; if not provided, the name will be inferred from the given function.

    None description str

    An optional string description for the task.

    None tags Iterable[str]

    An optional set of tags to be associated with runs of this task. These tags are combined with any tags defined by a prefect.tags context at task runtime.

    None version str

    An optional string specifying the version of this task definition

    None cache_key_fn Callable[[TaskRunContext, Dict[str, Any]], Optional[str]]

    An optional callable that, given the task run context and call parameters, generates a string key; if the key matches a previous completed state, that state result will be restored instead of running the task again.

    None cache_expiration timedelta

    An optional amount of time indicating how long cached states for this task should be restorable; if not provided, cached states will never expire.

    None task_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this task; this name can be provided as a string template with the task's keyword arguments as variables, or a function that returns a string.

    None retries Optional[int]

    An optional number of times to retry on task run failure.

    None retry_delay_seconds Optional[Union[float, int, List[float], Callable[[int], List[float]]]]

    Optionally configures how long to wait before retrying the task after failure. This is only applicable if retries is nonzero. This setting can either be a number of seconds, a list of retry delays, or a callable that, given the total number of retries, generates a list of retry delays. If a number of seconds, that delay will be applied to all retries. If a list, each retry will wait for the corresponding delay before retrying. When passing a callable or a list, the number of configured retry delays cannot exceed 50.

    None retry_jitter_factor Optional[float]

    An optional factor that defines the factor to which a retry can be jittered in order to avoid a \"thundering herd\".

    None persist_result Optional[bool]

    An optional toggle indicating whether the result of this task should be persisted to result storage. Defaults to None, which indicates that Prefect should choose whether the result should be persisted depending on the features being used.

    None result_storage Optional[ResultStorage]

    An optional block to use to persist the result of this task. Defaults to the value set in the flow the task is called in.

    None result_storage_key Optional[str]

    An optional key to store the result in storage at when persisted. Defaults to a unique identifier.

    None result_serializer Optional[ResultSerializer]

    An optional serializer to use to serialize the result of this task for persistence. Defaults to the value set in the flow the task is called in.

    None timeout_seconds Union[int, float]

    An optional number of seconds indicating a maximum runtime for the task. If the task exceeds this runtime, it will be marked as failed.

    None log_prints Optional[bool]

    If set, print statements in the task will be redirected to the Prefect logger for the task run. Defaults to None, which indicates that the value from the flow should be used.

    False refresh_cache Optional[bool]

    If set, cached results for the cache key are not used. Defaults to None, which indicates that a cached result from a previous execution with matching cache key is used.

    None on_failure Optional[List[Callable[[Task, TaskRun, State], None]]]

    An optional list of callables to run when the task enters a failed state.

    None on_completion Optional[List[Callable[[Task, TaskRun, State], None]]]

    An optional list of callables to run when the task enters a completed state.

    None retry_condition_fn Optional[Callable[[Task, TaskRun, State], bool]]

    An optional callable run when a task run returns a Failed state. Should return True if the task should continue to its retry policy (e.g. retries=3), and False if the task should end as failed. Defaults to None, indicating the task should always continue to its retry policy.

    None viz_return_value Optional[Any]

    An optional value to return when the task dependency tree is visualized.

    None Source code in prefect/tasks.py
    @PrefectObjectRegistry.register_instances\nclass Task(Generic[P, R]):\n    \"\"\"\n    A Prefect task definition.\n\n    !!! note\n        We recommend using [the `@task` decorator][prefect.tasks.task] for most use-cases.\n\n    Wraps a function with an entrypoint to the Prefect engine. Calling this class within a flow function\n    creates a new task run.\n\n    To preserve the input and output types, we use the generic type variables P and R for \"Parameters\" and\n    \"Returns\" respectively.\n\n    Args:\n        fn: The function defining the task.\n        name: An optional name for the task; if not provided, the name will be inferred\n            from the given function.\n        description: An optional string description for the task.\n        tags: An optional set of tags to be associated with runs of this task. These\n            tags are combined with any tags defined by a `prefect.tags` context at\n            task runtime.\n        version: An optional string specifying the version of this task definition\n        cache_key_fn: An optional callable that, given the task run context and call\n            parameters, generates a string key; if the key matches a previous completed\n            state, that state result will be restored instead of running the task again.\n        cache_expiration: An optional amount of time indicating how long cached states\n            for this task should be restorable; if not provided, cached states will\n            never expire.\n        task_run_name: An optional name to distinguish runs of this task; this name can be provided\n            as a string template with the task's keyword arguments as variables,\n            or a function that returns a string.\n        retries: An optional number of times to retry on task run failure.\n        retry_delay_seconds: Optionally configures how long to wait before retrying the\n            task after failure. This is only applicable if `retries` is nonzero. This\n            setting can either be a number of seconds, a list of retry delays, or a\n            callable that, given the total number of retries, generates a list of retry\n            delays. If a number of seconds, that delay will be applied to all retries.\n            If a list, each retry will wait for the corresponding delay before retrying.\n            When passing a callable or a list, the number of configured retry delays\n            cannot exceed 50.\n        retry_jitter_factor: An optional factor that defines the factor to which a retry\n            can be jittered in order to avoid a \"thundering herd\".\n        persist_result: An optional toggle indicating whether the result of this task\n            should be persisted to result storage. Defaults to `None`, which indicates\n            that Prefect should choose whether the result should be persisted depending on\n            the features being used.\n        result_storage: An optional block to use to persist the result of this task.\n            Defaults to the value set in the flow the task is called in.\n        result_storage_key: An optional key to store the result in storage at when persisted.\n            Defaults to a unique identifier.\n        result_serializer: An optional serializer to use to serialize the result of this\n            task for persistence. Defaults to the value set in the flow the task is\n            called in.\n        timeout_seconds: An optional number of seconds indicating a maximum runtime for\n            the task. If the task exceeds this runtime, it will be marked as failed.\n        log_prints: If set, `print` statements in the task will be redirected to the\n            Prefect logger for the task run. Defaults to `None`, which indicates\n            that the value from the flow should be used.\n        refresh_cache: If set, cached results for the cache key are not used.\n            Defaults to `None`, which indicates that a cached result from a previous\n            execution with matching cache key is used.\n        on_failure: An optional list of callables to run when the task enters a failed state.\n        on_completion: An optional list of callables to run when the task enters a completed state.\n        retry_condition_fn: An optional callable run when a task run returns a Failed state. Should\n            return `True` if the task should continue to its retry policy (e.g. `retries=3`), and `False` if the task\n            should end as failed. Defaults to `None`, indicating the task should always continue\n            to its retry policy.\n        viz_return_value: An optional value to return when the task dependency tree is visualized.\n    \"\"\"\n\n    # NOTE: These parameters (types, defaults, and docstrings) should be duplicated\n    #       exactly in the @task decorator\n    def __init__(\n        self,\n        fn: Callable[P, R],\n        name: str = None,\n        description: str = None,\n        tags: Iterable[str] = None,\n        version: str = None,\n        cache_key_fn: Callable[\n            [\"TaskRunContext\", Dict[str, Any]], Optional[str]\n        ] = None,\n        cache_expiration: datetime.timedelta = None,\n        task_run_name: Optional[Union[Callable[[], str], str]] = None,\n        retries: Optional[int] = None,\n        retry_delay_seconds: Optional[\n            Union[\n                float,\n                int,\n                List[float],\n                Callable[[int], List[float]],\n            ]\n        ] = None,\n        retry_jitter_factor: Optional[float] = None,\n        persist_result: Optional[bool] = None,\n        result_storage: Optional[ResultStorage] = None,\n        result_serializer: Optional[ResultSerializer] = None,\n        result_storage_key: Optional[str] = None,\n        cache_result_in_memory: bool = True,\n        timeout_seconds: Union[int, float] = None,\n        log_prints: Optional[bool] = False,\n        refresh_cache: Optional[bool] = None,\n        on_completion: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n        on_failure: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n        retry_condition_fn: Optional[Callable[[\"Task\", TaskRun, State], bool]] = None,\n        viz_return_value: Optional[Any] = None,\n    ):\n        # Validate if hook passed is list and contains callables\n        hook_categories = [on_completion, on_failure]\n        hook_names = [\"on_completion\", \"on_failure\"]\n        for hooks, hook_name in zip(hook_categories, hook_names):\n            if hooks is not None:\n                if not hooks:\n                    raise ValueError(f\"Empty list passed for '{hook_name}'\")\n                try:\n                    hooks = list(hooks)\n                except TypeError:\n                    raise TypeError(\n                        f\"Expected iterable for '{hook_name}'; got\"\n                        f\" {type(hooks).__name__} instead. Please provide a list of\"\n                        f\" hooks to '{hook_name}':\\n\\n\"\n                        f\"@flow({hook_name}=[hook1, hook2])\\ndef\"\n                        \" my_flow():\\n\\tpass\"\n                    )\n\n                for hook in hooks:\n                    if not callable(hook):\n                        raise TypeError(\n                            f\"Expected callables in '{hook_name}'; got\"\n                            f\" {type(hook).__name__} instead. Please provide a list of\"\n                            f\" hooks to '{hook_name}':\\n\\n\"\n                            f\"@flow({hook_name}=[hook1, hook2])\\ndef\"\n                            \" my_flow():\\n\\tpass\"\n                        )\n\n        if not callable(fn):\n            raise TypeError(\"'fn' must be callable\")\n\n        self.description = description or inspect.getdoc(fn)\n        update_wrapper(self, fn)\n        self.fn = fn\n        self.isasync = inspect.iscoroutinefunction(self.fn)\n\n        if not name:\n            if not hasattr(self.fn, \"__name__\"):\n                self.name = type(self.fn).__name__\n            else:\n                self.name = self.fn.__name__\n        else:\n            self.name = name\n\n        if task_run_name is not None:\n            if not isinstance(task_run_name, str) and not callable(task_run_name):\n                raise TypeError(\n                    \"Expected string or callable for 'task_run_name'; got\"\n                    f\" {type(task_run_name).__name__} instead.\"\n                )\n        self.task_run_name = task_run_name\n\n        self.version = version\n        self.log_prints = log_prints\n\n        raise_for_reserved_arguments(self.fn, [\"return_state\", \"wait_for\"])\n\n        self.tags = set(tags if tags else [])\n\n        if not hasattr(self.fn, \"__qualname__\"):\n            self.task_key = to_qualified_name(type(self.fn))\n        else:\n            self.task_key = to_qualified_name(self.fn)\n\n        self.cache_key_fn = cache_key_fn\n        self.cache_expiration = cache_expiration\n        self.refresh_cache = refresh_cache\n\n        # TaskRunPolicy settings\n        # TODO: We can instantiate a `TaskRunPolicy` and add Pydantic bound checks to\n        #       validate that the user passes positive numbers here\n\n        self.retries = (\n            retries if retries is not None else PREFECT_TASK_DEFAULT_RETRIES.value()\n        )\n        if retry_delay_seconds is None:\n            retry_delay_seconds = PREFECT_TASK_DEFAULT_RETRY_DELAY_SECONDS.value()\n\n        if callable(retry_delay_seconds):\n            self.retry_delay_seconds = retry_delay_seconds(retries)\n        else:\n            self.retry_delay_seconds = retry_delay_seconds\n\n        if isinstance(self.retry_delay_seconds, list) and (\n            len(self.retry_delay_seconds) > 50\n        ):\n            raise ValueError(\"Can not configure more than 50 retry delays per task.\")\n\n        if retry_jitter_factor is not None and retry_jitter_factor < 0:\n            raise ValueError(\"`retry_jitter_factor` must be >= 0.\")\n\n        self.retry_jitter_factor = retry_jitter_factor\n        self.persist_result = persist_result\n        self.result_storage = result_storage\n        self.result_serializer = result_serializer\n        self.result_storage_key = result_storage_key\n        self.cache_result_in_memory = cache_result_in_memory\n        self.timeout_seconds = float(timeout_seconds) if timeout_seconds else None\n        # Warn if this task's `name` conflicts with another task while having a\n        # different function. This is to detect the case where two or more tasks\n        # share a name or are lambdas, which should result in a warning, and to\n        # differentiate it from the case where the task was 'copied' via\n        # `with_options`, which should not result in a warning.\n        registry = PrefectObjectRegistry.get()\n\n        if registry and any(\n            other\n            for other in registry.get_instances(Task)\n            if other.name == self.name and id(other.fn) != id(self.fn)\n        ):\n            try:\n                file = inspect.getsourcefile(self.fn)\n                line_number = inspect.getsourcelines(self.fn)[1]\n            except TypeError:\n                file = \"unknown\"\n                line_number = \"unknown\"\n\n            warnings.warn(\n                f\"A task named {self.name!r} and defined at '{file}:{line_number}' \"\n                \"conflicts with another task. Consider specifying a unique `name` \"\n                \"parameter in the task definition:\\n\\n \"\n                \"`@task(name='my_unique_name', ...)`\"\n            )\n        self.on_completion = on_completion\n        self.on_failure = on_failure\n\n        # retry_condition_fn must be a callable or None. If it is neither, raise a TypeError\n        if retry_condition_fn is not None and not (callable(retry_condition_fn)):\n            raise TypeError(\n                \"Expected `retry_condition_fn` to be callable, got\"\n                f\" {type(retry_condition_fn).__name__} instead.\"\n            )\n\n        self.retry_condition_fn = retry_condition_fn\n        self.viz_return_value = viz_return_value\n\n    def with_options(\n        self,\n        *,\n        name: str = None,\n        description: str = None,\n        tags: Iterable[str] = None,\n        cache_key_fn: Callable[\n            [\"TaskRunContext\", Dict[str, Any]], Optional[str]\n        ] = None,\n        task_run_name: Optional[Union[Callable[[], str], str]] = None,\n        cache_expiration: datetime.timedelta = None,\n        retries: Optional[int] = NotSet,\n        retry_delay_seconds: Union[\n            float,\n            int,\n            List[float],\n            Callable[[int], List[float]],\n        ] = NotSet,\n        retry_jitter_factor: Optional[float] = NotSet,\n        persist_result: Optional[bool] = NotSet,\n        result_storage: Optional[ResultStorage] = NotSet,\n        result_serializer: Optional[ResultSerializer] = NotSet,\n        result_storage_key: Optional[str] = NotSet,\n        cache_result_in_memory: Optional[bool] = None,\n        timeout_seconds: Union[int, float] = None,\n        log_prints: Optional[bool] = NotSet,\n        refresh_cache: Optional[bool] = NotSet,\n        on_completion: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n        on_failure: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n        retry_condition_fn: Optional[Callable[[\"Task\", TaskRun, State], bool]] = None,\n        viz_return_value: Optional[Any] = None,\n    ):\n        \"\"\"\n        Create a new task from the current object, updating provided options.\n\n        Args:\n            name: A new name for the task.\n            description: A new description for the task.\n            tags: A new set of tags for the task. If given, existing tags are ignored,\n                not merged.\n            cache_key_fn: A new cache key function for the task.\n            cache_expiration: A new cache expiration time for the task.\n            task_run_name: An optional name to distinguish runs of this task; this name can be provided\n                as a string template with the task's keyword arguments as variables,\n                or a function that returns a string.\n            retries: A new number of times to retry on task run failure.\n            retry_delay_seconds: Optionally configures how long to wait before retrying\n                the task after failure. This is only applicable if `retries` is nonzero.\n                This setting can either be a number of seconds, a list of retry delays,\n                or a callable that, given the total number of retries, generates a list\n                of retry delays. If a number of seconds, that delay will be applied to\n                all retries. If a list, each retry will wait for the corresponding delay\n                before retrying. When passing a callable or a list, the number of\n                configured retry delays cannot exceed 50.\n            retry_jitter_factor: An optional factor that defines the factor to which a\n                retry can be jittered in order to avoid a \"thundering herd\".\n            persist_result: A new option for enabling or disabling result persistence.\n            result_storage: A new storage type to use for results.\n            result_serializer: A new serializer to use for results.\n            result_storage_key: A new key for the persisted result to be stored at.\n            timeout_seconds: A new maximum time for the task to complete in seconds.\n            log_prints: A new option for enabling or disabling redirection of `print` statements.\n            refresh_cache: A new option for enabling or disabling cache refresh.\n            on_completion: A new list of callables to run when the task enters a completed state.\n            on_failure: A new list of callables to run when the task enters a failed state.\n            retry_condition_fn: An optional callable run when a task run returns a Failed state.\n                Should return `True` if the task should continue to its retry policy, and `False`\n                if the task should end as failed. Defaults to `None`, indicating the task should\n                always continue to its retry policy.\n            viz_return_value: An optional value to return when the task dependency tree is visualized.\n\n        Returns:\n            A new `Task` instance.\n\n        Examples:\n\n            Create a new task from an existing task and update the name\n\n            >>> @task(name=\"My task\")\n            >>> def my_task():\n            >>>     return 1\n            >>>\n            >>> new_task = my_task.with_options(name=\"My new task\")\n\n            Create a new task from an existing task and update the retry settings\n\n            >>> from random import randint\n            >>>\n            >>> @task(retries=1, retry_delay_seconds=5)\n            >>> def my_task():\n            >>>     x = randint(0, 5)\n            >>>     if x >= 3:  # Make a task that fails sometimes\n            >>>         raise ValueError(\"Retry me please!\")\n            >>>     return x\n            >>>\n            >>> new_task = my_task.with_options(retries=5, retry_delay_seconds=2)\n\n            Use a task with updated options within a flow\n\n            >>> @task(name=\"My task\")\n            >>> def my_task():\n            >>>     return 1\n            >>>\n            >>> @flow\n            >>> my_flow():\n            >>>     new_task = my_task.with_options(name=\"My new task\")\n            >>>     new_task()\n        \"\"\"\n        return Task(\n            fn=self.fn,\n            name=name or self.name,\n            description=description or self.description,\n            tags=tags or copy(self.tags),\n            cache_key_fn=cache_key_fn or self.cache_key_fn,\n            cache_expiration=cache_expiration or self.cache_expiration,\n            task_run_name=task_run_name,\n            retries=retries if retries is not NotSet else self.retries,\n            retry_delay_seconds=(\n                retry_delay_seconds\n                if retry_delay_seconds is not NotSet\n                else self.retry_delay_seconds\n            ),\n            retry_jitter_factor=(\n                retry_jitter_factor\n                if retry_jitter_factor is not NotSet\n                else self.retry_jitter_factor\n            ),\n            persist_result=(\n                persist_result if persist_result is not NotSet else self.persist_result\n            ),\n            result_storage=(\n                result_storage if result_storage is not NotSet else self.result_storage\n            ),\n            result_storage_key=(\n                result_storage_key\n                if result_storage_key is not NotSet\n                else self.result_storage_key\n            ),\n            result_serializer=(\n                result_serializer\n                if result_serializer is not NotSet\n                else self.result_serializer\n            ),\n            cache_result_in_memory=(\n                cache_result_in_memory\n                if cache_result_in_memory is not None\n                else self.cache_result_in_memory\n            ),\n            timeout_seconds=(\n                timeout_seconds if timeout_seconds is not None else self.timeout_seconds\n            ),\n            log_prints=(log_prints if log_prints is not NotSet else self.log_prints),\n            refresh_cache=(\n                refresh_cache if refresh_cache is not NotSet else self.refresh_cache\n            ),\n            on_completion=on_completion or self.on_completion,\n            on_failure=on_failure or self.on_failure,\n            retry_condition_fn=retry_condition_fn or self.retry_condition_fn,\n            viz_return_value=viz_return_value or self.viz_return_value,\n        )\n\n    @overload\n    def __call__(\n        self: \"Task[P, NoReturn]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> None:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def __call__(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> T:\n        ...\n\n    @overload\n    def __call__(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        return_state: Literal[True],\n        **kwargs: P.kwargs,\n    ) -> State[T]:\n        ...\n\n    def __call__(\n        self,\n        *args: P.args,\n        return_state: bool = False,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: P.kwargs,\n    ):\n        \"\"\"\n        Run the task and return the result. If `return_state` is True returns\n        the result is wrapped in a Prefect State which provides error handling.\n        \"\"\"\n        from prefect.engine import enter_task_run_engine\n        from prefect.task_runners import SequentialTaskRunner\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n\n        return_type = \"state\" if return_state else \"result\"\n\n        task_run_tracker = get_task_viz_tracker()\n        if task_run_tracker:\n            return track_viz_task(\n                self.isasync, self.name, parameters, self.viz_return_value\n            )\n\n        return enter_task_run_engine(\n            self,\n            parameters=parameters,\n            wait_for=wait_for,\n            task_runner=SequentialTaskRunner(),\n            return_type=return_type,\n            mapped=False,\n        )\n\n    @overload\n    def _run(\n        self: \"Task[P, NoReturn]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> PrefectFuture[None, Sync]:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def _run(\n        self: \"Task[P, Coroutine[Any, Any, T]]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> Awaitable[State[T]]:\n        ...\n\n    @overload\n    def _run(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> State[T]:\n        ...\n\n    def _run(\n        self,\n        *args: P.args,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: P.kwargs,\n    ) -> Union[State, Awaitable[State]]:\n        \"\"\"\n        Run the task and return the final state.\n        \"\"\"\n        from prefect.engine import enter_task_run_engine\n        from prefect.task_runners import SequentialTaskRunner\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n\n        return enter_task_run_engine(\n            self,\n            parameters=parameters,\n            wait_for=wait_for,\n            return_type=\"state\",\n            task_runner=SequentialTaskRunner(),\n            mapped=False,\n        )\n\n    @overload\n    def submit(\n        self: \"Task[P, NoReturn]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> PrefectFuture[None, Sync]:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def submit(\n        self: \"Task[P, Coroutine[Any, Any, T]]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> Awaitable[PrefectFuture[T, Async]]:\n        ...\n\n    @overload\n    def submit(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> PrefectFuture[T, Sync]:\n        ...\n\n    @overload\n    def submit(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        return_state: Literal[True],\n        **kwargs: P.kwargs,\n    ) -> State[T]:\n        ...\n\n    def submit(\n        self,\n        *args: Any,\n        return_state: bool = False,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: Any,\n    ) -> Union[PrefectFuture, Awaitable[PrefectFuture]]:\n        \"\"\"\n        Submit a run of the task to a worker.\n\n        Must be called within a flow function. If writing an async task, this call must\n        be awaited.\n\n        Will create a new task run in the backing API and submit the task to the flow's\n        task runner. This call only blocks execution while the task is being submitted,\n        once it is submitted, the flow function will continue executing. However, note\n        that the `SequentialTaskRunner` does not implement parallel execution for sync tasks\n        and they are fully resolved on submission.\n\n        Args:\n            *args: Arguments to run the task with\n            return_state: Return the result of the flow run wrapped in a\n                Prefect State.\n            wait_for: Upstream task futures to wait for before starting the task\n            **kwargs: Keyword arguments to run the task with\n\n        Returns:\n            If `return_state` is False a future allowing asynchronous access to\n                the state of the task\n            If `return_state` is True a future wrapped in a Prefect State allowing asynchronous access to\n                the state of the task\n\n        Examples:\n\n            Define a task\n\n            >>> from prefect import task\n            >>> @task\n            >>> def my_task():\n            >>>     return \"hello\"\n\n            Run a task in a flow\n\n            >>> from prefect import flow\n            >>> @flow\n            >>> def my_flow():\n            >>>     my_task.submit()\n\n            Wait for a task to finish\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     my_task.submit().wait()\n\n            Use the result from a task in a flow\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     print(my_task.submit().result())\n            >>>\n            >>> my_flow()\n            hello\n\n            Run an async task in an async flow\n\n            >>> @task\n            >>> async def my_async_task():\n            >>>     pass\n            >>>\n            >>> @flow\n            >>> async def my_flow():\n            >>>     await my_async_task.submit()\n\n            Run a sync task in an async flow\n\n            >>> @flow\n            >>> async def my_flow():\n            >>>     my_task.submit()\n\n            Enforce ordering between tasks that do not exchange data\n            >>> @task\n            >>> def task_1():\n            >>>     pass\n            >>>\n            >>> @task\n            >>> def task_2():\n            >>>     pass\n            >>>\n            >>> @flow\n            >>> def my_flow():\n            >>>     x = task_1.submit()\n            >>>\n            >>>     # task 2 will wait for task_1 to complete\n            >>>     y = task_2.submit(wait_for=[x])\n\n        \"\"\"\n\n        from prefect.engine import enter_task_run_engine\n\n        # Convert the call args/kwargs to a parameter dict\n        parameters = get_call_parameters(self.fn, args, kwargs)\n        return_type = \"state\" if return_state else \"future\"\n\n        task_viz_tracker = get_task_viz_tracker()\n        if task_viz_tracker:\n            raise VisualizationUnsupportedError(\n                \"`task.submit()` is not currently supported by `flow.visualize()`\"\n            )\n\n        return enter_task_run_engine(\n            self,\n            parameters=parameters,\n            wait_for=wait_for,\n            return_type=return_type,\n            task_runner=None,  # Use the flow's task runner\n            mapped=False,\n        )\n\n    @overload\n    def map(\n        self: \"Task[P, NoReturn]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> List[PrefectFuture[None, Sync]]:\n        # `NoReturn` matches if a type can't be inferred for the function which stops a\n        # sync function from matching the `Coroutine` overload\n        ...\n\n    @overload\n    def map(\n        self: \"Task[P, Coroutine[Any, Any, T]]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> Awaitable[List[PrefectFuture[T, Async]]]:\n        ...\n\n    @overload\n    def map(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        **kwargs: P.kwargs,\n    ) -> List[PrefectFuture[T, Sync]]:\n        ...\n\n    @overload\n    def map(\n        self: \"Task[P, T]\",\n        *args: P.args,\n        return_state: Literal[True],\n        **kwargs: P.kwargs,\n    ) -> List[State[T]]:\n        ...\n\n    def map(\n        self,\n        *args: Any,\n        return_state: bool = False,\n        wait_for: Optional[Iterable[PrefectFuture]] = None,\n        **kwargs: Any,\n    ) -> Any:\n        \"\"\"\n        Submit a mapped run of the task to a worker.\n\n        Must be called within a flow function. If writing an async task, this\n        call must be awaited.\n\n        Must be called with at least one iterable and all iterables must be\n        the same length. Any arguments that are not iterable will be treated as\n        a static value and each task run will receive the same value.\n\n        Will create as many task runs as the length of the iterable(s) in the\n        backing API and submit the task runs to the flow's task runner. This\n        call blocks if given a future as input while the future is resolved. It\n        also blocks while the tasks are being submitted, once they are\n        submitted, the flow function will continue executing. However, note\n        that the `SequentialTaskRunner` does not implement parallel execution\n        for sync tasks and they are fully resolved on submission.\n\n        Args:\n            *args: Iterable and static arguments to run the tasks with\n            return_state: Return a list of Prefect States that wrap the results\n                of each task run.\n            wait_for: Upstream task futures to wait for before starting the\n                task\n            **kwargs: Keyword iterable arguments to run the task with\n\n        Returns:\n            A list of futures allowing asynchronous access to the state of the\n            tasks\n\n        Examples:\n\n            Define a task\n\n            >>> from prefect import task\n            >>> @task\n            >>> def my_task(x):\n            >>>     return x + 1\n\n            Create mapped tasks\n\n            >>> from prefect import flow\n            >>> @flow\n            >>> def my_flow():\n            >>>     my_task.map([1, 2, 3])\n\n            Wait for all mapped tasks to finish\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     futures = my_task.map([1, 2, 3])\n            >>>     for future in futures:\n            >>>         future.wait()\n            >>>     # Now all of the mapped tasks have finished\n            >>>     my_task(10)\n\n            Use the result from mapped tasks in a flow\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     futures = my_task.map([1, 2, 3])\n            >>>     for future in futures:\n            >>>         print(future.result())\n            >>> my_flow()\n            2\n            3\n            4\n\n            Enforce ordering between tasks that do not exchange data\n            >>> @task\n            >>> def task_1(x):\n            >>>     pass\n            >>>\n            >>> @task\n            >>> def task_2(y):\n            >>>     pass\n            >>>\n            >>> @flow\n            >>> def my_flow():\n            >>>     x = task_1.submit()\n            >>>\n            >>>     # task 2 will wait for task_1 to complete\n            >>>     y = task_2.map([1, 2, 3], wait_for=[x])\n\n            Use a non-iterable input as a constant across mapped tasks\n            >>> @task\n            >>> def display(prefix, item):\n            >>>    print(prefix, item)\n            >>>\n            >>> @flow\n            >>> def my_flow():\n            >>>     display.map(\"Check it out: \", [1, 2, 3])\n            >>>\n            >>> my_flow()\n            Check it out: 1\n            Check it out: 2\n            Check it out: 3\n\n            Use `unmapped` to treat an iterable argument as a constant\n            >>> from prefect import unmapped\n            >>>\n            >>> @task\n            >>> def add_n_to_items(items, n):\n            >>>     return [item + n for item in items]\n            >>>\n            >>> @flow\n            >>> def my_flow():\n            >>>     return add_n_to_items.map(unmapped([10, 20]), n=[1, 2, 3])\n            >>>\n            >>> my_flow()\n            [[11, 21], [12, 22], [13, 23]]\n        \"\"\"\n\n        from prefect.engine import enter_task_run_engine\n\n        # Convert the call args/kwargs to a parameter dict; do not apply defaults\n        # since they should not be mapped over\n        parameters = get_call_parameters(self.fn, args, kwargs, apply_defaults=False)\n        return_type = \"state\" if return_state else \"future\"\n\n        task_viz_tracker = get_task_viz_tracker()\n        if task_viz_tracker:\n            raise VisualizationUnsupportedError(\n                \"`task.map()` is not currently supported by `flow.visualize()`\"\n            )\n\n        return enter_task_run_engine(\n            self,\n            parameters=parameters,\n            wait_for=wait_for,\n            return_type=return_type,\n            task_runner=None,\n            mapped=True,\n        )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.Task.map","title":"map","text":"

    Submit a mapped run of the task to a worker.

    Must be called within a flow function. If writing an async task, this call must be awaited.

    Must be called with at least one iterable and all iterables must be the same length. Any arguments that are not iterable will be treated as a static value and each task run will receive the same value.

    Will create as many task runs as the length of the iterable(s) in the backing API and submit the task runs to the flow's task runner. This call blocks if given a future as input while the future is resolved. It also blocks while the tasks are being submitted, once they are submitted, the flow function will continue executing. However, note that the SequentialTaskRunner does not implement parallel execution for sync tasks and they are fully resolved on submission.

    Parameters:

    Name Type Description Default *args Any

    Iterable and static arguments to run the tasks with

    () return_state bool

    Return a list of Prefect States that wrap the results of each task run.

    False wait_for Optional[Iterable[PrefectFuture]]

    Upstream task futures to wait for before starting the task

    None **kwargs Any

    Keyword iterable arguments to run the task with

    {}

    Returns:

    Type Description Any

    A list of futures allowing asynchronous access to the state of the

    Any

    tasks

    Define a task\n\n>>> from prefect import task\n>>> @task\n>>> def my_task(x):\n>>>     return x + 1\n\nCreate mapped tasks\n\n>>> from prefect import flow\n>>> @flow\n>>> def my_flow():\n>>>     my_task.map([1, 2, 3])\n\nWait for all mapped tasks to finish\n\n>>> @flow\n>>> def my_flow():\n>>>     futures = my_task.map([1, 2, 3])\n>>>     for future in futures:\n>>>         future.wait()\n>>>     # Now all of the mapped tasks have finished\n>>>     my_task(10)\n\nUse the result from mapped tasks in a flow\n\n>>> @flow\n>>> def my_flow():\n>>>     futures = my_task.map([1, 2, 3])\n>>>     for future in futures:\n>>>         print(future.result())\n>>> my_flow()\n2\n3\n4\n\nEnforce ordering between tasks that do not exchange data\n>>> @task\n>>> def task_1(x):\n>>>     pass\n>>>\n>>> @task\n>>> def task_2(y):\n>>>     pass\n>>>\n>>> @flow\n>>> def my_flow():\n>>>     x = task_1.submit()\n>>>\n>>>     # task 2 will wait for task_1 to complete\n>>>     y = task_2.map([1, 2, 3], wait_for=[x])\n\nUse a non-iterable input as a constant across mapped tasks\n>>> @task\n>>> def display(prefix, item):\n>>>    print(prefix, item)\n>>>\n>>> @flow\n>>> def my_flow():\n>>>     display.map(\"Check it out: \", [1, 2, 3])\n>>>\n>>> my_flow()\nCheck it out: 1\nCheck it out: 2\nCheck it out: 3\n\nUse `unmapped` to treat an iterable argument as a constant\n>>> from prefect import unmapped\n>>>\n>>> @task\n>>> def add_n_to_items(items, n):\n>>>     return [item + n for item in items]\n>>>\n>>> @flow\n>>> def my_flow():\n>>>     return add_n_to_items.map(unmapped([10, 20]), n=[1, 2, 3])\n>>>\n>>> my_flow()\n[[11, 21], [12, 22], [13, 23]]\n
    Source code in prefect/tasks.py
    def map(\n    self,\n    *args: Any,\n    return_state: bool = False,\n    wait_for: Optional[Iterable[PrefectFuture]] = None,\n    **kwargs: Any,\n) -> Any:\n    \"\"\"\n    Submit a mapped run of the task to a worker.\n\n    Must be called within a flow function. If writing an async task, this\n    call must be awaited.\n\n    Must be called with at least one iterable and all iterables must be\n    the same length. Any arguments that are not iterable will be treated as\n    a static value and each task run will receive the same value.\n\n    Will create as many task runs as the length of the iterable(s) in the\n    backing API and submit the task runs to the flow's task runner. This\n    call blocks if given a future as input while the future is resolved. It\n    also blocks while the tasks are being submitted, once they are\n    submitted, the flow function will continue executing. However, note\n    that the `SequentialTaskRunner` does not implement parallel execution\n    for sync tasks and they are fully resolved on submission.\n\n    Args:\n        *args: Iterable and static arguments to run the tasks with\n        return_state: Return a list of Prefect States that wrap the results\n            of each task run.\n        wait_for: Upstream task futures to wait for before starting the\n            task\n        **kwargs: Keyword iterable arguments to run the task with\n\n    Returns:\n        A list of futures allowing asynchronous access to the state of the\n        tasks\n\n    Examples:\n\n        Define a task\n\n        >>> from prefect import task\n        >>> @task\n        >>> def my_task(x):\n        >>>     return x + 1\n\n        Create mapped tasks\n\n        >>> from prefect import flow\n        >>> @flow\n        >>> def my_flow():\n        >>>     my_task.map([1, 2, 3])\n\n        Wait for all mapped tasks to finish\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     futures = my_task.map([1, 2, 3])\n        >>>     for future in futures:\n        >>>         future.wait()\n        >>>     # Now all of the mapped tasks have finished\n        >>>     my_task(10)\n\n        Use the result from mapped tasks in a flow\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     futures = my_task.map([1, 2, 3])\n        >>>     for future in futures:\n        >>>         print(future.result())\n        >>> my_flow()\n        2\n        3\n        4\n\n        Enforce ordering between tasks that do not exchange data\n        >>> @task\n        >>> def task_1(x):\n        >>>     pass\n        >>>\n        >>> @task\n        >>> def task_2(y):\n        >>>     pass\n        >>>\n        >>> @flow\n        >>> def my_flow():\n        >>>     x = task_1.submit()\n        >>>\n        >>>     # task 2 will wait for task_1 to complete\n        >>>     y = task_2.map([1, 2, 3], wait_for=[x])\n\n        Use a non-iterable input as a constant across mapped tasks\n        >>> @task\n        >>> def display(prefix, item):\n        >>>    print(prefix, item)\n        >>>\n        >>> @flow\n        >>> def my_flow():\n        >>>     display.map(\"Check it out: \", [1, 2, 3])\n        >>>\n        >>> my_flow()\n        Check it out: 1\n        Check it out: 2\n        Check it out: 3\n\n        Use `unmapped` to treat an iterable argument as a constant\n        >>> from prefect import unmapped\n        >>>\n        >>> @task\n        >>> def add_n_to_items(items, n):\n        >>>     return [item + n for item in items]\n        >>>\n        >>> @flow\n        >>> def my_flow():\n        >>>     return add_n_to_items.map(unmapped([10, 20]), n=[1, 2, 3])\n        >>>\n        >>> my_flow()\n        [[11, 21], [12, 22], [13, 23]]\n    \"\"\"\n\n    from prefect.engine import enter_task_run_engine\n\n    # Convert the call args/kwargs to a parameter dict; do not apply defaults\n    # since they should not be mapped over\n    parameters = get_call_parameters(self.fn, args, kwargs, apply_defaults=False)\n    return_type = \"state\" if return_state else \"future\"\n\n    task_viz_tracker = get_task_viz_tracker()\n    if task_viz_tracker:\n        raise VisualizationUnsupportedError(\n            \"`task.map()` is not currently supported by `flow.visualize()`\"\n        )\n\n    return enter_task_run_engine(\n        self,\n        parameters=parameters,\n        wait_for=wait_for,\n        return_type=return_type,\n        task_runner=None,\n        mapped=True,\n    )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.Task.submit","title":"submit","text":"

    Submit a run of the task to a worker.

    Must be called within a flow function. If writing an async task, this call must be awaited.

    Will create a new task run in the backing API and submit the task to the flow's task runner. This call only blocks execution while the task is being submitted, once it is submitted, the flow function will continue executing. However, note that the SequentialTaskRunner does not implement parallel execution for sync tasks and they are fully resolved on submission.

    Parameters:

    Name Type Description Default *args Any

    Arguments to run the task with

    () return_state bool

    Return the result of the flow run wrapped in a Prefect State.

    False wait_for Optional[Iterable[PrefectFuture]]

    Upstream task futures to wait for before starting the task

    None **kwargs Any

    Keyword arguments to run the task with

    {}

    Returns:

    Type Description Union[PrefectFuture, Awaitable[PrefectFuture]]

    If return_state is False a future allowing asynchronous access to the state of the task

    Union[PrefectFuture, Awaitable[PrefectFuture]]

    If return_state is True a future wrapped in a Prefect State allowing asynchronous access to the state of the task

    Define a task\n\n>>> from prefect import task\n>>> @task\n>>> def my_task():\n>>>     return \"hello\"\n\nRun a task in a flow\n\n>>> from prefect import flow\n>>> @flow\n>>> def my_flow():\n>>>     my_task.submit()\n\nWait for a task to finish\n\n>>> @flow\n>>> def my_flow():\n>>>     my_task.submit().wait()\n\nUse the result from a task in a flow\n\n>>> @flow\n>>> def my_flow():\n>>>     print(my_task.submit().result())\n>>>\n>>> my_flow()\nhello\n\nRun an async task in an async flow\n\n>>> @task\n>>> async def my_async_task():\n>>>     pass\n>>>\n>>> @flow\n>>> async def my_flow():\n>>>     await my_async_task.submit()\n\nRun a sync task in an async flow\n\n>>> @flow\n>>> async def my_flow():\n>>>     my_task.submit()\n\nEnforce ordering between tasks that do not exchange data\n>>> @task\n>>> def task_1():\n>>>     pass\n>>>\n>>> @task\n>>> def task_2():\n>>>     pass\n>>>\n>>> @flow\n>>> def my_flow():\n>>>     x = task_1.submit()\n>>>\n>>>     # task 2 will wait for task_1 to complete\n>>>     y = task_2.submit(wait_for=[x])\n
    Source code in prefect/tasks.py
    def submit(\n    self,\n    *args: Any,\n    return_state: bool = False,\n    wait_for: Optional[Iterable[PrefectFuture]] = None,\n    **kwargs: Any,\n) -> Union[PrefectFuture, Awaitable[PrefectFuture]]:\n    \"\"\"\n    Submit a run of the task to a worker.\n\n    Must be called within a flow function. If writing an async task, this call must\n    be awaited.\n\n    Will create a new task run in the backing API and submit the task to the flow's\n    task runner. This call only blocks execution while the task is being submitted,\n    once it is submitted, the flow function will continue executing. However, note\n    that the `SequentialTaskRunner` does not implement parallel execution for sync tasks\n    and they are fully resolved on submission.\n\n    Args:\n        *args: Arguments to run the task with\n        return_state: Return the result of the flow run wrapped in a\n            Prefect State.\n        wait_for: Upstream task futures to wait for before starting the task\n        **kwargs: Keyword arguments to run the task with\n\n    Returns:\n        If `return_state` is False a future allowing asynchronous access to\n            the state of the task\n        If `return_state` is True a future wrapped in a Prefect State allowing asynchronous access to\n            the state of the task\n\n    Examples:\n\n        Define a task\n\n        >>> from prefect import task\n        >>> @task\n        >>> def my_task():\n        >>>     return \"hello\"\n\n        Run a task in a flow\n\n        >>> from prefect import flow\n        >>> @flow\n        >>> def my_flow():\n        >>>     my_task.submit()\n\n        Wait for a task to finish\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     my_task.submit().wait()\n\n        Use the result from a task in a flow\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     print(my_task.submit().result())\n        >>>\n        >>> my_flow()\n        hello\n\n        Run an async task in an async flow\n\n        >>> @task\n        >>> async def my_async_task():\n        >>>     pass\n        >>>\n        >>> @flow\n        >>> async def my_flow():\n        >>>     await my_async_task.submit()\n\n        Run a sync task in an async flow\n\n        >>> @flow\n        >>> async def my_flow():\n        >>>     my_task.submit()\n\n        Enforce ordering between tasks that do not exchange data\n        >>> @task\n        >>> def task_1():\n        >>>     pass\n        >>>\n        >>> @task\n        >>> def task_2():\n        >>>     pass\n        >>>\n        >>> @flow\n        >>> def my_flow():\n        >>>     x = task_1.submit()\n        >>>\n        >>>     # task 2 will wait for task_1 to complete\n        >>>     y = task_2.submit(wait_for=[x])\n\n    \"\"\"\n\n    from prefect.engine import enter_task_run_engine\n\n    # Convert the call args/kwargs to a parameter dict\n    parameters = get_call_parameters(self.fn, args, kwargs)\n    return_type = \"state\" if return_state else \"future\"\n\n    task_viz_tracker = get_task_viz_tracker()\n    if task_viz_tracker:\n        raise VisualizationUnsupportedError(\n            \"`task.submit()` is not currently supported by `flow.visualize()`\"\n        )\n\n    return enter_task_run_engine(\n        self,\n        parameters=parameters,\n        wait_for=wait_for,\n        return_type=return_type,\n        task_runner=None,  # Use the flow's task runner\n        mapped=False,\n    )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.Task.with_options","title":"with_options","text":"

    Create a new task from the current object, updating provided options.

    Parameters:

    Name Type Description Default name str

    A new name for the task.

    None description str

    A new description for the task.

    None tags Iterable[str]

    A new set of tags for the task. If given, existing tags are ignored, not merged.

    None cache_key_fn Callable[[TaskRunContext, Dict[str, Any]], Optional[str]]

    A new cache key function for the task.

    None cache_expiration timedelta

    A new cache expiration time for the task.

    None task_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this task; this name can be provided as a string template with the task's keyword arguments as variables, or a function that returns a string.

    None retries Optional[int]

    A new number of times to retry on task run failure.

    NotSet retry_delay_seconds Union[float, int, List[float], Callable[[int], List[float]]]

    Optionally configures how long to wait before retrying the task after failure. This is only applicable if retries is nonzero. This setting can either be a number of seconds, a list of retry delays, or a callable that, given the total number of retries, generates a list of retry delays. If a number of seconds, that delay will be applied to all retries. If a list, each retry will wait for the corresponding delay before retrying. When passing a callable or a list, the number of configured retry delays cannot exceed 50.

    NotSet retry_jitter_factor Optional[float]

    An optional factor that defines the factor to which a retry can be jittered in order to avoid a \"thundering herd\".

    NotSet persist_result Optional[bool]

    A new option for enabling or disabling result persistence.

    NotSet result_storage Optional[ResultStorage]

    A new storage type to use for results.

    NotSet result_serializer Optional[ResultSerializer]

    A new serializer to use for results.

    NotSet result_storage_key Optional[str]

    A new key for the persisted result to be stored at.

    NotSet timeout_seconds Union[int, float]

    A new maximum time for the task to complete in seconds.

    None log_prints Optional[bool]

    A new option for enabling or disabling redirection of print statements.

    NotSet refresh_cache Optional[bool]

    A new option for enabling or disabling cache refresh.

    NotSet on_completion Optional[List[Callable[[Task, TaskRun, State], None]]]

    A new list of callables to run when the task enters a completed state.

    None on_failure Optional[List[Callable[[Task, TaskRun, State], None]]]

    A new list of callables to run when the task enters a failed state.

    None retry_condition_fn Optional[Callable[[Task, TaskRun, State], bool]]

    An optional callable run when a task run returns a Failed state. Should return True if the task should continue to its retry policy, and False if the task should end as failed. Defaults to None, indicating the task should always continue to its retry policy.

    None viz_return_value Optional[Any]

    An optional value to return when the task dependency tree is visualized.

    None

    Returns:

    Type Description

    A new Task instance.

    Create a new task from an existing task and update the name\n\n>>> @task(name=\"My task\")\n>>> def my_task():\n>>>     return 1\n>>>\n>>> new_task = my_task.with_options(name=\"My new task\")\n\nCreate a new task from an existing task and update the retry settings\n\n>>> from random import randint\n>>>\n>>> @task(retries=1, retry_delay_seconds=5)\n>>> def my_task():\n>>>     x = randint(0, 5)\n>>>     if x >= 3:  # Make a task that fails sometimes\n>>>         raise ValueError(\"Retry me please!\")\n>>>     return x\n>>>\n>>> new_task = my_task.with_options(retries=5, retry_delay_seconds=2)\n\nUse a task with updated options within a flow\n\n>>> @task(name=\"My task\")\n>>> def my_task():\n>>>     return 1\n>>>\n>>> @flow\n>>> my_flow():\n>>>     new_task = my_task.with_options(name=\"My new task\")\n>>>     new_task()\n
    Source code in prefect/tasks.py
    def with_options(\n    self,\n    *,\n    name: str = None,\n    description: str = None,\n    tags: Iterable[str] = None,\n    cache_key_fn: Callable[\n        [\"TaskRunContext\", Dict[str, Any]], Optional[str]\n    ] = None,\n    task_run_name: Optional[Union[Callable[[], str], str]] = None,\n    cache_expiration: datetime.timedelta = None,\n    retries: Optional[int] = NotSet,\n    retry_delay_seconds: Union[\n        float,\n        int,\n        List[float],\n        Callable[[int], List[float]],\n    ] = NotSet,\n    retry_jitter_factor: Optional[float] = NotSet,\n    persist_result: Optional[bool] = NotSet,\n    result_storage: Optional[ResultStorage] = NotSet,\n    result_serializer: Optional[ResultSerializer] = NotSet,\n    result_storage_key: Optional[str] = NotSet,\n    cache_result_in_memory: Optional[bool] = None,\n    timeout_seconds: Union[int, float] = None,\n    log_prints: Optional[bool] = NotSet,\n    refresh_cache: Optional[bool] = NotSet,\n    on_completion: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n    on_failure: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n    retry_condition_fn: Optional[Callable[[\"Task\", TaskRun, State], bool]] = None,\n    viz_return_value: Optional[Any] = None,\n):\n    \"\"\"\n    Create a new task from the current object, updating provided options.\n\n    Args:\n        name: A new name for the task.\n        description: A new description for the task.\n        tags: A new set of tags for the task. If given, existing tags are ignored,\n            not merged.\n        cache_key_fn: A new cache key function for the task.\n        cache_expiration: A new cache expiration time for the task.\n        task_run_name: An optional name to distinguish runs of this task; this name can be provided\n            as a string template with the task's keyword arguments as variables,\n            or a function that returns a string.\n        retries: A new number of times to retry on task run failure.\n        retry_delay_seconds: Optionally configures how long to wait before retrying\n            the task after failure. This is only applicable if `retries` is nonzero.\n            This setting can either be a number of seconds, a list of retry delays,\n            or a callable that, given the total number of retries, generates a list\n            of retry delays. If a number of seconds, that delay will be applied to\n            all retries. If a list, each retry will wait for the corresponding delay\n            before retrying. When passing a callable or a list, the number of\n            configured retry delays cannot exceed 50.\n        retry_jitter_factor: An optional factor that defines the factor to which a\n            retry can be jittered in order to avoid a \"thundering herd\".\n        persist_result: A new option for enabling or disabling result persistence.\n        result_storage: A new storage type to use for results.\n        result_serializer: A new serializer to use for results.\n        result_storage_key: A new key for the persisted result to be stored at.\n        timeout_seconds: A new maximum time for the task to complete in seconds.\n        log_prints: A new option for enabling or disabling redirection of `print` statements.\n        refresh_cache: A new option for enabling or disabling cache refresh.\n        on_completion: A new list of callables to run when the task enters a completed state.\n        on_failure: A new list of callables to run when the task enters a failed state.\n        retry_condition_fn: An optional callable run when a task run returns a Failed state.\n            Should return `True` if the task should continue to its retry policy, and `False`\n            if the task should end as failed. Defaults to `None`, indicating the task should\n            always continue to its retry policy.\n        viz_return_value: An optional value to return when the task dependency tree is visualized.\n\n    Returns:\n        A new `Task` instance.\n\n    Examples:\n\n        Create a new task from an existing task and update the name\n\n        >>> @task(name=\"My task\")\n        >>> def my_task():\n        >>>     return 1\n        >>>\n        >>> new_task = my_task.with_options(name=\"My new task\")\n\n        Create a new task from an existing task and update the retry settings\n\n        >>> from random import randint\n        >>>\n        >>> @task(retries=1, retry_delay_seconds=5)\n        >>> def my_task():\n        >>>     x = randint(0, 5)\n        >>>     if x >= 3:  # Make a task that fails sometimes\n        >>>         raise ValueError(\"Retry me please!\")\n        >>>     return x\n        >>>\n        >>> new_task = my_task.with_options(retries=5, retry_delay_seconds=2)\n\n        Use a task with updated options within a flow\n\n        >>> @task(name=\"My task\")\n        >>> def my_task():\n        >>>     return 1\n        >>>\n        >>> @flow\n        >>> my_flow():\n        >>>     new_task = my_task.with_options(name=\"My new task\")\n        >>>     new_task()\n    \"\"\"\n    return Task(\n        fn=self.fn,\n        name=name or self.name,\n        description=description or self.description,\n        tags=tags or copy(self.tags),\n        cache_key_fn=cache_key_fn or self.cache_key_fn,\n        cache_expiration=cache_expiration or self.cache_expiration,\n        task_run_name=task_run_name,\n        retries=retries if retries is not NotSet else self.retries,\n        retry_delay_seconds=(\n            retry_delay_seconds\n            if retry_delay_seconds is not NotSet\n            else self.retry_delay_seconds\n        ),\n        retry_jitter_factor=(\n            retry_jitter_factor\n            if retry_jitter_factor is not NotSet\n            else self.retry_jitter_factor\n        ),\n        persist_result=(\n            persist_result if persist_result is not NotSet else self.persist_result\n        ),\n        result_storage=(\n            result_storage if result_storage is not NotSet else self.result_storage\n        ),\n        result_storage_key=(\n            result_storage_key\n            if result_storage_key is not NotSet\n            else self.result_storage_key\n        ),\n        result_serializer=(\n            result_serializer\n            if result_serializer is not NotSet\n            else self.result_serializer\n        ),\n        cache_result_in_memory=(\n            cache_result_in_memory\n            if cache_result_in_memory is not None\n            else self.cache_result_in_memory\n        ),\n        timeout_seconds=(\n            timeout_seconds if timeout_seconds is not None else self.timeout_seconds\n        ),\n        log_prints=(log_prints if log_prints is not NotSet else self.log_prints),\n        refresh_cache=(\n            refresh_cache if refresh_cache is not NotSet else self.refresh_cache\n        ),\n        on_completion=on_completion or self.on_completion,\n        on_failure=on_failure or self.on_failure,\n        retry_condition_fn=retry_condition_fn or self.retry_condition_fn,\n        viz_return_value=viz_return_value or self.viz_return_value,\n    )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.exponential_backoff","title":"exponential_backoff","text":"

    A task retry backoff utility that configures exponential backoff for task retries. The exponential backoff design matches the urllib3 implementation.

    Parameters:

    Name Type Description Default backoff_factor float

    the base delay for the first retry, subsequent retries will increase the delay time by powers of 2.

    required

    Returns:

    Type Description Callable[[int], List[float]]

    a callable that can be passed to the task constructor

    Source code in prefect/tasks.py
    def exponential_backoff(backoff_factor: float) -> Callable[[int], List[float]]:\n    \"\"\"\n    A task retry backoff utility that configures exponential backoff for task retries.\n    The exponential backoff design matches the urllib3 implementation.\n\n    Arguments:\n        backoff_factor: the base delay for the first retry, subsequent retries will\n            increase the delay time by powers of 2.\n\n    Returns:\n        a callable that can be passed to the task constructor\n    \"\"\"\n\n    def retry_backoff_callable(retries: int) -> List[float]:\n        # no more than 50 retry delays can be configured on a task\n        retries = min(retries, 50)\n\n        return [backoff_factor * max(0, 2**r) for r in range(retries)]\n\n    return retry_backoff_callable\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.task","title":"task","text":"

    Decorator to designate a function as a task in a Prefect workflow.

    This decorator may be used for asynchronous or synchronous functions.

    Parameters:

    Name Type Description Default name str

    An optional name for the task; if not provided, the name will be inferred from the given function.

    None description str

    An optional string description for the task.

    None tags Iterable[str]

    An optional set of tags to be associated with runs of this task. These tags are combined with any tags defined by a prefect.tags context at task runtime.

    None version str

    An optional string specifying the version of this task definition

    None cache_key_fn Callable[[TaskRunContext, Dict[str, Any]], Optional[str]]

    An optional callable that, given the task run context and call parameters, generates a string key; if the key matches a previous completed state, that state result will be restored instead of running the task again.

    None cache_expiration timedelta

    An optional amount of time indicating how long cached states for this task should be restorable; if not provided, cached states will never expire.

    None task_run_name Optional[Union[Callable[[], str], str]]

    An optional name to distinguish runs of this task; this name can be provided as a string template with the task's keyword arguments as variables, or a function that returns a string.

    None retries int

    An optional number of times to retry on task run failure

    None retry_delay_seconds Union[float, int, List[float], Callable[[int], List[float]]]

    Optionally configures how long to wait before retrying the task after failure. This is only applicable if retries is nonzero. This setting can either be a number of seconds, a list of retry delays, or a callable that, given the total number of retries, generates a list of retry delays. If a number of seconds, that delay will be applied to all retries. If a list, each retry will wait for the corresponding delay before retrying. When passing a callable or a list, the number of configured retry delays cannot exceed 50.

    None retry_jitter_factor Optional[float]

    An optional factor that defines the factor to which a retry can be jittered in order to avoid a \"thundering herd\".

    None persist_result Optional[bool]

    An optional toggle indicating whether the result of this task should be persisted to result storage. Defaults to None, which indicates that Prefect should choose whether the result should be persisted depending on the features being used.

    None result_storage Optional[ResultStorage]

    An optional block to use to persist the result of this task. Defaults to the value set in the flow the task is called in.

    None result_storage_key Optional[str]

    An optional key to store the result in storage at when persisted. Defaults to a unique identifier.

    None result_serializer Optional[ResultSerializer]

    An optional serializer to use to serialize the result of this task for persistence. Defaults to the value set in the flow the task is called in.

    None timeout_seconds Union[int, float]

    An optional number of seconds indicating a maximum runtime for the task. If the task exceeds this runtime, it will be marked as failed.

    None log_prints Optional[bool]

    If set, print statements in the task will be redirected to the Prefect logger for the task run. Defaults to None, which indicates that the value from the flow should be used.

    None refresh_cache Optional[bool]

    If set, cached results for the cache key are not used. Defaults to None, which indicates that a cached result from a previous execution with matching cache key is used.

    None on_failure Optional[List[Callable[[Task, TaskRun, State], None]]]

    An optional list of callables to run when the task enters a failed state.

    None on_completion Optional[List[Callable[[Task, TaskRun, State], None]]]

    An optional list of callables to run when the task enters a completed state.

    None retry_condition_fn Optional[Callable[[Task, TaskRun, State], bool]]

    An optional callable run when a task run returns a Failed state. Should return True if the task should continue to its retry policy (e.g. retries=3), and False if the task should end as failed. Defaults to None, indicating the task should always continue to its retry policy.

    None viz_return_value Any

    An optional value to return when the task dependency tree is visualized.

    None

    Returns:

    Type Description

    A callable Task object which, when called, will submit the task for execution.

    Examples:

    Define a simple task

    >>> @task\n>>> def add(x, y):\n>>>     return x + y\n

    Define an async task

    >>> @task\n>>> async def add(x, y):\n>>>     return x + y\n

    Define a task with tags and a description

    >>> @task(tags={\"a\", \"b\"}, description=\"This task is empty but its my first!\")\n>>> def my_task():\n>>>     pass\n

    Define a task with a custom name

    >>> @task(name=\"The Ultimate Task\")\n>>> def my_task():\n>>>     pass\n

    Define a task that retries 3 times with a 5 second delay between attempts

    >>> from random import randint\n>>>\n>>> @task(retries=3, retry_delay_seconds=5)\n>>> def my_task():\n>>>     x = randint(0, 5)\n>>>     if x >= 3:  # Make a task that fails sometimes\n>>>         raise ValueError(\"Retry me please!\")\n>>>     return x\n

    Define a task that is cached for a day based on its inputs

    >>> from prefect.tasks import task_input_hash\n>>> from datetime import timedelta\n>>>\n>>> @task(cache_key_fn=task_input_hash, cache_expiration=timedelta(days=1))\n>>> def my_task():\n>>>     return \"hello\"\n
    Source code in prefect/tasks.py
    def task(\n    __fn=None,\n    *,\n    name: str = None,\n    description: str = None,\n    tags: Iterable[str] = None,\n    version: str = None,\n    cache_key_fn: Callable[[\"TaskRunContext\", Dict[str, Any]], Optional[str]] = None,\n    cache_expiration: datetime.timedelta = None,\n    task_run_name: Optional[Union[Callable[[], str], str]] = None,\n    retries: int = None,\n    retry_delay_seconds: Union[\n        float,\n        int,\n        List[float],\n        Callable[[int], List[float]],\n    ] = None,\n    retry_jitter_factor: Optional[float] = None,\n    persist_result: Optional[bool] = None,\n    result_storage: Optional[ResultStorage] = None,\n    result_storage_key: Optional[str] = None,\n    result_serializer: Optional[ResultSerializer] = None,\n    cache_result_in_memory: bool = True,\n    timeout_seconds: Union[int, float] = None,\n    log_prints: Optional[bool] = None,\n    refresh_cache: Optional[bool] = None,\n    on_completion: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n    on_failure: Optional[List[Callable[[\"Task\", TaskRun, State], None]]] = None,\n    retry_condition_fn: Optional[Callable[[\"Task\", TaskRun, State], bool]] = None,\n    viz_return_value: Any = None,\n):\n    \"\"\"\n    Decorator to designate a function as a task in a Prefect workflow.\n\n    This decorator may be used for asynchronous or synchronous functions.\n\n    Args:\n        name: An optional name for the task; if not provided, the name will be inferred\n            from the given function.\n        description: An optional string description for the task.\n        tags: An optional set of tags to be associated with runs of this task. These\n            tags are combined with any tags defined by a `prefect.tags` context at\n            task runtime.\n        version: An optional string specifying the version of this task definition\n        cache_key_fn: An optional callable that, given the task run context and call\n            parameters, generates a string key; if the key matches a previous completed\n            state, that state result will be restored instead of running the task again.\n        cache_expiration: An optional amount of time indicating how long cached states\n            for this task should be restorable; if not provided, cached states will\n            never expire.\n        task_run_name: An optional name to distinguish runs of this task; this name can be provided\n            as a string template with the task's keyword arguments as variables,\n            or a function that returns a string.\n        retries: An optional number of times to retry on task run failure\n        retry_delay_seconds: Optionally configures how long to wait before retrying the\n            task after failure. This is only applicable if `retries` is nonzero. This\n            setting can either be a number of seconds, a list of retry delays, or a\n            callable that, given the total number of retries, generates a list of retry\n            delays. If a number of seconds, that delay will be applied to all retries.\n            If a list, each retry will wait for the corresponding delay before retrying.\n            When passing a callable or a list, the number of configured retry delays\n            cannot exceed 50.\n        retry_jitter_factor: An optional factor that defines the factor to which a retry\n            can be jittered in order to avoid a \"thundering herd\".\n        persist_result: An optional toggle indicating whether the result of this task\n            should be persisted to result storage. Defaults to `None`, which indicates\n            that Prefect should choose whether the result should be persisted depending on\n            the features being used.\n        result_storage: An optional block to use to persist the result of this task.\n            Defaults to the value set in the flow the task is called in.\n        result_storage_key: An optional key to store the result in storage at when persisted.\n            Defaults to a unique identifier.\n        result_serializer: An optional serializer to use to serialize the result of this\n            task for persistence. Defaults to the value set in the flow the task is\n            called in.\n        timeout_seconds: An optional number of seconds indicating a maximum runtime for\n            the task. If the task exceeds this runtime, it will be marked as failed.\n        log_prints: If set, `print` statements in the task will be redirected to the\n            Prefect logger for the task run. Defaults to `None`, which indicates\n            that the value from the flow should be used.\n        refresh_cache: If set, cached results for the cache key are not used.\n            Defaults to `None`, which indicates that a cached result from a previous\n            execution with matching cache key is used.\n        on_failure: An optional list of callables to run when the task enters a failed state.\n        on_completion: An optional list of callables to run when the task enters a completed state.\n        retry_condition_fn: An optional callable run when a task run returns a Failed state. Should\n            return `True` if the task should continue to its retry policy (e.g. `retries=3`), and `False` if the task\n            should end as failed. Defaults to `None`, indicating the task should always continue\n            to its retry policy.\n        viz_return_value: An optional value to return when the task dependency tree is visualized.\n\n    Returns:\n        A callable `Task` object which, when called, will submit the task for execution.\n\n    Examples:\n        Define a simple task\n\n        >>> @task\n        >>> def add(x, y):\n        >>>     return x + y\n\n        Define an async task\n\n        >>> @task\n        >>> async def add(x, y):\n        >>>     return x + y\n\n        Define a task with tags and a description\n\n        >>> @task(tags={\"a\", \"b\"}, description=\"This task is empty but its my first!\")\n        >>> def my_task():\n        >>>     pass\n\n        Define a task with a custom name\n\n        >>> @task(name=\"The Ultimate Task\")\n        >>> def my_task():\n        >>>     pass\n\n        Define a task that retries 3 times with a 5 second delay between attempts\n\n        >>> from random import randint\n        >>>\n        >>> @task(retries=3, retry_delay_seconds=5)\n        >>> def my_task():\n        >>>     x = randint(0, 5)\n        >>>     if x >= 3:  # Make a task that fails sometimes\n        >>>         raise ValueError(\"Retry me please!\")\n        >>>     return x\n\n        Define a task that is cached for a day based on its inputs\n\n        >>> from prefect.tasks import task_input_hash\n        >>> from datetime import timedelta\n        >>>\n        >>> @task(cache_key_fn=task_input_hash, cache_expiration=timedelta(days=1))\n        >>> def my_task():\n        >>>     return \"hello\"\n    \"\"\"\n\n    if __fn:\n        return cast(\n            Task[P, R],\n            Task(\n                fn=__fn,\n                name=name,\n                description=description,\n                tags=tags,\n                version=version,\n                cache_key_fn=cache_key_fn,\n                cache_expiration=cache_expiration,\n                task_run_name=task_run_name,\n                retries=retries,\n                retry_delay_seconds=retry_delay_seconds,\n                retry_jitter_factor=retry_jitter_factor,\n                persist_result=persist_result,\n                result_storage=result_storage,\n                result_storage_key=result_storage_key,\n                result_serializer=result_serializer,\n                cache_result_in_memory=cache_result_in_memory,\n                timeout_seconds=timeout_seconds,\n                log_prints=log_prints,\n                refresh_cache=refresh_cache,\n                on_completion=on_completion,\n                on_failure=on_failure,\n                retry_condition_fn=retry_condition_fn,\n                viz_return_value=viz_return_value,\n            ),\n        )\n    else:\n        return cast(\n            Callable[[Callable[P, R]], Task[P, R]],\n            partial(\n                task,\n                name=name,\n                description=description,\n                tags=tags,\n                version=version,\n                cache_key_fn=cache_key_fn,\n                cache_expiration=cache_expiration,\n                task_run_name=task_run_name,\n                retries=retries,\n                retry_delay_seconds=retry_delay_seconds,\n                retry_jitter_factor=retry_jitter_factor,\n                persist_result=persist_result,\n                result_storage=result_storage,\n                result_storage_key=result_storage_key,\n                result_serializer=result_serializer,\n                cache_result_in_memory=cache_result_in_memory,\n                timeout_seconds=timeout_seconds,\n                log_prints=log_prints,\n                refresh_cache=refresh_cache,\n                on_completion=on_completion,\n                on_failure=on_failure,\n                retry_condition_fn=retry_condition_fn,\n                viz_return_value=viz_return_value,\n            ),\n        )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/tasks/#prefect.tasks.task_input_hash","title":"task_input_hash","text":"

    A task cache key implementation which hashes all inputs to the task using a JSON or cloudpickle serializer. If any arguments are not JSON serializable, the pickle serializer is used as a fallback. If cloudpickle fails, this will return a null key indicating that a cache key could not be generated for the given inputs.

    Parameters:

    Name Type Description Default context TaskRunContext

    the active TaskRunContext

    required arguments Dict[str, Any]

    a dictionary of arguments to be passed to the underlying task

    required

    Returns:

    Type Description Optional[str]

    a string hash if hashing succeeded, else None

    Source code in prefect/tasks.py
    def task_input_hash(\n    context: \"TaskRunContext\", arguments: Dict[str, Any]\n) -> Optional[str]:\n    \"\"\"\n    A task cache key implementation which hashes all inputs to the task using a JSON or\n    cloudpickle serializer. If any arguments are not JSON serializable, the pickle\n    serializer is used as a fallback. If cloudpickle fails, this will return a null key\n    indicating that a cache key could not be generated for the given inputs.\n\n    Arguments:\n        context: the active `TaskRunContext`\n        arguments: a dictionary of arguments to be passed to the underlying task\n\n    Returns:\n        a string hash if hashing succeeded, else `None`\n    \"\"\"\n    return hash_objects(\n        # We use the task key to get the qualified name for the task and include the\n        # task functions `co_code` bytes to avoid caching when the underlying function\n        # changes\n        context.task.task_key,\n        context.task.fn.__code__.co_code.hex(),\n        arguments,\n    )\n
    ","tags":["Python API","tasks","caching"]},{"location":"api-ref/prefect/testing/","title":"prefect.testing","text":"","tags":["Python API","testing"]},{"location":"api-ref/prefect/testing/#prefect.testing","title":"prefect.testing","text":"","tags":["Python API","testing"]},{"location":"api-ref/prefect/variables/","title":"prefect.variables","text":"","tags":["Python API","variables"]},{"location":"api-ref/prefect/variables/#prefect.variables","title":"prefect.variables","text":"","tags":["Python API","variables"]},{"location":"api-ref/prefect/variables/#prefect.variables.get","title":"get async","text":"

    Get a variable by name. If doesn't exist return the default.

        from prefect import variables\n\n    @flow\n    def my_flow():\n        var = variables.get(\"my_var\")\n
    or
        from prefect import variables\n\n    @flow\n    async def my_flow():\n        var = await variables.get(\"my_var\")\n

    Source code in prefect/variables.py
    @sync_compatible\nasync def get(name: str, default: str = None) -> Optional[str]:\n    \"\"\"\n    Get a variable by name. If doesn't exist return the default.\n    ```\n        from prefect import variables\n\n        @flow\n        def my_flow():\n            var = variables.get(\"my_var\")\n    ```\n    or\n    ```\n        from prefect import variables\n\n        @flow\n        async def my_flow():\n            var = await variables.get(\"my_var\")\n    ```\n    \"\"\"\n    variable = await _get_variable_by_name(name)\n    return variable.value if variable else default\n
    ","tags":["Python API","variables"]},{"location":"api-ref/prefect/blocks/core/","title":"core","text":"","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core","title":"prefect.blocks.core","text":"","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block","title":"Block","text":"

    Bases: BaseModel, ABC

    A base class for implementing a block that wraps an external service.

    This class can be defined with an arbitrary set of fields and methods, and couples business logic with data contained in an block document. _block_document_name, _block_document_id, _block_schema_id, and _block_type_id are reserved by Prefect as Block metadata fields, but otherwise a Block can implement arbitrary logic. Blocks can be instantiated without populating these metadata fields, but can only be used interactively, not with the Prefect API.

    Instead of the init method, a block implementation allows the definition of a block_initialization method that is called after initialization.

    Source code in prefect/blocks/core.py
    @register_base_type\n@instrument_method_calls_on_class_instances\nclass Block(BaseModel, ABC):\n    \"\"\"\n    A base class for implementing a block that wraps an external service.\n\n    This class can be defined with an arbitrary set of fields and methods, and\n    couples business logic with data contained in an block document.\n    `_block_document_name`, `_block_document_id`, `_block_schema_id`, and\n    `_block_type_id` are reserved by Prefect as Block metadata fields, but\n    otherwise a Block can implement arbitrary logic. Blocks can be instantiated\n    without populating these metadata fields, but can only be used interactively,\n    not with the Prefect API.\n\n    Instead of the __init__ method, a block implementation allows the\n    definition of a `block_initialization` method that is called after\n    initialization.\n    \"\"\"\n\n    class Config:\n        extra = \"allow\"\n\n        json_encoders = {SecretDict: lambda v: v.dict()}\n\n        @staticmethod\n        def schema_extra(schema: Dict[str, Any], model: Type[\"Block\"]):\n            \"\"\"\n            Customizes Pydantic's schema generation feature to add blocks related information.\n            \"\"\"\n            schema[\"block_type_slug\"] = model.get_block_type_slug()\n            # Ensures args and code examples aren't included in the schema\n            description = model.get_description()\n            if description:\n                schema[\"description\"] = description\n            else:\n                # Prevent the description of the base class from being included in the schema\n                schema.pop(\"description\", None)\n\n            # create a list of secret field names\n            # secret fields include both top-level keys and dot-delimited nested secret keys\n            # A wildcard (*) means that all fields under a given key are secret.\n            # for example: [\"x\", \"y\", \"z.*\", \"child.a\"]\n            # means the top-level keys \"x\" and \"y\", all keys under \"z\", and the key \"a\" of a block\n            # nested under the \"child\" key are all secret. There is no limit to nesting.\n            secrets = schema[\"secret_fields\"] = []\n            for field in model.__fields__.values():\n                _collect_secret_fields(field.name, field.type_, secrets)\n\n            # create block schema references\n            refs = schema[\"block_schema_references\"] = {}\n            for field in model.__fields__.values():\n                if Block.is_block_class(field.type_):\n                    refs[field.name] = field.type_._to_block_schema_reference_dict()\n                if get_origin(field.type_) is Union:\n                    for type_ in get_args(field.type_):\n                        if Block.is_block_class(type_):\n                            if isinstance(refs.get(field.name), list):\n                                refs[field.name].append(\n                                    type_._to_block_schema_reference_dict()\n                                )\n                            elif isinstance(refs.get(field.name), dict):\n                                refs[field.name] = [\n                                    refs[field.name],\n                                    type_._to_block_schema_reference_dict(),\n                                ]\n                            else:\n                                refs[field.name] = (\n                                    type_._to_block_schema_reference_dict()\n                                )\n\n    def __init__(self, *args, **kwargs):\n        super().__init__(*args, **kwargs)\n        self.block_initialization()\n\n    def __str__(self) -> str:\n        return self.__repr__()\n\n    def __repr_args__(self):\n        repr_args = super().__repr_args__()\n        data_keys = self.schema()[\"properties\"].keys()\n        return [\n            (key, value) for key, value in repr_args if key is None or key in data_keys\n        ]\n\n    def block_initialization(self) -> None:\n        pass\n\n    # -- private class variables\n    # set by the class itself\n\n    # Attribute to customize the name of the block type created\n    # when the block is registered with the API. If not set, block\n    # type name will default to the class name.\n    _block_type_name: Optional[str] = None\n    _block_type_slug: Optional[str] = None\n\n    # Attributes used to set properties on a block type when registered\n    # with the API.\n    _logo_url: Optional[HttpUrl] = None\n    _documentation_url: Optional[HttpUrl] = None\n    _description: Optional[str] = None\n    _code_example: Optional[str] = None\n\n    # -- private instance variables\n    # these are set when blocks are loaded from the API\n    _block_type_id: Optional[UUID] = None\n    _block_schema_id: Optional[UUID] = None\n    _block_schema_capabilities: Optional[List[str]] = None\n    _block_schema_version: Optional[str] = None\n    _block_document_id: Optional[UUID] = None\n    _block_document_name: Optional[str] = None\n    _is_anonymous: Optional[bool] = None\n\n    # Exclude `save` as it uses the `sync_compatible` decorator and needs to be\n    # decorated directly.\n    _events_excluded_methods = [\"block_initialization\", \"save\", \"dict\"]\n\n    @classmethod\n    def __dispatch_key__(cls):\n        if cls.__name__ == \"Block\":\n            return None  # The base class is abstract\n        return block_schema_to_key(cls._to_block_schema())\n\n    @classmethod\n    def get_block_type_name(cls):\n        return cls._block_type_name or cls.__name__\n\n    @classmethod\n    def get_block_type_slug(cls):\n        return slugify(cls._block_type_slug or cls.get_block_type_name())\n\n    @classmethod\n    def get_block_capabilities(cls) -> FrozenSet[str]:\n        \"\"\"\n        Returns the block capabilities for this Block. Recursively collects all block\n        capabilities of all parent classes into a single frozenset.\n        \"\"\"\n        return frozenset(\n            {\n                c\n                for base in (cls,) + cls.__mro__\n                for c in getattr(base, \"_block_schema_capabilities\", []) or []\n            }\n        )\n\n    @classmethod\n    def _get_current_package_version(cls):\n        current_module = inspect.getmodule(cls)\n        if current_module:\n            top_level_module = sys.modules[\n                current_module.__name__.split(\".\")[0] or \"__main__\"\n            ]\n            try:\n                version = Version(top_level_module.__version__)\n                # Strips off any local version information\n                return version.base_version\n            except (AttributeError, InvalidVersion):\n                # Module does not have a __version__ attribute or is not a parsable format\n                pass\n        return DEFAULT_BLOCK_SCHEMA_VERSION\n\n    @classmethod\n    def get_block_schema_version(cls) -> str:\n        return cls._block_schema_version or cls._get_current_package_version()\n\n    @classmethod\n    def _to_block_schema_reference_dict(cls):\n        return dict(\n            block_type_slug=cls.get_block_type_slug(),\n            block_schema_checksum=cls._calculate_schema_checksum(),\n        )\n\n    @classmethod\n    def _calculate_schema_checksum(\n        cls, block_schema_fields: Optional[Dict[str, Any]] = None\n    ):\n        \"\"\"\n        Generates a unique hash for the underlying schema of block.\n\n        Args:\n            block_schema_fields: Dictionary detailing block schema fields to generate a\n                checksum for. The fields of the current class is used if this parameter\n                is not provided.\n\n        Returns:\n            str: The calculated checksum prefixed with the hashing algorithm used.\n        \"\"\"\n        block_schema_fields = (\n            cls.schema() if block_schema_fields is None else block_schema_fields\n        )\n        fields_for_checksum = remove_nested_keys([\"secret_fields\"], block_schema_fields)\n        if fields_for_checksum.get(\"definitions\"):\n            non_block_definitions = _get_non_block_reference_definitions(\n                fields_for_checksum, fields_for_checksum[\"definitions\"]\n            )\n            if non_block_definitions:\n                fields_for_checksum[\"definitions\"] = non_block_definitions\n            else:\n                # Pop off definitions entirely instead of empty dict for consistency\n                # with the OpenAPI specification\n                fields_for_checksum.pop(\"definitions\")\n        checksum = hash_objects(fields_for_checksum, hash_algo=hashlib.sha256)\n        if checksum is None:\n            raise ValueError(\"Unable to compute checksum for block schema\")\n        else:\n            return f\"sha256:{checksum}\"\n\n    def _to_block_document(\n        self,\n        name: Optional[str] = None,\n        block_schema_id: Optional[UUID] = None,\n        block_type_id: Optional[UUID] = None,\n        is_anonymous: Optional[bool] = None,\n    ) -> BlockDocument:\n        \"\"\"\n        Creates the corresponding block document based on the data stored in a block.\n        The corresponding block document name, block type ID, and block schema ID must\n        either be passed into the method or configured on the block.\n\n        Args:\n            name: The name of the created block document. Not required if anonymous.\n            block_schema_id: UUID of the corresponding block schema.\n            block_type_id: UUID of the corresponding block type.\n            is_anonymous: if True, an anonymous block is created. Anonymous\n                blocks are not displayed in the UI and used primarily for system\n                operations and features that need to automatically generate blocks.\n\n        Returns:\n            BlockDocument: Corresponding block document\n                populated with the block's configured data.\n        \"\"\"\n        if is_anonymous is None:\n            is_anonymous = self._is_anonymous or False\n\n        # name must be present if not anonymous\n        if not is_anonymous and not name and not self._block_document_name:\n            raise ValueError(\"No name provided, either as an argument or on the block.\")\n\n        if not block_schema_id and not self._block_schema_id:\n            raise ValueError(\n                \"No block schema ID provided, either as an argument or on the block.\"\n            )\n        if not block_type_id and not self._block_type_id:\n            raise ValueError(\n                \"No block type ID provided, either as an argument or on the block.\"\n            )\n\n        # The keys passed to `include` must NOT be aliases, else some items will be missed\n        # i.e. must do `self.schema_` vs `self.schema` to get a `schema_ = Field(alias=\"schema\")`\n        # reported from https://github.com/PrefectHQ/prefect-dbt/issues/54\n        data_keys = self.schema(by_alias=False)[\"properties\"].keys()\n\n        # `block_document_data`` must return the aliased version for it to show in the UI\n        block_document_data = self.dict(by_alias=True, include=data_keys)\n\n        # Iterate through and find blocks that already have saved block documents to\n        # create references to those saved block documents.\n        for key in data_keys:\n            field_value = getattr(self, key)\n            if (\n                isinstance(field_value, Block)\n                and field_value._block_document_id is not None\n            ):\n                block_document_data[key] = {\n                    \"$ref\": {\"block_document_id\": field_value._block_document_id}\n                }\n\n        return BlockDocument(\n            id=self._block_document_id or uuid4(),\n            name=(name or self._block_document_name) if not is_anonymous else None,\n            block_schema_id=block_schema_id or self._block_schema_id,\n            block_type_id=block_type_id or self._block_type_id,\n            data=block_document_data,\n            block_schema=self._to_block_schema(\n                block_type_id=block_type_id or self._block_type_id,\n            ),\n            block_type=self._to_block_type(),\n            is_anonymous=is_anonymous,\n        )\n\n    @classmethod\n    def _to_block_schema(cls, block_type_id: Optional[UUID] = None) -> BlockSchema:\n        \"\"\"\n        Creates the corresponding block schema of the block.\n        The corresponding block_type_id must either be passed into\n        the method or configured on the block.\n\n        Args:\n            block_type_id: UUID of the corresponding block type.\n\n        Returns:\n            BlockSchema: The corresponding block schema.\n        \"\"\"\n        fields = cls.schema()\n        return BlockSchema(\n            id=cls._block_schema_id if cls._block_schema_id is not None else uuid4(),\n            checksum=cls._calculate_schema_checksum(),\n            fields=fields,\n            block_type_id=block_type_id or cls._block_type_id,\n            block_type=cls._to_block_type(),\n            capabilities=list(cls.get_block_capabilities()),\n            version=cls.get_block_schema_version(),\n        )\n\n    @classmethod\n    def _parse_docstring(cls) -> List[DocstringSection]:\n        \"\"\"\n        Parses the docstring into list of DocstringSection objects.\n        Helper method used primarily to suppress irrelevant logs, e.g.\n        `<module>:11: No type or annotation for parameter 'write_json'`\n        because griffe is unable to parse the types from pydantic.BaseModel.\n        \"\"\"\n        with disable_logger(\"griffe.docstrings.google\"):\n            with disable_logger(\"griffe.agents.nodes\"):\n                docstring = Docstring(cls.__doc__)\n                parsed = parse(docstring, Parser.google)\n        return parsed\n\n    @classmethod\n    def get_description(cls) -> Optional[str]:\n        \"\"\"\n        Returns the description for the current block. Attempts to parse\n        description from class docstring if an override is not defined.\n        \"\"\"\n        description = cls._description\n        # If no description override has been provided, find the first text section\n        # and use that as the description\n        if description is None and cls.__doc__ is not None:\n            parsed = cls._parse_docstring()\n            parsed_description = next(\n                (\n                    section.as_dict().get(\"value\")\n                    for section in parsed\n                    if section.kind == DocstringSectionKind.text\n                ),\n                None,\n            )\n            if isinstance(parsed_description, str):\n                description = parsed_description.strip()\n        return description\n\n    @classmethod\n    def get_code_example(cls) -> Optional[str]:\n        \"\"\"\n        Returns the code example for the given block. Attempts to parse\n        code example from the class docstring if an override is not provided.\n        \"\"\"\n        code_example = (\n            dedent(cls._code_example) if cls._code_example is not None else None\n        )\n        # If no code example override has been provided, attempt to find a examples\n        # section or an admonition with the annotation \"example\" and use that as the\n        # code example\n        if code_example is None and cls.__doc__ is not None:\n            parsed = cls._parse_docstring()\n            for section in parsed:\n                # Section kind will be \"examples\" if Examples section heading is used.\n                if section.kind == DocstringSectionKind.examples:\n                    # Examples sections are made up of smaller sections that need to be\n                    # joined with newlines. Smaller sections are represented as tuples\n                    # with shape (DocstringSectionKind, str)\n                    code_example = \"\\n\".join(\n                        (part[1] for part in section.as_dict().get(\"value\", []))\n                    )\n                    break\n                # Section kind will be \"admonition\" if Example section heading is used.\n                if section.kind == DocstringSectionKind.admonition:\n                    value = section.as_dict().get(\"value\", {})\n                    if value.get(\"annotation\") == \"example\":\n                        code_example = value.get(\"description\")\n                        break\n\n        if code_example is None:\n            # If no code example has been specified or extracted from the class\n            # docstring, generate a sensible default\n            code_example = cls._generate_code_example()\n\n        return code_example\n\n    @classmethod\n    def _generate_code_example(cls) -> str:\n        \"\"\"Generates a default code example for the current class\"\"\"\n        qualified_name = to_qualified_name(cls)\n        module_str = \".\".join(qualified_name.split(\".\")[:-1])\n        class_name = cls.__name__\n        block_variable_name = f'{cls.get_block_type_slug().replace(\"-\", \"_\")}_block'\n\n        return dedent(\n            f\"\"\"\\\n        ```python\n        from {module_str} import {class_name}\n\n        {block_variable_name} = {class_name}.load(\"BLOCK_NAME\")\n        ```\"\"\"\n        )\n\n    @classmethod\n    def _to_block_type(cls) -> BlockType:\n        \"\"\"\n        Creates the corresponding block type of the block.\n\n        Returns:\n            BlockType: The corresponding block type.\n        \"\"\"\n        return BlockType(\n            id=cls._block_type_id or uuid4(),\n            slug=cls.get_block_type_slug(),\n            name=cls.get_block_type_name(),\n            logo_url=cls._logo_url,\n            documentation_url=cls._documentation_url,\n            description=cls.get_description(),\n            code_example=cls.get_code_example(),\n        )\n\n    @classmethod\n    def _from_block_document(cls, block_document: BlockDocument):\n        \"\"\"\n        Instantiates a block from a given block document. The corresponding block class\n        will be looked up in the block registry based on the corresponding block schema\n        of the provided block document.\n\n        Args:\n            block_document: The block document used to instantiate a block.\n\n        Raises:\n            ValueError: If the provided block document doesn't have a corresponding block\n                schema.\n\n        Returns:\n            Block: Hydrated block with data from block document.\n        \"\"\"\n        if block_document.block_schema is None:\n            raise ValueError(\n                \"Unable to determine block schema for provided block document\"\n            )\n\n        block_cls = (\n            cls\n            if cls.__name__ != \"Block\"\n            # Look up the block class by dispatch\n            else cls.get_block_class_from_schema(block_document.block_schema)\n        )\n\n        block_cls = instrument_method_calls_on_class_instances(block_cls)\n\n        block = block_cls.parse_obj(block_document.data)\n        block._block_document_id = block_document.id\n        block.__class__._block_schema_id = block_document.block_schema_id\n        block.__class__._block_type_id = block_document.block_type_id\n        block._block_document_name = block_document.name\n        block._is_anonymous = block_document.is_anonymous\n        block._define_metadata_on_nested_blocks(\n            block_document.block_document_references\n        )\n\n        # Due to the way blocks are loaded we can't directly instrument the\n        # `load` method and have the data be about the block document. Instead\n        # this will emit a proxy event for the load method so that block\n        # document data can be included instead of the event being about an\n        # 'anonymous' block.\n\n        emit_instance_method_called_event(block, \"load\", successful=True)\n\n        return block\n\n    def _event_kind(self) -> str:\n        return f\"prefect.block.{self.get_block_type_slug()}\"\n\n    def _event_method_called_resources(self) -> Optional[ResourceTuple]:\n        if not (self._block_document_id and self._block_document_name):\n            return None\n\n        return (\n            {\n                \"prefect.resource.id\": (\n                    f\"prefect.block-document.{self._block_document_id}\"\n                ),\n                \"prefect.resource.name\": self._block_document_name,\n            },\n            [\n                {\n                    \"prefect.resource.id\": (\n                        f\"prefect.block-type.{self.get_block_type_slug()}\"\n                    ),\n                    \"prefect.resource.role\": \"block-type\",\n                }\n            ],\n        )\n\n    @classmethod\n    def get_block_class_from_schema(cls: Type[Self], schema: BlockSchema) -> Type[Self]:\n        \"\"\"\n        Retrieve the block class implementation given a schema.\n        \"\"\"\n        return cls.get_block_class_from_key(block_schema_to_key(schema))\n\n    @classmethod\n    def get_block_class_from_key(cls: Type[Self], key: str) -> Type[Self]:\n        \"\"\"\n        Retrieve the block class implementation given a key.\n        \"\"\"\n        # Ensure collections are imported and have the opportunity to register types\n        # before looking up the block class\n        prefect.plugins.load_prefect_collections()\n\n        return lookup_type(cls, key)\n\n    def _define_metadata_on_nested_blocks(\n        self, block_document_references: Dict[str, Dict[str, Any]]\n    ):\n        \"\"\"\n        Recursively populates metadata fields on nested blocks based on the\n        provided block document references.\n        \"\"\"\n        for item in block_document_references.items():\n            field_name, block_document_reference = item\n            nested_block = getattr(self, field_name)\n            if isinstance(nested_block, Block):\n                nested_block_document_info = block_document_reference.get(\n                    \"block_document\", {}\n                )\n                nested_block._define_metadata_on_nested_blocks(\n                    nested_block_document_info.get(\"block_document_references\", {})\n                )\n                nested_block_document_id = nested_block_document_info.get(\"id\")\n                nested_block._block_document_id = (\n                    UUID(nested_block_document_id) if nested_block_document_id else None\n                )\n                nested_block._block_document_name = nested_block_document_info.get(\n                    \"name\"\n                )\n                nested_block._is_anonymous = nested_block_document_info.get(\n                    \"is_anonymous\"\n                )\n\n    @classmethod\n    @inject_client\n    async def _get_block_document(\n        cls,\n        name: str,\n        client: \"PrefectClient\" = None,\n    ):\n        if cls.__name__ == \"Block\":\n            block_type_slug, block_document_name = name.split(\"/\", 1)\n        else:\n            block_type_slug = cls.get_block_type_slug()\n            block_document_name = name\n\n        try:\n            block_document = await client.read_block_document_by_name(\n                name=block_document_name, block_type_slug=block_type_slug\n            )\n        except prefect.exceptions.ObjectNotFound as e:\n            raise ValueError(\n                f\"Unable to find block document named {block_document_name} for block\"\n                f\" type {block_type_slug}\"\n            ) from e\n\n        return block_document, block_document_name\n\n    @classmethod\n    @sync_compatible\n    @inject_client\n    async def load(\n        cls,\n        name: str,\n        validate: bool = True,\n        client: \"PrefectClient\" = None,\n    ):\n        \"\"\"\n        Retrieves data from the block document with the given name for the block type\n        that corresponds with the current class and returns an instantiated version of\n        the current class with the data stored in the block document.\n\n        If a block document for a given block type is saved with a different schema\n        than the current class calling `load`, a warning will be raised.\n\n        If the current class schema is a subset of the block document schema, the block\n        can be loaded as normal using the default `validate = True`.\n\n        If the current class schema is a superset of the block document schema, `load`\n        must be called with `validate` set to False to prevent a validation error. In\n        this case, the block attributes will default to `None` and must be set manually\n        and saved to a new block document before the block can be used as expected.\n\n        Args:\n            name: The name or slug of the block document. A block document slug is a\n                string with the format <block_type_slug>/<block_document_name>\n            validate: If False, the block document will be loaded without Pydantic\n                validating the block schema. This is useful if the block schema has\n                changed client-side since the block document referred to by `name` was saved.\n            client: The client to use to load the block document. If not provided, the\n                default client will be injected.\n\n        Raises:\n            ValueError: If the requested block document is not found.\n\n        Returns:\n            An instance of the current class hydrated with the data stored in the\n            block document with the specified name.\n\n        Examples:\n            Load from a Block subclass with a block document name:\n            ```python\n            class Custom(Block):\n                message: str\n\n            Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n            loaded_block = Custom.load(\"my-custom-message\")\n            ```\n\n            Load from Block with a block document slug:\n            ```python\n            class Custom(Block):\n                message: str\n\n            Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n            loaded_block = Block.load(\"custom/my-custom-message\")\n            ```\n\n            Migrate a block document to a new schema:\n            ```python\n            # original class\n            class Custom(Block):\n                message: str\n\n            Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n            # Updated class with new required field\n            class Custom(Block):\n                message: str\n                number_of_ducks: int\n\n            loaded_block = Custom.load(\"my-custom-message\", validate=False)\n\n            # Prints UserWarning about schema mismatch\n\n            loaded_block.number_of_ducks = 42\n\n            loaded_block.save(\"my-custom-message\", overwrite=True)\n            ```\n        \"\"\"\n        block_document, block_document_name = await cls._get_block_document(name)\n\n        try:\n            return cls._from_block_document(block_document)\n        except ValidationError as e:\n            if not validate:\n                missing_fields = tuple(err[\"loc\"][0] for err in e.errors())\n                missing_block_data = {field: None for field in missing_fields}\n                warnings.warn(\n                    f\"Could not fully load {block_document_name!r} of block type\"\n                    f\" {cls._block_type_slug!r} - this is likely because one or more\"\n                    \" required fields were added to the schema for\"\n                    f\" {cls.__name__!r} that did not exist on the class when this block\"\n                    \" was last saved. Please specify values for new field(s):\"\n                    f\" {listrepr(missing_fields)}, then run\"\n                    f' `{cls.__name__}.save(\"{block_document_name}\", overwrite=True)`,'\n                    \" and load this block again before attempting to use it.\"\n                )\n                return cls.construct(**block_document.data, **missing_block_data)\n            raise RuntimeError(\n                f\"Unable to load {block_document_name!r} of block type\"\n                f\" {cls._block_type_slug!r} due to failed validation. To load without\"\n                \" validation, try loading again with `validate=False`.\"\n            ) from e\n\n    @staticmethod\n    def is_block_class(block) -> bool:\n        return _is_subclass(block, Block)\n\n    @classmethod\n    @sync_compatible\n    @inject_client\n    async def register_type_and_schema(cls, client: \"PrefectClient\" = None):\n        \"\"\"\n        Makes block available for configuration with current Prefect API.\n        Recursively registers all nested blocks. Registration is idempotent.\n\n        Args:\n            client: Optional client to use for registering type and schema with the\n                Prefect API. A new client will be created and used if one is not\n                provided.\n        \"\"\"\n        if cls.__name__ == \"Block\":\n            raise InvalidBlockRegistration(\n                \"`register_type_and_schema` should be called on a Block \"\n                \"subclass and not on the Block class directly.\"\n            )\n        if ABC in getattr(cls, \"__bases__\", []):\n            raise InvalidBlockRegistration(\n                \"`register_type_and_schema` should be called on a Block \"\n                \"subclass and not on a Block interface class directly.\"\n            )\n\n        for field in cls.__fields__.values():\n            if Block.is_block_class(field.type_):\n                await field.type_.register_type_and_schema(client=client)\n            if get_origin(field.type_) is Union:\n                for type_ in get_args(field.type_):\n                    if Block.is_block_class(type_):\n                        await type_.register_type_and_schema(client=client)\n\n        try:\n            block_type = await client.read_block_type_by_slug(\n                slug=cls.get_block_type_slug()\n            )\n            cls._block_type_id = block_type.id\n            local_block_type = cls._to_block_type()\n            if _should_update_block_type(\n                local_block_type=local_block_type, server_block_type=block_type\n            ):\n                await client.update_block_type(\n                    block_type_id=block_type.id, block_type=local_block_type\n                )\n        except prefect.exceptions.ObjectNotFound:\n            block_type = await client.create_block_type(block_type=cls._to_block_type())\n            cls._block_type_id = block_type.id\n\n        try:\n            block_schema = await client.read_block_schema_by_checksum(\n                checksum=cls._calculate_schema_checksum(),\n                version=cls.get_block_schema_version(),\n            )\n        except prefect.exceptions.ObjectNotFound:\n            block_schema = await client.create_block_schema(\n                block_schema=cls._to_block_schema(block_type_id=block_type.id)\n            )\n\n        cls._block_schema_id = block_schema.id\n\n    @inject_client\n    async def _save(\n        self,\n        name: Optional[str] = None,\n        is_anonymous: bool = False,\n        overwrite: bool = False,\n        client: \"PrefectClient\" = None,\n    ):\n        \"\"\"\n        Saves the values of a block as a block document with an option to save as an\n        anonymous block document.\n\n        Args:\n            name: User specified name to give saved block document which can later be used to load the\n                block document.\n            is_anonymous: Boolean value specifying whether the block document is anonymous. Anonymous\n                blocks are intended for system use and are not shown in the UI. Anonymous blocks do not\n                require a user-supplied name.\n            overwrite: Boolean value specifying if values should be overwritten if a block document with\n                the specified name already exists.\n\n        Raises:\n            ValueError: If a name is not given and `is_anonymous` is `False` or a name is given and\n                `is_anonymous` is `True`.\n        \"\"\"\n        if name is None and not is_anonymous:\n            raise ValueError(\n                \"You're attempting to save a block document without a name. \"\n                \"Please either save a block document with a name or set \"\n                \"is_anonymous to True.\"\n            )\n\n        self._is_anonymous = is_anonymous\n\n        # Ensure block type and schema are registered before saving block document.\n        await self.register_type_and_schema(client=client)\n\n        try:\n            block_document = await client.create_block_document(\n                block_document=self._to_block_document(name=name)\n            )\n        except prefect.exceptions.ObjectAlreadyExists as err:\n            if overwrite:\n                block_document_id = self._block_document_id\n                if block_document_id is None:\n                    existing_block_document = await client.read_block_document_by_name(\n                        name=name, block_type_slug=self.get_block_type_slug()\n                    )\n                    block_document_id = existing_block_document.id\n                await client.update_block_document(\n                    block_document_id=block_document_id,\n                    block_document=self._to_block_document(name=name),\n                )\n                block_document = await client.read_block_document(\n                    block_document_id=block_document_id\n                )\n            else:\n                raise ValueError(\n                    \"You are attempting to save values with a name that is already in\"\n                    \" use for this block type. If you would like to overwrite the\"\n                    \" values that are saved, then save with `overwrite=True`.\"\n                ) from err\n\n        # Update metadata on block instance for later use.\n        self._block_document_name = block_document.name\n        self._block_document_id = block_document.id\n        return self._block_document_id\n\n    @sync_compatible\n    @instrument_instance_method_call()\n    async def save(\n        self, name: str, overwrite: bool = False, client: \"PrefectClient\" = None\n    ):\n        \"\"\"\n        Saves the values of a block as a block document.\n\n        Args:\n            name: User specified name to give saved block document which can later be used to load the\n                block document.\n            overwrite: Boolean value specifying if values should be overwritten if a block document with\n                the specified name already exists.\n\n        \"\"\"\n        document_id = await self._save(name=name, overwrite=overwrite, client=client)\n\n        return document_id\n\n    @classmethod\n    @sync_compatible\n    @inject_client\n    async def delete(\n        cls,\n        name: str,\n        client: \"PrefectClient\" = None,\n    ):\n        block_document, block_document_name = await cls._get_block_document(name)\n\n        await client.delete_block_document(block_document.id)\n\n    def _iter(self, *, include=None, exclude=None, **kwargs):\n        # Injects the `block_type_slug` into serialized payloads for dispatch\n        for key_value in super()._iter(include=include, exclude=exclude, **kwargs):\n            yield key_value\n\n        # Respect inclusion and exclusion still\n        if include and \"block_type_slug\" not in include:\n            return\n        if exclude and \"block_type_slug\" in exclude:\n            return\n\n        yield \"block_type_slug\", self.get_block_type_slug()\n\n    def __new__(cls: Type[Self], **kwargs) -> Self:\n        \"\"\"\n        Create an instance of the Block subclass type if a `block_type_slug` is\n        present in the data payload.\n        \"\"\"\n        block_type_slug = kwargs.pop(\"block_type_slug\", None)\n        if block_type_slug:\n            subcls = lookup_type(cls, dispatch_key=block_type_slug)\n            m = super().__new__(subcls)\n            # NOTE: This is a workaround for an obscure issue where copied models were\n            #       missing attributes. This pattern is from Pydantic's\n            #       `BaseModel._copy_and_set_values`.\n            #       The issue this fixes could not be reproduced in unit tests that\n            #       directly targeted dispatch handling and was only observed when\n            #       copying then saving infrastructure blocks on deployment models.\n            object.__setattr__(m, \"__dict__\", kwargs)\n            object.__setattr__(m, \"__fields_set__\", set(kwargs.keys()))\n            return m\n        else:\n            m = super().__new__(cls)\n            object.__setattr__(m, \"__dict__\", kwargs)\n            object.__setattr__(m, \"__fields_set__\", set(kwargs.keys()))\n            return m\n\n    def get_block_placeholder(self) -> str:\n        \"\"\"\n        Returns the block placeholder for the current block which can be used for\n        templating.\n\n        Returns:\n            str: The block placeholder for the current block in the format\n                `prefect.blocks.{block_type_name}.{block_document_name}`\n\n        Raises:\n            BlockNotSavedError: Raised if the block has not been saved.\n\n        If a block has not been saved, the return value will be `None`.\n        \"\"\"\n        block_document_name = self._block_document_name\n        if not block_document_name:\n            raise BlockNotSavedError(\n                \"Could not generate block placeholder for unsaved block.\"\n            )\n\n        return f\"prefect.blocks.{self.get_block_type_slug()}.{block_document_name}\"\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.Config","title":"Config","text":"Source code in prefect/blocks/core.py
    class Config:\n    extra = \"allow\"\n\n    json_encoders = {SecretDict: lambda v: v.dict()}\n\n    @staticmethod\n    def schema_extra(schema: Dict[str, Any], model: Type[\"Block\"]):\n        \"\"\"\n        Customizes Pydantic's schema generation feature to add blocks related information.\n        \"\"\"\n        schema[\"block_type_slug\"] = model.get_block_type_slug()\n        # Ensures args and code examples aren't included in the schema\n        description = model.get_description()\n        if description:\n            schema[\"description\"] = description\n        else:\n            # Prevent the description of the base class from being included in the schema\n            schema.pop(\"description\", None)\n\n        # create a list of secret field names\n        # secret fields include both top-level keys and dot-delimited nested secret keys\n        # A wildcard (*) means that all fields under a given key are secret.\n        # for example: [\"x\", \"y\", \"z.*\", \"child.a\"]\n        # means the top-level keys \"x\" and \"y\", all keys under \"z\", and the key \"a\" of a block\n        # nested under the \"child\" key are all secret. There is no limit to nesting.\n        secrets = schema[\"secret_fields\"] = []\n        for field in model.__fields__.values():\n            _collect_secret_fields(field.name, field.type_, secrets)\n\n        # create block schema references\n        refs = schema[\"block_schema_references\"] = {}\n        for field in model.__fields__.values():\n            if Block.is_block_class(field.type_):\n                refs[field.name] = field.type_._to_block_schema_reference_dict()\n            if get_origin(field.type_) is Union:\n                for type_ in get_args(field.type_):\n                    if Block.is_block_class(type_):\n                        if isinstance(refs.get(field.name), list):\n                            refs[field.name].append(\n                                type_._to_block_schema_reference_dict()\n                            )\n                        elif isinstance(refs.get(field.name), dict):\n                            refs[field.name] = [\n                                refs[field.name],\n                                type_._to_block_schema_reference_dict(),\n                            ]\n                        else:\n                            refs[field.name] = (\n                                type_._to_block_schema_reference_dict()\n                            )\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.Config.schema_extra","title":"schema_extra staticmethod","text":"

    Customizes Pydantic's schema generation feature to add blocks related information.

    Source code in prefect/blocks/core.py
    @staticmethod\ndef schema_extra(schema: Dict[str, Any], model: Type[\"Block\"]):\n    \"\"\"\n    Customizes Pydantic's schema generation feature to add blocks related information.\n    \"\"\"\n    schema[\"block_type_slug\"] = model.get_block_type_slug()\n    # Ensures args and code examples aren't included in the schema\n    description = model.get_description()\n    if description:\n        schema[\"description\"] = description\n    else:\n        # Prevent the description of the base class from being included in the schema\n        schema.pop(\"description\", None)\n\n    # create a list of secret field names\n    # secret fields include both top-level keys and dot-delimited nested secret keys\n    # A wildcard (*) means that all fields under a given key are secret.\n    # for example: [\"x\", \"y\", \"z.*\", \"child.a\"]\n    # means the top-level keys \"x\" and \"y\", all keys under \"z\", and the key \"a\" of a block\n    # nested under the \"child\" key are all secret. There is no limit to nesting.\n    secrets = schema[\"secret_fields\"] = []\n    for field in model.__fields__.values():\n        _collect_secret_fields(field.name, field.type_, secrets)\n\n    # create block schema references\n    refs = schema[\"block_schema_references\"] = {}\n    for field in model.__fields__.values():\n        if Block.is_block_class(field.type_):\n            refs[field.name] = field.type_._to_block_schema_reference_dict()\n        if get_origin(field.type_) is Union:\n            for type_ in get_args(field.type_):\n                if Block.is_block_class(type_):\n                    if isinstance(refs.get(field.name), list):\n                        refs[field.name].append(\n                            type_._to_block_schema_reference_dict()\n                        )\n                    elif isinstance(refs.get(field.name), dict):\n                        refs[field.name] = [\n                            refs[field.name],\n                            type_._to_block_schema_reference_dict(),\n                        ]\n                    else:\n                        refs[field.name] = (\n                            type_._to_block_schema_reference_dict()\n                        )\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_block_capabilities","title":"get_block_capabilities classmethod","text":"

    Returns the block capabilities for this Block. Recursively collects all block capabilities of all parent classes into a single frozenset.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_block_capabilities(cls) -> FrozenSet[str]:\n    \"\"\"\n    Returns the block capabilities for this Block. Recursively collects all block\n    capabilities of all parent classes into a single frozenset.\n    \"\"\"\n    return frozenset(\n        {\n            c\n            for base in (cls,) + cls.__mro__\n            for c in getattr(base, \"_block_schema_capabilities\", []) or []\n        }\n    )\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_block_class_from_key","title":"get_block_class_from_key classmethod","text":"

    Retrieve the block class implementation given a key.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_block_class_from_key(cls: Type[Self], key: str) -> Type[Self]:\n    \"\"\"\n    Retrieve the block class implementation given a key.\n    \"\"\"\n    # Ensure collections are imported and have the opportunity to register types\n    # before looking up the block class\n    prefect.plugins.load_prefect_collections()\n\n    return lookup_type(cls, key)\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_block_class_from_schema","title":"get_block_class_from_schema classmethod","text":"

    Retrieve the block class implementation given a schema.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_block_class_from_schema(cls: Type[Self], schema: BlockSchema) -> Type[Self]:\n    \"\"\"\n    Retrieve the block class implementation given a schema.\n    \"\"\"\n    return cls.get_block_class_from_key(block_schema_to_key(schema))\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_block_placeholder","title":"get_block_placeholder","text":"

    Returns the block placeholder for the current block which can be used for templating.

    Returns:

    Name Type Description str str

    The block placeholder for the current block in the format prefect.blocks.{block_type_name}.{block_document_name}

    Raises:

    Type Description BlockNotSavedError

    Raised if the block has not been saved.

    If a block has not been saved, the return value will be None.

    Source code in prefect/blocks/core.py
    def get_block_placeholder(self) -> str:\n    \"\"\"\n    Returns the block placeholder for the current block which can be used for\n    templating.\n\n    Returns:\n        str: The block placeholder for the current block in the format\n            `prefect.blocks.{block_type_name}.{block_document_name}`\n\n    Raises:\n        BlockNotSavedError: Raised if the block has not been saved.\n\n    If a block has not been saved, the return value will be `None`.\n    \"\"\"\n    block_document_name = self._block_document_name\n    if not block_document_name:\n        raise BlockNotSavedError(\n            \"Could not generate block placeholder for unsaved block.\"\n        )\n\n    return f\"prefect.blocks.{self.get_block_type_slug()}.{block_document_name}\"\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_code_example","title":"get_code_example classmethod","text":"

    Returns the code example for the given block. Attempts to parse code example from the class docstring if an override is not provided.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_code_example(cls) -> Optional[str]:\n    \"\"\"\n    Returns the code example for the given block. Attempts to parse\n    code example from the class docstring if an override is not provided.\n    \"\"\"\n    code_example = (\n        dedent(cls._code_example) if cls._code_example is not None else None\n    )\n    # If no code example override has been provided, attempt to find a examples\n    # section or an admonition with the annotation \"example\" and use that as the\n    # code example\n    if code_example is None and cls.__doc__ is not None:\n        parsed = cls._parse_docstring()\n        for section in parsed:\n            # Section kind will be \"examples\" if Examples section heading is used.\n            if section.kind == DocstringSectionKind.examples:\n                # Examples sections are made up of smaller sections that need to be\n                # joined with newlines. Smaller sections are represented as tuples\n                # with shape (DocstringSectionKind, str)\n                code_example = \"\\n\".join(\n                    (part[1] for part in section.as_dict().get(\"value\", []))\n                )\n                break\n            # Section kind will be \"admonition\" if Example section heading is used.\n            if section.kind == DocstringSectionKind.admonition:\n                value = section.as_dict().get(\"value\", {})\n                if value.get(\"annotation\") == \"example\":\n                    code_example = value.get(\"description\")\n                    break\n\n    if code_example is None:\n        # If no code example has been specified or extracted from the class\n        # docstring, generate a sensible default\n        code_example = cls._generate_code_example()\n\n    return code_example\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.get_description","title":"get_description classmethod","text":"

    Returns the description for the current block. Attempts to parse description from class docstring if an override is not defined.

    Source code in prefect/blocks/core.py
    @classmethod\ndef get_description(cls) -> Optional[str]:\n    \"\"\"\n    Returns the description for the current block. Attempts to parse\n    description from class docstring if an override is not defined.\n    \"\"\"\n    description = cls._description\n    # If no description override has been provided, find the first text section\n    # and use that as the description\n    if description is None and cls.__doc__ is not None:\n        parsed = cls._parse_docstring()\n        parsed_description = next(\n            (\n                section.as_dict().get(\"value\")\n                for section in parsed\n                if section.kind == DocstringSectionKind.text\n            ),\n            None,\n        )\n        if isinstance(parsed_description, str):\n            description = parsed_description.strip()\n    return description\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.load","title":"load async classmethod","text":"

    Retrieves data from the block document with the given name for the block type that corresponds with the current class and returns an instantiated version of the current class with the data stored in the block document.

    If a block document for a given block type is saved with a different schema than the current class calling load, a warning will be raised.

    If the current class schema is a subset of the block document schema, the block can be loaded as normal using the default validate = True.

    If the current class schema is a superset of the block document schema, load must be called with validate set to False to prevent a validation error. In this case, the block attributes will default to None and must be set manually and saved to a new block document before the block can be used as expected.

    Parameters:

    Name Type Description Default name str

    The name or slug of the block document. A block document slug is a string with the format / required validate bool

    If False, the block document will be loaded without Pydantic validating the block schema. This is useful if the block schema has changed client-side since the block document referred to by name was saved.

    True client PrefectClient

    The client to use to load the block document. If not provided, the default client will be injected.

    None

    Raises:

    Type Description ValueError

    If the requested block document is not found.

    Returns:

    Type Description

    An instance of the current class hydrated with the data stored in the

    block document with the specified name.

    Examples:

    Load from a Block subclass with a block document name:

    class Custom(Block):\n    message: str\n\nCustom(message=\"Hello!\").save(\"my-custom-message\")\n\nloaded_block = Custom.load(\"my-custom-message\")\n

    Load from Block with a block document slug:

    class Custom(Block):\n    message: str\n\nCustom(message=\"Hello!\").save(\"my-custom-message\")\n\nloaded_block = Block.load(\"custom/my-custom-message\")\n

    Migrate a block document to a new schema:

    # original class\nclass Custom(Block):\n    message: str\n\nCustom(message=\"Hello!\").save(\"my-custom-message\")\n\n# Updated class with new required field\nclass Custom(Block):\n    message: str\n    number_of_ducks: int\n\nloaded_block = Custom.load(\"my-custom-message\", validate=False)\n\n# Prints UserWarning about schema mismatch\n\nloaded_block.number_of_ducks = 42\n\nloaded_block.save(\"my-custom-message\", overwrite=True)\n

    Source code in prefect/blocks/core.py
    @classmethod\n@sync_compatible\n@inject_client\nasync def load(\n    cls,\n    name: str,\n    validate: bool = True,\n    client: \"PrefectClient\" = None,\n):\n    \"\"\"\n    Retrieves data from the block document with the given name for the block type\n    that corresponds with the current class and returns an instantiated version of\n    the current class with the data stored in the block document.\n\n    If a block document for a given block type is saved with a different schema\n    than the current class calling `load`, a warning will be raised.\n\n    If the current class schema is a subset of the block document schema, the block\n    can be loaded as normal using the default `validate = True`.\n\n    If the current class schema is a superset of the block document schema, `load`\n    must be called with `validate` set to False to prevent a validation error. In\n    this case, the block attributes will default to `None` and must be set manually\n    and saved to a new block document before the block can be used as expected.\n\n    Args:\n        name: The name or slug of the block document. A block document slug is a\n            string with the format <block_type_slug>/<block_document_name>\n        validate: If False, the block document will be loaded without Pydantic\n            validating the block schema. This is useful if the block schema has\n            changed client-side since the block document referred to by `name` was saved.\n        client: The client to use to load the block document. If not provided, the\n            default client will be injected.\n\n    Raises:\n        ValueError: If the requested block document is not found.\n\n    Returns:\n        An instance of the current class hydrated with the data stored in the\n        block document with the specified name.\n\n    Examples:\n        Load from a Block subclass with a block document name:\n        ```python\n        class Custom(Block):\n            message: str\n\n        Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n        loaded_block = Custom.load(\"my-custom-message\")\n        ```\n\n        Load from Block with a block document slug:\n        ```python\n        class Custom(Block):\n            message: str\n\n        Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n        loaded_block = Block.load(\"custom/my-custom-message\")\n        ```\n\n        Migrate a block document to a new schema:\n        ```python\n        # original class\n        class Custom(Block):\n            message: str\n\n        Custom(message=\"Hello!\").save(\"my-custom-message\")\n\n        # Updated class with new required field\n        class Custom(Block):\n            message: str\n            number_of_ducks: int\n\n        loaded_block = Custom.load(\"my-custom-message\", validate=False)\n\n        # Prints UserWarning about schema mismatch\n\n        loaded_block.number_of_ducks = 42\n\n        loaded_block.save(\"my-custom-message\", overwrite=True)\n        ```\n    \"\"\"\n    block_document, block_document_name = await cls._get_block_document(name)\n\n    try:\n        return cls._from_block_document(block_document)\n    except ValidationError as e:\n        if not validate:\n            missing_fields = tuple(err[\"loc\"][0] for err in e.errors())\n            missing_block_data = {field: None for field in missing_fields}\n            warnings.warn(\n                f\"Could not fully load {block_document_name!r} of block type\"\n                f\" {cls._block_type_slug!r} - this is likely because one or more\"\n                \" required fields were added to the schema for\"\n                f\" {cls.__name__!r} that did not exist on the class when this block\"\n                \" was last saved. Please specify values for new field(s):\"\n                f\" {listrepr(missing_fields)}, then run\"\n                f' `{cls.__name__}.save(\"{block_document_name}\", overwrite=True)`,'\n                \" and load this block again before attempting to use it.\"\n            )\n            return cls.construct(**block_document.data, **missing_block_data)\n        raise RuntimeError(\n            f\"Unable to load {block_document_name!r} of block type\"\n            f\" {cls._block_type_slug!r} due to failed validation. To load without\"\n            \" validation, try loading again with `validate=False`.\"\n        ) from e\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.register_type_and_schema","title":"register_type_and_schema async classmethod","text":"

    Makes block available for configuration with current Prefect API. Recursively registers all nested blocks. Registration is idempotent.

    Parameters:

    Name Type Description Default client PrefectClient

    Optional client to use for registering type and schema with the Prefect API. A new client will be created and used if one is not provided.

    None Source code in prefect/blocks/core.py
    @classmethod\n@sync_compatible\n@inject_client\nasync def register_type_and_schema(cls, client: \"PrefectClient\" = None):\n    \"\"\"\n    Makes block available for configuration with current Prefect API.\n    Recursively registers all nested blocks. Registration is idempotent.\n\n    Args:\n        client: Optional client to use for registering type and schema with the\n            Prefect API. A new client will be created and used if one is not\n            provided.\n    \"\"\"\n    if cls.__name__ == \"Block\":\n        raise InvalidBlockRegistration(\n            \"`register_type_and_schema` should be called on a Block \"\n            \"subclass and not on the Block class directly.\"\n        )\n    if ABC in getattr(cls, \"__bases__\", []):\n        raise InvalidBlockRegistration(\n            \"`register_type_and_schema` should be called on a Block \"\n            \"subclass and not on a Block interface class directly.\"\n        )\n\n    for field in cls.__fields__.values():\n        if Block.is_block_class(field.type_):\n            await field.type_.register_type_and_schema(client=client)\n        if get_origin(field.type_) is Union:\n            for type_ in get_args(field.type_):\n                if Block.is_block_class(type_):\n                    await type_.register_type_and_schema(client=client)\n\n    try:\n        block_type = await client.read_block_type_by_slug(\n            slug=cls.get_block_type_slug()\n        )\n        cls._block_type_id = block_type.id\n        local_block_type = cls._to_block_type()\n        if _should_update_block_type(\n            local_block_type=local_block_type, server_block_type=block_type\n        ):\n            await client.update_block_type(\n                block_type_id=block_type.id, block_type=local_block_type\n            )\n    except prefect.exceptions.ObjectNotFound:\n        block_type = await client.create_block_type(block_type=cls._to_block_type())\n        cls._block_type_id = block_type.id\n\n    try:\n        block_schema = await client.read_block_schema_by_checksum(\n            checksum=cls._calculate_schema_checksum(),\n            version=cls.get_block_schema_version(),\n        )\n    except prefect.exceptions.ObjectNotFound:\n        block_schema = await client.create_block_schema(\n            block_schema=cls._to_block_schema(block_type_id=block_type.id)\n        )\n\n    cls._block_schema_id = block_schema.id\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.Block.save","title":"save async","text":"

    Saves the values of a block as a block document.

    Parameters:

    Name Type Description Default name str

    User specified name to give saved block document which can later be used to load the block document.

    required overwrite bool

    Boolean value specifying if values should be overwritten if a block document with the specified name already exists.

    False Source code in prefect/blocks/core.py
    @sync_compatible\n@instrument_instance_method_call()\nasync def save(\n    self, name: str, overwrite: bool = False, client: \"PrefectClient\" = None\n):\n    \"\"\"\n    Saves the values of a block as a block document.\n\n    Args:\n        name: User specified name to give saved block document which can later be used to load the\n            block document.\n        overwrite: Boolean value specifying if values should be overwritten if a block document with\n            the specified name already exists.\n\n    \"\"\"\n    document_id = await self._save(name=name, overwrite=overwrite, client=client)\n\n    return document_id\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.BlockNotSavedError","title":"BlockNotSavedError","text":"

    Bases: RuntimeError

    Raised when a given block is not saved and an operation that requires the block to be saved is attempted.

    Source code in prefect/blocks/core.py
    class BlockNotSavedError(RuntimeError):\n    \"\"\"\n    Raised when a given block is not saved and an operation that requires\n    the block to be saved is attempted.\n    \"\"\"\n\n    pass\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.InvalidBlockRegistration","title":"InvalidBlockRegistration","text":"

    Bases: Exception

    Raised on attempted registration of the base Block class or a Block interface class

    Source code in prefect/blocks/core.py
    class InvalidBlockRegistration(Exception):\n    \"\"\"\n    Raised on attempted registration of the base Block\n    class or a Block interface class\n    \"\"\"\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/core/#prefect.blocks.core.block_schema_to_key","title":"block_schema_to_key","text":"

    Defines the unique key used to lookup the Block class for a given schema.

    Source code in prefect/blocks/core.py
    def block_schema_to_key(schema: BlockSchema) -> str:\n    \"\"\"\n    Defines the unique key used to lookup the Block class for a given schema.\n    \"\"\"\n    return f\"{schema.block_type.slug}\"\n
    ","tags":["Python API","blocks"]},{"location":"api-ref/prefect/blocks/fields/","title":"fields","text":"","tags":["Python API","fields"]},{"location":"api-ref/prefect/blocks/fields/#prefect.blocks.fields","title":"prefect.blocks.fields","text":"","tags":["Python API","fields"]},{"location":"api-ref/prefect/blocks/kubernetes/","title":"kubernetes","text":"","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes","title":"prefect.blocks.kubernetes","text":"","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig","title":"KubernetesClusterConfig","text":"

    Bases: Block

    Stores configuration for interaction with Kubernetes clusters.

    See from_file for creation.

    Attributes:

    Name Type Description config Dict

    The entire loaded YAML contents of a kubectl config file

    context_name str

    The name of the kubectl context to use

    Example

    Load a saved Kubernetes cluster config:

    from prefect.blocks.kubernetes import KubernetesClusterConfig\n\ncluster_config_block = KubernetesClusterConfig.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/kubernetes.py
    class KubernetesClusterConfig(Block):\n    \"\"\"\n    Stores configuration for interaction with Kubernetes clusters.\n\n    See `from_file` for creation.\n\n    Attributes:\n        config: The entire loaded YAML contents of a kubectl config file\n        context_name: The name of the kubectl context to use\n\n    Example:\n        Load a saved Kubernetes cluster config:\n        ```python\n        from prefect.blocks.kubernetes import KubernetesClusterConfig\n\n        cluster_config_block = KubernetesClusterConfig.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Kubernetes Cluster Config\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/2d0b896006ad463b49c28aaac14f31e00e32cfab-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig\"\n\n    config: Dict = Field(\n        default=..., description=\"The entire contents of a kubectl config file.\"\n    )\n    context_name: str = Field(\n        default=..., description=\"The name of the kubectl context to use.\"\n    )\n\n    @validator(\"config\", pre=True)\n    def parse_yaml_config(cls, value):\n        if isinstance(value, str):\n            return yaml.safe_load(value)\n        return value\n\n    @classmethod\n    def from_file(cls: Type[Self], path: Path = None, context_name: str = None) -> Self:\n        \"\"\"\n        Create a cluster config from the a Kubernetes config file.\n\n        By default, the current context in the default Kubernetes config file will be\n        used.\n\n        An alternative file or context may be specified.\n\n        The entire config file will be loaded and stored.\n        \"\"\"\n        kube_config = kubernetes.config.kube_config\n\n        path = Path(path or kube_config.KUBE_CONFIG_DEFAULT_LOCATION)\n        path = path.expanduser().resolve()\n\n        # Determine the context\n        existing_contexts, current_context = kube_config.list_kube_config_contexts(\n            config_file=str(path)\n        )\n        context_names = {ctx[\"name\"] for ctx in existing_contexts}\n        if context_name:\n            if context_name not in context_names:\n                raise ValueError(\n                    f\"Context {context_name!r} not found. \"\n                    f\"Specify one of: {listrepr(context_names, sep=', ')}.\"\n                )\n        else:\n            context_name = current_context[\"name\"]\n\n        # Load the entire config file\n        config_file_contents = path.read_text()\n        config_dict = yaml.safe_load(config_file_contents)\n\n        return cls(config=config_dict, context_name=context_name)\n\n    def get_api_client(self) -> \"ApiClient\":\n        \"\"\"\n        Returns a Kubernetes API client for this cluster config.\n        \"\"\"\n        return kubernetes.config.kube_config.new_client_from_config_dict(\n            config_dict=self.config, context=self.context_name\n        )\n\n    def configure_client(self) -> None:\n        \"\"\"\n        Activates this cluster configuration by loading the configuration into the\n        Kubernetes Python client. After calling this, Kubernetes API clients can use\n        this config's context.\n        \"\"\"\n        kubernetes.config.kube_config.load_kube_config_from_dict(\n            config_dict=self.config, context=self.context_name\n        )\n
    ","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig.configure_client","title":"configure_client","text":"

    Activates this cluster configuration by loading the configuration into the Kubernetes Python client. After calling this, Kubernetes API clients can use this config's context.

    Source code in prefect/blocks/kubernetes.py
    def configure_client(self) -> None:\n    \"\"\"\n    Activates this cluster configuration by loading the configuration into the\n    Kubernetes Python client. After calling this, Kubernetes API clients can use\n    this config's context.\n    \"\"\"\n    kubernetes.config.kube_config.load_kube_config_from_dict(\n        config_dict=self.config, context=self.context_name\n    )\n
    ","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig.from_file","title":"from_file classmethod","text":"

    Create a cluster config from the a Kubernetes config file.

    By default, the current context in the default Kubernetes config file will be used.

    An alternative file or context may be specified.

    The entire config file will be loaded and stored.

    Source code in prefect/blocks/kubernetes.py
    @classmethod\ndef from_file(cls: Type[Self], path: Path = None, context_name: str = None) -> Self:\n    \"\"\"\n    Create a cluster config from the a Kubernetes config file.\n\n    By default, the current context in the default Kubernetes config file will be\n    used.\n\n    An alternative file or context may be specified.\n\n    The entire config file will be loaded and stored.\n    \"\"\"\n    kube_config = kubernetes.config.kube_config\n\n    path = Path(path or kube_config.KUBE_CONFIG_DEFAULT_LOCATION)\n    path = path.expanduser().resolve()\n\n    # Determine the context\n    existing_contexts, current_context = kube_config.list_kube_config_contexts(\n        config_file=str(path)\n    )\n    context_names = {ctx[\"name\"] for ctx in existing_contexts}\n    if context_name:\n        if context_name not in context_names:\n            raise ValueError(\n                f\"Context {context_name!r} not found. \"\n                f\"Specify one of: {listrepr(context_names, sep=', ')}.\"\n            )\n    else:\n        context_name = current_context[\"name\"]\n\n    # Load the entire config file\n    config_file_contents = path.read_text()\n    config_dict = yaml.safe_load(config_file_contents)\n\n    return cls(config=config_dict, context_name=context_name)\n
    ","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/kubernetes/#prefect.blocks.kubernetes.KubernetesClusterConfig.get_api_client","title":"get_api_client","text":"

    Returns a Kubernetes API client for this cluster config.

    Source code in prefect/blocks/kubernetes.py
    def get_api_client(self) -> \"ApiClient\":\n    \"\"\"\n    Returns a Kubernetes API client for this cluster config.\n    \"\"\"\n    return kubernetes.config.kube_config.new_client_from_config_dict(\n        config_dict=self.config, context=self.context_name\n    )\n
    ","tags":["Python API","blocks","Kubernetes"]},{"location":"api-ref/prefect/blocks/notifications/","title":"notifications","text":"","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications","title":"prefect.blocks.notifications","text":"","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.AbstractAppriseNotificationBlock","title":"AbstractAppriseNotificationBlock","text":"

    Bases: NotificationBlock, ABC

    An abstract class for sending notifications using Apprise.

    Source code in prefect/blocks/notifications.py
    class AbstractAppriseNotificationBlock(NotificationBlock, ABC):\n    \"\"\"\n    An abstract class for sending notifications using Apprise.\n    \"\"\"\n\n    notify_type: Literal[\"prefect_default\", \"info\", \"success\", \"warning\", \"failure\"] = (\n        Field(\n            default=PREFECT_NOTIFY_TYPE_DEFAULT,\n            description=(\n                \"The type of notification being performed; the prefect_default \"\n                \"is a plain notification that does not attach an image.\"\n            ),\n        )\n    )\n\n    def __init__(self, *args, **kwargs):\n        import apprise\n\n        if PREFECT_NOTIFY_TYPE_DEFAULT not in apprise.NOTIFY_TYPES:\n            apprise.NOTIFY_TYPES += (PREFECT_NOTIFY_TYPE_DEFAULT,)\n\n        super().__init__(*args, **kwargs)\n\n    def _start_apprise_client(self, url: SecretStr):\n        from apprise import Apprise, AppriseAsset\n\n        # A custom `AppriseAsset` that ensures Prefect Notifications\n        # appear correctly across multiple messaging platforms\n        prefect_app_data = AppriseAsset(\n            app_id=\"Prefect Notifications\",\n            app_desc=\"Prefect Notifications\",\n            app_url=\"https://prefect.io\",\n        )\n\n        self._apprise_client = Apprise(asset=prefect_app_data)\n        self._apprise_client.add(url.get_secret_value())\n\n    def block_initialization(self) -> None:\n        self._start_apprise_client(self.url)\n\n    @sync_compatible\n    @instrument_instance_method_call()\n    async def notify(self, body: str, subject: Optional[str] = None):\n        await self._apprise_client.async_notify(\n            body=body, title=subject, notify_type=self.notify_type\n        )\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.AppriseNotificationBlock","title":"AppriseNotificationBlock","text":"

    Bases: AbstractAppriseNotificationBlock, ABC

    A base class for sending notifications using Apprise, through webhook URLs.

    Source code in prefect/blocks/notifications.py
    class AppriseNotificationBlock(AbstractAppriseNotificationBlock, ABC):\n    \"\"\"\n    A base class for sending notifications using Apprise, through webhook URLs.\n    \"\"\"\n\n    _documentation_url = \"https://docs.prefect.io/ui/notifications/\"\n    url: SecretStr = Field(\n        default=...,\n        title=\"Webhook URL\",\n        description=\"Incoming webhook URL used to send notifications.\",\n        example=\"https://hooks.example.com/XXX\",\n    )\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.CustomWebhookNotificationBlock","title":"CustomWebhookNotificationBlock","text":"

    Bases: NotificationBlock

    Enables sending notifications via any custom webhook.

    All nested string param contains {{key}} will be substituted with value from context/secrets.

    Context values include: subject, body and name.

    Examples:

    Load a saved custom webhook and send a message:

    from prefect.blocks.notifications import CustomWebhookNotificationBlock\n\ncustom_webhook_block = CustomWebhookNotificationBlock.load(\"BLOCK_NAME\")\n\ncustom_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class CustomWebhookNotificationBlock(NotificationBlock):\n    \"\"\"\n    Enables sending notifications via any custom webhook.\n\n    All nested string param contains `{{key}}` will be substituted with value from context/secrets.\n\n    Context values include: `subject`, `body` and `name`.\n\n    Examples:\n        Load a saved custom webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import CustomWebhookNotificationBlock\n\n        custom_webhook_block = CustomWebhookNotificationBlock.load(\"BLOCK_NAME\")\n\n        custom_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Custom Webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c7247cb359eb6cf276734d4b1fbf00fb8930e89e-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.CustomWebhookNotificationBlock\"\n\n    name: str = Field(title=\"Name\", description=\"Name of the webhook.\")\n\n    url: str = Field(\n        title=\"Webhook URL\",\n        description=\"The webhook URL.\",\n        example=\"https://hooks.slack.com/XXX\",\n    )\n\n    method: Literal[\"GET\", \"POST\", \"PUT\", \"PATCH\", \"DELETE\"] = Field(\n        default=\"POST\", description=\"The webhook request method. Defaults to `POST`.\"\n    )\n\n    params: Optional[Dict[str, str]] = Field(\n        default=None, title=\"Query Params\", description=\"Custom query params.\"\n    )\n    json_data: Optional[dict] = Field(\n        default=None,\n        title=\"JSON Data\",\n        description=\"Send json data as payload.\",\n        example=(\n            '{\"text\": \"{{subject}}\\\\n{{body}}\", \"title\": \"{{name}}\", \"token\":'\n            ' \"{{tokenFromSecrets}}\"}'\n        ),\n    )\n    form_data: Optional[Dict[str, str]] = Field(\n        default=None,\n        title=\"Form Data\",\n        description=(\n            \"Send form data as payload. Should not be used together with _JSON Data_.\"\n        ),\n        example=(\n            '{\"text\": \"{{subject}}\\\\n{{body}}\", \"title\": \"{{name}}\", \"token\":'\n            ' \"{{tokenFromSecrets}}\"}'\n        ),\n    )\n\n    headers: Optional[Dict[str, str]] = Field(None, description=\"Custom headers.\")\n    cookies: Optional[Dict[str, str]] = Field(None, description=\"Custom cookies.\")\n\n    timeout: float = Field(\n        default=10, description=\"Request timeout in seconds. Defaults to 10.\"\n    )\n\n    secrets: SecretDict = Field(\n        default_factory=lambda: SecretDict(dict()),\n        title=\"Custom Secret Values\",\n        description=\"A dictionary of secret values to be substituted in other configs.\",\n        example='{\"tokenFromSecrets\":\"SomeSecretToken\"}',\n    )\n\n    def _build_request_args(self, body: str, subject: Optional[str]):\n        \"\"\"Build kwargs for httpx.AsyncClient.request\"\"\"\n        # prepare values\n        values = self.secrets.get_secret_value()\n        # use 'null' when subject is None\n        values.update(\n            {\n                \"subject\": \"null\" if subject is None else subject,\n                \"body\": body,\n                \"name\": self.name,\n            }\n        )\n        # do substution\n        return apply_values(\n            {\n                \"method\": self.method,\n                \"url\": self.url,\n                \"params\": self.params,\n                \"data\": self.form_data,\n                \"json\": self.json_data,\n                \"headers\": self.headers,\n                \"cookies\": self.cookies,\n                \"timeout\": self.timeout,\n            },\n            values,\n        )\n\n    def block_initialization(self) -> None:\n        # check form_data and json_data\n        if self.form_data is not None and self.json_data is not None:\n            raise ValueError(\"both `Form Data` and `JSON Data` provided\")\n        allowed_keys = {\"subject\", \"body\", \"name\"}.union(\n            self.secrets.get_secret_value().keys()\n        )\n        # test template to raise a error early\n        for name in [\"url\", \"params\", \"form_data\", \"json_data\", \"headers\", \"cookies\"]:\n            template = getattr(self, name)\n            if template is None:\n                continue\n            # check for placeholders not in predefined keys and secrets\n            placeholders = find_placeholders(template)\n            for placeholder in placeholders:\n                if placeholder.name not in allowed_keys:\n                    raise KeyError(f\"{name}/{placeholder}\")\n\n    @sync_compatible\n    @instrument_instance_method_call()\n    async def notify(self, body: str, subject: Optional[str] = None):\n        import httpx\n\n        # make request with httpx\n        client = httpx.AsyncClient(headers={\"user-agent\": \"Prefect Notifications\"})\n        resp = await client.request(**self._build_request_args(body, subject))\n        resp.raise_for_status()\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.DiscordWebhook","title":"DiscordWebhook","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via a provided Discord webhook. See Apprise notify_Discord docs # noqa

    Examples:

    Load a saved Discord webhook and send a message:

    from prefect.blocks.notifications import DiscordWebhook\n\ndiscord_webhook_block = DiscordWebhook.load(\"BLOCK_NAME\")\n\ndiscord_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class DiscordWebhook(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Discord webhook.\n    See [Apprise notify_Discord docs](https://github.com/caronc/apprise/wiki/Notify_Discord) # noqa\n\n    Examples:\n        Load a saved Discord webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import DiscordWebhook\n\n        discord_webhook_block = DiscordWebhook.load(\"BLOCK_NAME\")\n\n        discord_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via a provided Discord webhook.\"\n    _block_type_name = \"Discord Webhook\"\n    _block_type_slug = \"discord-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/9e94976c80ef925b66d24e5d14f0d47baa6b8f88-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.DiscordWebhook\"\n\n    webhook_id: SecretStr = Field(\n        default=...,\n        description=(\n            \"The first part of 2 tokens provided to you after creating a\"\n            \" incoming-webhook.\"\n        ),\n    )\n\n    webhook_token: SecretStr = Field(\n        default=...,\n        description=(\n            \"The second part of 2 tokens provided to you after creating a\"\n            \" incoming-webhook.\"\n        ),\n    )\n\n    botname: Optional[str] = Field(\n        title=\"Bot name\",\n        default=None,\n        description=(\n            \"Identify the name of the bot that should issue the message. If one isn't\"\n            \" specified then the default is to just use your account (associated with\"\n            \" the incoming-webhook).\"\n        ),\n    )\n\n    tts: bool = Field(\n        default=False,\n        description=\"Whether to enable Text-To-Speech.\",\n    )\n\n    include_image: bool = Field(\n        default=False,\n        description=(\n            \"Whether to include an image in-line with the message describing the\"\n            \" notification type.\"\n        ),\n    )\n\n    avatar: bool = Field(\n        default=False,\n        description=\"Whether to override the default discord avatar icon.\",\n    )\n\n    avatar_url: Optional[str] = Field(\n        title=\"Avatar URL\",\n        default=False,\n        description=(\n            \"Over-ride the default discord avatar icon URL. By default this is not set\"\n            \" and Apprise chooses the URL dynamically based on the type of message\"\n            \" (info, success, warning, or error).\"\n        ),\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyDiscord import NotifyDiscord\n\n        url = SecretStr(\n            NotifyDiscord(\n                webhook_id=self.webhook_id.get_secret_value(),\n                webhook_token=self.webhook_token.get_secret_value(),\n                botname=self.botname,\n                tts=self.tts,\n                include_image=self.include_image,\n                avatar=self.avatar,\n                avatar_url=self.avatar_url,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.MattermostWebhook","title":"MattermostWebhook","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via a provided Mattermost webhook. See Apprise notify_Mattermost docs # noqa

    Examples:

    Load a saved Mattermost webhook and send a message:

    from prefect.blocks.notifications import MattermostWebhook\n\nmattermost_webhook_block = MattermostWebhook.load(\"BLOCK_NAME\")\n\nmattermost_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class MattermostWebhook(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Mattermost webhook.\n    See [Apprise notify_Mattermost docs](https://github.com/caronc/apprise/wiki/Notify_Mattermost) # noqa\n\n\n    Examples:\n        Load a saved Mattermost webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import MattermostWebhook\n\n        mattermost_webhook_block = MattermostWebhook.load(\"BLOCK_NAME\")\n\n        mattermost_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via a provided Mattermost webhook.\"\n    _block_type_name = \"Mattermost Webhook\"\n    _block_type_slug = \"mattermost-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/1350a147130bf82cbc799a5f868d2c0116207736-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.MattermostWebhook\"\n\n    hostname: str = Field(\n        default=...,\n        description=\"The hostname of your Mattermost server.\",\n        example=\"Mattermost.example.com\",\n    )\n\n    token: SecretStr = Field(\n        default=...,\n        description=\"The token associated with your Mattermost webhook.\",\n    )\n\n    botname: Optional[str] = Field(\n        title=\"Bot name\",\n        default=None,\n        description=\"The name of the bot that will send the message.\",\n    )\n\n    channels: Optional[List[str]] = Field(\n        default=None,\n        description=\"The channel(s) you wish to notify.\",\n    )\n\n    include_image: bool = Field(\n        default=False,\n        description=\"Whether to include the Apprise status image in the message.\",\n    )\n\n    path: Optional[str] = Field(\n        default=None,\n        description=\"An optional sub-path specification to append to the hostname.\",\n    )\n\n    port: int = Field(\n        default=8065,\n        description=\"The port of your Mattermost server.\",\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyMattermost import NotifyMattermost\n\n        url = SecretStr(\n            NotifyMattermost(\n                token=self.token.get_secret_value(),\n                fullpath=self.path,\n                host=self.hostname,\n                botname=self.botname,\n                channels=self.channels,\n                include_image=self.include_image,\n                port=self.port,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.MicrosoftTeamsWebhook","title":"MicrosoftTeamsWebhook","text":"

    Bases: AppriseNotificationBlock

    Enables sending notifications via a provided Microsoft Teams webhook.

    Examples:

    Load a saved Teams webhook and send a message:

    from prefect.blocks.notifications import MicrosoftTeamsWebhook\nteams_webhook_block = MicrosoftTeamsWebhook.load(\"BLOCK_NAME\")\nteams_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class MicrosoftTeamsWebhook(AppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Microsoft Teams webhook.\n\n    Examples:\n        Load a saved Teams webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import MicrosoftTeamsWebhook\n        teams_webhook_block = MicrosoftTeamsWebhook.load(\"BLOCK_NAME\")\n        teams_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Microsoft Teams Webhook\"\n    _block_type_slug = \"ms-teams-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/817efe008a57f0a24f3587414714b563e5e23658-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.MicrosoftTeamsWebhook\"\n\n    url: SecretStr = Field(\n        ...,\n        title=\"Webhook URL\",\n        description=\"The Teams incoming webhook URL used to send notifications.\",\n        example=(\n            \"https://your-org.webhook.office.com/webhookb2/XXX/IncomingWebhook/YYY/ZZZ\"\n        ),\n    )\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.OpsgenieWebhook","title":"OpsgenieWebhook","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via a provided Opsgenie webhook. See Apprise notify_opsgenie docs for more info on formatting the URL.

    Examples:

    Load a saved Opsgenie webhook and send a message:

    from prefect.blocks.notifications import OpsgenieWebhook\nopsgenie_webhook_block = OpsgenieWebhook.load(\"BLOCK_NAME\")\nopsgenie_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class OpsgenieWebhook(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Opsgenie webhook.\n    See [Apprise notify_opsgenie docs](https://github.com/caronc/apprise/wiki/Notify_opsgenie)\n    for more info on formatting the URL.\n\n    Examples:\n        Load a saved Opsgenie webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import OpsgenieWebhook\n        opsgenie_webhook_block = OpsgenieWebhook.load(\"BLOCK_NAME\")\n        opsgenie_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via a provided Opsgenie webhook.\"\n\n    _block_type_name = \"Opsgenie Webhook\"\n    _block_type_slug = \"opsgenie-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/d8b5bc6244ae6cd83b62ec42f10d96e14d6e9113-280x280.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.OpsgenieWebhook\"\n\n    apikey: SecretStr = Field(\n        default=...,\n        title=\"API Key\",\n        description=\"The API Key associated with your Opsgenie account.\",\n    )\n\n    target_user: Optional[List] = Field(\n        default=None, description=\"The user(s) you wish to notify.\"\n    )\n\n    target_team: Optional[List] = Field(\n        default=None, description=\"The team(s) you wish to notify.\"\n    )\n\n    target_schedule: Optional[List] = Field(\n        default=None, description=\"The schedule(s) you wish to notify.\"\n    )\n\n    target_escalation: Optional[List] = Field(\n        default=None, description=\"The escalation(s) you wish to notify.\"\n    )\n\n    region_name: Literal[\"us\", \"eu\"] = Field(\n        default=\"us\", description=\"The 2-character region code.\"\n    )\n\n    batch: bool = Field(\n        default=False,\n        description=\"Notify all targets in batches (instead of individually).\",\n    )\n\n    tags: Optional[List] = Field(\n        default=None,\n        description=(\n            \"A comma-separated list of tags you can associate with your Opsgenie\"\n            \" message.\"\n        ),\n        example='[\"tag1\", \"tag2\"]',\n    )\n\n    priority: Optional[str] = Field(\n        default=3,\n        description=(\n            \"The priority to associate with the message. It is on a scale between 1\"\n            \" (LOW) and 5 (EMERGENCY).\"\n        ),\n    )\n\n    alias: Optional[str] = Field(\n        default=None, description=\"The alias to associate with the message.\"\n    )\n\n    entity: Optional[str] = Field(\n        default=None, description=\"The entity to associate with the message.\"\n    )\n\n    details: Optional[Dict[str, str]] = Field(\n        default=None,\n        description=\"Additional details composed of key/values pairs.\",\n        example='{\"key1\": \"value1\", \"key2\": \"value2\"}',\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyOpsgenie import NotifyOpsgenie\n\n        targets = []\n        if self.target_user:\n            [targets.append(f\"@{x}\") for x in self.target_user]\n        if self.target_team:\n            [targets.append(f\"#{x}\") for x in self.target_team]\n        if self.target_schedule:\n            [targets.append(f\"*{x}\") for x in self.target_schedule]\n        if self.target_escalation:\n            [targets.append(f\"^{x}\") for x in self.target_escalation]\n        url = SecretStr(\n            NotifyOpsgenie(\n                apikey=self.apikey.get_secret_value(),\n                targets=targets,\n                region_name=self.region_name,\n                details=self.details,\n                priority=self.priority,\n                alias=self.alias,\n                entity=self.entity,\n                batch=self.batch,\n                tags=self.tags,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.PagerDutyWebHook","title":"PagerDutyWebHook","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via a provided PagerDuty webhook. See Apprise notify_pagerduty docs for more info on formatting the URL.

    Examples:

    Load a saved PagerDuty webhook and send a message:

    from prefect.blocks.notifications import PagerDutyWebHook\npagerduty_webhook_block = PagerDutyWebHook.load(\"BLOCK_NAME\")\npagerduty_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class PagerDutyWebHook(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided PagerDuty webhook.\n    See [Apprise notify_pagerduty docs](https://github.com/caronc/apprise/wiki/Notify_pagerduty)\n    for more info on formatting the URL.\n\n    Examples:\n        Load a saved PagerDuty webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import PagerDutyWebHook\n        pagerduty_webhook_block = PagerDutyWebHook.load(\"BLOCK_NAME\")\n        pagerduty_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via a provided PagerDuty webhook.\"\n\n    _block_type_name = \"Pager Duty Webhook\"\n    _block_type_slug = \"pager-duty-webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/8dbf37d17089c1ce531708eac2e510801f7b3aee-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.PagerDutyWebHook\"\n\n    # The default cannot be prefect_default because NotifyPagerDuty's\n    # PAGERDUTY_SEVERITY_MAP only has these notify types defined as keys\n    notify_type: Literal[\"info\", \"success\", \"warning\", \"failure\"] = Field(\n        default=\"info\", description=\"The severity of the notification.\"\n    )\n\n    integration_key: SecretStr = Field(\n        default=...,\n        description=(\n            \"This can be found on the Events API V2 \"\n            \"integration's detail page, and is also referred to as a Routing Key. \"\n            \"This must be provided alongside `api_key`, but will error if provided \"\n            \"alongside `url`.\"\n        ),\n    )\n\n    api_key: SecretStr = Field(\n        default=...,\n        title=\"API Key\",\n        description=(\n            \"This can be found under Integrations. \"\n            \"This must be provided alongside `integration_key`, but will error if \"\n            \"provided alongside `url`.\"\n        ),\n    )\n\n    source: Optional[str] = Field(\n        default=\"Prefect\", description=\"The source string as part of the payload.\"\n    )\n\n    component: str = Field(\n        default=\"Notification\",\n        description=\"The component string as part of the payload.\",\n    )\n\n    group: Optional[str] = Field(\n        default=None, description=\"The group string as part of the payload.\"\n    )\n\n    class_id: Optional[str] = Field(\n        default=None,\n        title=\"Class ID\",\n        description=\"The class string as part of the payload.\",\n    )\n\n    region_name: Literal[\"us\", \"eu\"] = Field(\n        default=\"us\", description=\"The region name.\"\n    )\n\n    clickable_url: Optional[AnyHttpUrl] = Field(\n        default=None,\n        title=\"Clickable URL\",\n        description=\"A clickable URL to associate with the notice.\",\n    )\n\n    include_image: bool = Field(\n        default=True,\n        description=\"Associate the notification status via a represented icon.\",\n    )\n\n    custom_details: Optional[Dict[str, str]] = Field(\n        default=None,\n        description=\"Additional details to include as part of the payload.\",\n        example='{\"disk_space_left\": \"145GB\"}',\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyPagerDuty import NotifyPagerDuty\n\n        url = SecretStr(\n            NotifyPagerDuty(\n                apikey=self.api_key.get_secret_value(),\n                integrationkey=self.integration_key.get_secret_value(),\n                source=self.source,\n                component=self.component,\n                group=self.group,\n                class_id=self.class_id,\n                region_name=self.region_name,\n                click=self.clickable_url,\n                include_image=self.include_image,\n                details=self.custom_details,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.SendgridEmail","title":"SendgridEmail","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via any sendgrid account. See Apprise Notify_sendgrid docs

    Examples:

    Load a saved Sendgrid and send a email message: ```python from prefect.blocks.notifications import SendgridEmail

    sendgrid_block = SendgridEmail.load(\"BLOCK_NAME\")

    sendgrid_block.notify(\"Hello from Prefect!\")

    Source code in prefect/blocks/notifications.py
    class SendgridEmail(AbstractAppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via any sendgrid account.\n    See [Apprise Notify_sendgrid docs](https://github.com/caronc/apprise/wiki/Notify_Sendgrid)\n\n    Examples:\n        Load a saved Sendgrid and send a email message:\n        ```python\n        from prefect.blocks.notifications import SendgridEmail\n\n        sendgrid_block = SendgridEmail.load(\"BLOCK_NAME\")\n\n        sendgrid_block.notify(\"Hello from Prefect!\")\n    \"\"\"\n\n    _description = \"Enables sending notifications via Sendgrid email service.\"\n    _block_type_name = \"Sendgrid Email\"\n    _block_type_slug = \"sendgrid-email\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/82bc6ed16ca42a2252a5512c72233a253b8a58eb-250x250.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.SendgridEmail\"\n\n    api_key: SecretStr = Field(\n        default=...,\n        title=\"API Key\",\n        description=\"The API Key associated with your sendgrid account.\",\n    )\n\n    sender_email: str = Field(\n        title=\"Sender email id\",\n        description=\"The sender email id.\",\n        example=\"test-support@gmail.com\",\n    )\n\n    to_emails: List[str] = Field(\n        default=...,\n        title=\"Recipient emails\",\n        description=\"Email ids of all recipients.\",\n        example='\"recipient1@gmail.com\"',\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifySendGrid import NotifySendGrid\n\n        url = SecretStr(\n            NotifySendGrid(\n                apikey=self.api_key.get_secret_value(),\n                from_email=self.sender_email,\n                targets=self.to_emails,\n            ).url()\n        )\n\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.SlackWebhook","title":"SlackWebhook","text":"

    Bases: AppriseNotificationBlock

    Enables sending notifications via a provided Slack webhook.

    Examples:

    Load a saved Slack webhook and send a message:

    from prefect.blocks.notifications import SlackWebhook\n\nslack_webhook_block = SlackWebhook.load(\"BLOCK_NAME\")\nslack_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class SlackWebhook(AppriseNotificationBlock):\n    \"\"\"\n    Enables sending notifications via a provided Slack webhook.\n\n    Examples:\n        Load a saved Slack webhook and send a message:\n        ```python\n        from prefect.blocks.notifications import SlackWebhook\n\n        slack_webhook_block = SlackWebhook.load(\"BLOCK_NAME\")\n        slack_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Slack Webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c1965ecbf8704ee1ea20d77786de9a41ce1087d1-500x500.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.SlackWebhook\"\n\n    url: SecretStr = Field(\n        default=...,\n        title=\"Webhook URL\",\n        description=\"Slack incoming webhook URL used to send notifications.\",\n        example=\"https://hooks.slack.com/XXX\",\n    )\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.TwilioSMS","title":"TwilioSMS","text":"

    Bases: AbstractAppriseNotificationBlock

    Enables sending notifications via Twilio SMS. Find more on sending Twilio SMS messages in the docs.

    Examples:

    Load a saved TwilioSMS block and send a message:

    from prefect.blocks.notifications import TwilioSMS\ntwilio_webhook_block = TwilioSMS.load(\"BLOCK_NAME\")\ntwilio_webhook_block.notify(\"Hello from Prefect!\")\n

    Source code in prefect/blocks/notifications.py
    class TwilioSMS(AbstractAppriseNotificationBlock):\n    \"\"\"Enables sending notifications via Twilio SMS.\n    Find more on sending Twilio SMS messages in the [docs](https://www.twilio.com/docs/sms).\n\n    Examples:\n        Load a saved `TwilioSMS` block and send a message:\n        ```python\n        from prefect.blocks.notifications import TwilioSMS\n        twilio_webhook_block = TwilioSMS.load(\"BLOCK_NAME\")\n        twilio_webhook_block.notify(\"Hello from Prefect!\")\n        ```\n    \"\"\"\n\n    _description = \"Enables sending notifications via Twilio SMS.\"\n    _block_type_name = \"Twilio SMS\"\n    _block_type_slug = \"twilio-sms\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/8bd8777999f82112c09b9c8d57083ac75a4a0d65-250x250.png\"  # noqa\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/notifications/#prefect.blocks.notifications.TwilioSMS\"\n\n    account_sid: str = Field(\n        default=...,\n        description=(\n            \"The Twilio Account SID - it can be found on the homepage \"\n            \"of the Twilio console.\"\n        ),\n    )\n\n    auth_token: SecretStr = Field(\n        default=...,\n        description=(\n            \"The Twilio Authentication Token - \"\n            \"it can be found on the homepage of the Twilio console.\"\n        ),\n    )\n\n    from_phone_number: str = Field(\n        default=...,\n        description=\"The valid Twilio phone number to send the message from.\",\n        example=\"18001234567\",\n    )\n\n    to_phone_numbers: List[str] = Field(\n        default=...,\n        description=\"A list of valid Twilio phone number(s) to send the message to.\",\n        # not wrapped in brackets because of the way UI displays examples; in code should be [\"18004242424\"]\n        example=\"18004242424\",\n    )\n\n    def block_initialization(self) -> None:\n        from apprise.plugins.NotifyTwilio import NotifyTwilio\n\n        url = SecretStr(\n            NotifyTwilio(\n                account_sid=self.account_sid,\n                auth_token=self.auth_token.get_secret_value(),\n                source=self.from_phone_number,\n                targets=self.to_phone_numbers,\n            ).url()\n        )\n        self._start_apprise_client(url)\n
    ","tags":["Python API","blocks","notifications"]},{"location":"api-ref/prefect/blocks/system/","title":"system","text":"","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system","title":"prefect.blocks.system","text":"","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system.DateTime","title":"DateTime","text":"

    Bases: Block

    A block that represents a datetime

    Attributes:

    Name Type Description value DateTime

    An ISO 8601-compatible datetime value.

    Example

    Load a stored JSON value:

    from prefect.blocks.system import DateTime\n\ndata_time_block = DateTime.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/system.py
    class DateTime(Block):\n    \"\"\"\n    A block that represents a datetime\n\n    Attributes:\n        value: An ISO 8601-compatible datetime value.\n\n    Example:\n        Load a stored JSON value:\n        ```python\n        from prefect.blocks.system import DateTime\n\n        data_time_block = DateTime.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _block_type_name = \"Date Time\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/8b3da9a6621e92108b8e6a75b82e15374e170ff7-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/system/#prefect.blocks.system.DateTime\"\n\n    value: pendulum.DateTime = Field(\n        default=...,\n        description=\"An ISO 8601-compatible datetime value.\",\n    )\n
    ","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system.JSON","title":"JSON","text":"

    Bases: Block

    A block that represents JSON

    Attributes:

    Name Type Description value Any

    A JSON-compatible value.

    Example

    Load a stored JSON value:

    from prefect.blocks.system import JSON\n\njson_block = JSON.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/system.py
    class JSON(Block):\n    \"\"\"\n    A block that represents JSON\n\n    Attributes:\n        value: A JSON-compatible value.\n\n    Example:\n        Load a stored JSON value:\n        ```python\n        from prefect.blocks.system import JSON\n\n        json_block = JSON.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/4fcef2294b6eeb423b1332d1ece5156bf296ff96-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/system/#prefect.blocks.system.JSON\"\n\n    value: Any = Field(default=..., description=\"A JSON-compatible value.\")\n
    ","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system.Secret","title":"Secret","text":"

    Bases: Block

    A block that represents a secret value. The value stored in this block will be obfuscated when this block is logged or shown in the UI.

    Attributes:

    Name Type Description value SecretStr

    A string value that should be kept secret.

    Example
    from prefect.blocks.system import Secret\n\nsecret_block = Secret.load(\"BLOCK_NAME\")\n\n# Access the stored secret\nsecret_block.get()\n
    Source code in prefect/blocks/system.py
    class Secret(Block):\n    \"\"\"\n    A block that represents a secret value. The value stored in this block will be obfuscated when\n    this block is logged or shown in the UI.\n\n    Attributes:\n        value: A string value that should be kept secret.\n\n    Example:\n        ```python\n        from prefect.blocks.system import Secret\n\n        secret_block = Secret.load(\"BLOCK_NAME\")\n\n        # Access the stored secret\n        secret_block.get()\n        ```\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c6f20e556dd16effda9df16551feecfb5822092b-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/system/#prefect.blocks.system.Secret\"\n\n    value: SecretStr = Field(\n        default=..., description=\"A string value that should be kept secret.\"\n    )\n\n    def get(self):\n        return self.value.get_secret_value()\n
    ","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/system/#prefect.blocks.system.String","title":"String","text":"

    Bases: Block

    A block that represents a string

    Attributes:

    Name Type Description value str

    A string value.

    Example

    Load a stored string value:

    from prefect.blocks.system import String\n\nstring_block = String.load(\"BLOCK_NAME\")\n

    Source code in prefect/blocks/system.py
    class String(Block):\n    \"\"\"\n    A block that represents a string\n\n    Attributes:\n        value: A string value.\n\n    Example:\n        Load a stored string value:\n        ```python\n        from prefect.blocks.system import String\n\n        string_block = String.load(\"BLOCK_NAME\")\n        ```\n    \"\"\"\n\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c262ea2c80a2c043564e8763f3370c3db5a6b3e6-48x48.png\"\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/system/#prefect.blocks.system.String\"\n\n    value: str = Field(default=..., description=\"A string value.\")\n
    ","tags":["Python API","blocks","secret","config","json"]},{"location":"api-ref/prefect/blocks/webhook/","title":"webhook","text":"","tags":["Python API","blocks","webhook"]},{"location":"api-ref/prefect/blocks/webhook/#prefect.blocks.webhook","title":"prefect.blocks.webhook","text":"","tags":["Python API","blocks","webhook"]},{"location":"api-ref/prefect/blocks/webhook/#prefect.blocks.webhook.Webhook","title":"Webhook","text":"

    Bases: Block

    Block that enables calling webhooks.

    Source code in prefect/blocks/webhook.py
    class Webhook(Block):\n    \"\"\"\n    Block that enables calling webhooks.\n    \"\"\"\n\n    _block_type_name = \"Webhook\"\n    _logo_url = \"https://cdn.sanity.io/images/3ugk85nk/production/c7247cb359eb6cf276734d4b1fbf00fb8930e89e-250x250.png\"  # type: ignore\n    _documentation_url = \"https://docs.prefect.io/api-ref/prefect/blocks/webhook/#prefect.blocks.webhook.Webhook\"\n\n    method: Literal[\"GET\", \"POST\", \"PUT\", \"PATCH\", \"DELETE\"] = Field(\n        default=\"POST\", description=\"The webhook request method. Defaults to `POST`.\"\n    )\n\n    url: SecretStr = Field(\n        default=...,\n        title=\"Webhook URL\",\n        description=\"The webhook URL.\",\n        example=\"https://hooks.slack.com/XXX\",\n    )\n\n    headers: SecretDict = Field(\n        default_factory=lambda: SecretDict(dict()),\n        title=\"Webhook Headers\",\n        description=\"A dictionary of headers to send with the webhook request.\",\n    )\n\n    def block_initialization(self):\n        self._client = AsyncClient(transport=_http_transport)\n\n    async def call(self, payload: Optional[dict] = None) -> Response:\n        \"\"\"\n        Call the webhook.\n\n        Args:\n            payload: an optional payload to send when calling the webhook.\n        \"\"\"\n        async with self._client:\n            return await self._client.request(\n                method=self.method,\n                url=self.url.get_secret_value(),\n                headers=self.headers.get_secret_value(),\n                json=payload,\n            )\n
    ","tags":["Python API","blocks","webhook"]},{"location":"api-ref/prefect/blocks/webhook/#prefect.blocks.webhook.Webhook.call","title":"call async","text":"

    Call the webhook.

    Parameters:

    Name Type Description Default payload Optional[dict]

    an optional payload to send when calling the webhook.

    None Source code in prefect/blocks/webhook.py
    async def call(self, payload: Optional[dict] = None) -> Response:\n    \"\"\"\n    Call the webhook.\n\n    Args:\n        payload: an optional payload to send when calling the webhook.\n    \"\"\"\n    async with self._client:\n        return await self._client.request(\n            method=self.method,\n            url=self.url.get_secret_value(),\n            headers=self.headers.get_secret_value(),\n            json=payload,\n        )\n
    ","tags":["Python API","blocks","webhook"]},{"location":"api-ref/prefect/cli/agent/","title":"agent","text":"","tags":["Python API","agents","CLI"]},{"location":"api-ref/prefect/cli/agent/#prefect.cli.agent","title":"prefect.cli.agent","text":"

    Command line interface for working with agent services

    ","tags":["Python API","agents","CLI"]},{"location":"api-ref/prefect/cli/agent/#prefect.cli.agent.start","title":"start async","text":"

    Start an agent process to poll one or more work queues for flow runs.

    Source code in prefect/cli/agent.py
    @agent_app.command()\nasync def start(\n    # deprecated main argument\n    work_queue: str = typer.Argument(\n        None,\n        show_default=False,\n        help=\"DEPRECATED: A work queue name or ID\",\n    ),\n    work_queues: List[str] = typer.Option(\n        None,\n        \"-q\",\n        \"--work-queue\",\n        help=\"One or more work queue names for the agent to pull from.\",\n    ),\n    work_queue_prefix: List[str] = typer.Option(\n        None,\n        \"-m\",\n        \"--match\",\n        help=(\n            \"Dynamically matches work queue names with the specified prefix for the\"\n            \" agent to pull from,for example `dev-` will match all work queues with a\"\n            \" name that starts with `dev-`\"\n        ),\n    ),\n    work_pool_name: str = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"A work pool name for the agent to pull from.\",\n    ),\n    hide_welcome: bool = typer.Option(False, \"--hide-welcome\"),\n    api: str = SettingsOption(PREFECT_API_URL),\n    run_once: bool = typer.Option(\n        False, help=\"Run the agent loop once, instead of forever.\"\n    ),\n    prefetch_seconds: int = SettingsOption(PREFECT_AGENT_PREFETCH_SECONDS),\n    # deprecated tags\n    tags: List[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=(\n            \"DEPRECATED: One or more optional tags that will be used to create a work\"\n            \" queue. This option will be removed on 2023-02-23.\"\n        ),\n    ),\n    limit: int = typer.Option(\n        None,\n        \"-l\",\n        \"--limit\",\n        help=\"Maximum number of flow runs to start simultaneously.\",\n    ),\n):\n    \"\"\"\n    Start an agent process to poll one or more work queues for flow runs.\n    \"\"\"\n    work_queues = work_queues or []\n\n    if work_queue is not None:\n        # try to treat the work_queue as a UUID\n        try:\n            async with get_client() as client:\n                q = await client.read_work_queue(UUID(work_queue))\n                work_queue = q.name\n        # otherwise treat it as a string name\n        except (TypeError, ValueError):\n            pass\n        work_queues.append(work_queue)\n        app.console.print(\n            (\n                \"Agents now support multiple work queues. Instead of passing a single\"\n                \" argument, provide work queue names with the `-q` or `--work-queue`\"\n                f\" flag: `prefect agent start -q {work_queue}`\\n\"\n            ),\n            style=\"blue\",\n        )\n\n    if not work_queues and not tags and not work_queue_prefix and not work_pool_name:\n        exit_with_error(\"No work queues provided!\", style=\"red\")\n    elif bool(work_queues) + bool(tags) + bool(work_queue_prefix) > 1:\n        exit_with_error(\n            \"Only one of `work_queues`, `match`, or `tags` can be provided.\",\n            style=\"red\",\n        )\n    if work_pool_name and tags:\n        exit_with_error(\n            \"`tag` and `pool` options cannot be used together.\", style=\"red\"\n        )\n\n    if tags:\n        work_queue_name = f\"Agent queue {'-'.join(sorted(tags))}\"\n        app.console.print(\n            (\n                \"`tags` are deprecated. For backwards-compatibility with old versions\"\n                \" of Prefect, this agent will create a work queue named\"\n                f\" `{work_queue_name}` that uses legacy tag-based matching. This option\"\n                \" will be removed on 2023-02-23.\"\n            ),\n            style=\"red\",\n        )\n\n        async with get_client() as client:\n            try:\n                work_queue = await client.read_work_queue_by_name(work_queue_name)\n                if work_queue.filter is None:\n                    # ensure the work queue has legacy (deprecated) tag-based behavior\n                    await client.update_work_queue(filter=dict(tags=tags))\n            except ObjectNotFound:\n                # if the work queue doesn't already exist, we create it with tags\n                # to enable legacy (deprecated) tag-matching behavior\n                await client.create_work_queue(name=work_queue_name, tags=tags)\n\n        work_queues = [work_queue_name]\n\n    if not hide_welcome:\n        if api:\n            app.console.print(\n                f\"Starting v{prefect.__version__} agent connected to {api}...\"\n            )\n        else:\n            app.console.print(\n                f\"Starting v{prefect.__version__} agent with ephemeral API...\"\n            )\n\n    agent_process_id = os.getpid()\n    setup_signal_handlers_agent(\n        agent_process_id, \"the Prefect agent\", app.console.print\n    )\n\n    async with PrefectAgent(\n        work_queues=work_queues,\n        work_queue_prefix=work_queue_prefix,\n        work_pool_name=work_pool_name,\n        prefetch_seconds=prefetch_seconds,\n        limit=limit,\n    ) as agent:\n        if not hide_welcome:\n            app.console.print(ascii_name)\n            if work_pool_name:\n                app.console.print(\n                    \"Agent started! Looking for work from \"\n                    f\"work pool '{work_pool_name}'...\"\n                )\n            elif work_queue_prefix:\n                app.console.print(\n                    \"Agent started! Looking for work from \"\n                    f\"queue(s) that start with the prefix: {work_queue_prefix}...\"\n                )\n            else:\n                app.console.print(\n                    \"Agent started! Looking for work from \"\n                    f\"queue(s): {', '.join(work_queues)}...\"\n                )\n\n        async with anyio.create_task_group() as tg:\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    agent.get_and_submit_flow_runs,\n                    PREFECT_AGENT_QUERY_INTERVAL.value(),\n                    printer=app.console.print,\n                    run_once=run_once,\n                    jitter_range=0.3,\n                    backoff=4,  # Up to ~1 minute interval during backoff\n                )\n            )\n\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    agent.check_for_cancelled_flow_runs,\n                    PREFECT_AGENT_QUERY_INTERVAL.value() * 2,\n                    printer=app.console.print,\n                    run_once=run_once,\n                    jitter_range=0.3,\n                    backoff=4,\n                )\n            )\n\n    app.console.print(\"Agent stopped!\")\n
    ","tags":["Python API","agents","CLI"]},{"location":"api-ref/prefect/cli/artifact/","title":"artifact","text":"","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/artifact/#prefect.cli.artifact","title":"prefect.cli.artifact","text":"","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/artifact/#prefect.cli.artifact.delete","title":"delete async","text":"

    Delete an artifact.

    Parameters:

    Name Type Description Default key Optional[str]

    the key of the artifact to delete

    Argument(None, help='The key of the artifact to delete.')

    Examples:

    $ prefect artifact delete \"my-artifact\"

    Source code in prefect/cli/artifact.py
    @artifact_app.command(\"delete\")\nasync def delete(\n    key: Optional[str] = typer.Argument(\n        None, help=\"The key of the artifact to delete.\"\n    ),\n    artifact_id: Optional[str] = typer.Option(\n        None, \"--id\", help=\"The ID of the artifact to delete.\"\n    ),\n):\n    \"\"\"\n    Delete an artifact.\n\n    Arguments:\n        key: the key of the artifact to delete\n\n    Examples:\n        $ prefect artifact delete \"my-artifact\"\n    \"\"\"\n    if key and artifact_id:\n        exit_with_error(\"Please provide either a key or an artifact_id but not both.\")\n\n    async with get_client() as client:\n        if artifact_id is not None:\n            try:\n                confirm_delete = typer.confirm(\n                    (\n                        \"Are you sure you want to delete artifact with id\"\n                        f\" {artifact_id!r}?\"\n                    ),\n                    default=False,\n                )\n                if not confirm_delete:\n                    exit_with_error(\"Deletion aborted.\")\n\n                await client.delete_artifact(artifact_id)\n                exit_with_success(f\"Deleted artifact with id {artifact_id!r}.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Artifact with id {artifact_id!r} not found!\")\n\n        elif key is not None:\n            artifacts = await client.read_artifacts(\n                artifact_filter=ArtifactFilter(key=ArtifactFilterKey(any_=[key])),\n            )\n            if not artifacts:\n                exit_with_error(\n                    f\"Artifact with key {key!r} not found. You can also specify an\"\n                    \" artifact id with the --id flag.\"\n                )\n\n            confirm_delete = typer.confirm(\n                (\n                    f\"Are you sure you want to delete {len(artifacts)} artifact(s) with\"\n                    f\" key {key!r}?\"\n                ),\n                default=False,\n            )\n            if not confirm_delete:\n                exit_with_error(\"Deletion aborted.\")\n\n            for a in artifacts:\n                await client.delete_artifact(a.id)\n\n            exit_with_success(f\"Deleted {len(artifacts)} artifact(s) with key {key!r}.\")\n\n        else:\n            exit_with_error(\"Please provide a key or an artifact_id.\")\n
    ","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/artifact/#prefect.cli.artifact.inspect","title":"inspect async","text":"
    View details about an artifact.\n\nArguments:\n    key: the key of the artifact to inspect\n\nExamples:\n    $ prefect artifact inspect \"my-artifact\"\n   [\n    {\n        'id': 'ba1d67be-0bd7-452e-8110-247fe5e6d8cc',\n        'created': '2023-03-21T21:40:09.895910+00:00',\n        'updated': '2023-03-21T21:40:09.895910+00:00',\n        'key': 'my-artifact',\n        'type': 'markdown',\n        'description': None,\n        'data': 'my markdown',\n        'metadata_': None,\n        'flow_run_id': '8dc54b6f-6e24-4586-a05c-e98c6490cb98',\n        'task_run_id': None\n    },\n    {\n        'id': '57f235b5-2576-45a5-bd93-c829c2900966',\n        'created': '2023-03-27T23:16:15.536434+00:00',\n        'updated': '2023-03-27T23:16:15.536434+00:00',\n        'key': 'my-artifact',\n        'type': 'markdown',\n        'description': 'my-artifact-description',\n        'data': 'my markdown',\n        'metadata_': None,\n        'flow_run_id': 'ffa91051-f249-48c1-ae0f-4754fcb7eb29',\n        'task_run_id': None\n    }\n

    ]

    Source code in prefect/cli/artifact.py
    @artifact_app.command(\"inspect\")\nasync def inspect(\n    key: str,\n    limit: int = typer.Option(\n        10,\n        \"--limit\",\n        help=\"The maximum number of artifacts to return.\",\n    ),\n):\n    \"\"\"\n        View details about an artifact.\n\n        Arguments:\n            key: the key of the artifact to inspect\n\n        Examples:\n            $ prefect artifact inspect \"my-artifact\"\n           [\n            {\n                'id': 'ba1d67be-0bd7-452e-8110-247fe5e6d8cc',\n                'created': '2023-03-21T21:40:09.895910+00:00',\n                'updated': '2023-03-21T21:40:09.895910+00:00',\n                'key': 'my-artifact',\n                'type': 'markdown',\n                'description': None,\n                'data': 'my markdown',\n                'metadata_': None,\n                'flow_run_id': '8dc54b6f-6e24-4586-a05c-e98c6490cb98',\n                'task_run_id': None\n            },\n            {\n                'id': '57f235b5-2576-45a5-bd93-c829c2900966',\n                'created': '2023-03-27T23:16:15.536434+00:00',\n                'updated': '2023-03-27T23:16:15.536434+00:00',\n                'key': 'my-artifact',\n                'type': 'markdown',\n                'description': 'my-artifact-description',\n                'data': 'my markdown',\n                'metadata_': None,\n                'flow_run_id': 'ffa91051-f249-48c1-ae0f-4754fcb7eb29',\n                'task_run_id': None\n            }\n    ]\n    \"\"\"\n\n    async with get_client() as client:\n        artifacts = await client.read_artifacts(\n            limit=limit,\n            sort=ArtifactSort.UPDATED_DESC,\n            artifact_filter=ArtifactFilter(key=ArtifactFilterKey(any_=[key])),\n        )\n        if not artifacts:\n            exit_with_error(f\"Artifact {key!r} not found.\")\n\n        artifacts = [a.dict(json_compatible=True) for a in artifacts]\n\n        app.console.print(Pretty(artifacts))\n
    ","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/artifact/#prefect.cli.artifact.list_artifacts","title":"list_artifacts async","text":"

    List artifacts.

    Source code in prefect/cli/artifact.py
    @artifact_app.command(\"ls\")\nasync def list_artifacts(\n    limit: int = typer.Option(\n        100,\n        \"--limit\",\n        help=\"The maximum number of artifacts to return.\",\n    ),\n    all: bool = typer.Option(\n        False,\n        \"--all\",\n        \"-a\",\n        help=\"Whether or not to only return the latest version of each artifact.\",\n    ),\n):\n    \"\"\"\n    List artifacts.\n    \"\"\"\n    table = Table(\n        title=\"Artifacts\",\n        caption=\"List Artifacts using `prefect artifact ls`\",\n        show_header=True,\n    )\n\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Key\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Type\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Updated\", style=\"blue\", no_wrap=True)\n\n    async with get_client() as client:\n        if all:\n            artifacts = await client.read_artifacts(\n                sort=ArtifactSort.KEY_ASC,\n                limit=limit,\n            )\n\n            for artifact in sorted(artifacts, key=lambda x: f\"{x.key}\"):\n                table.add_row(\n                    str(artifact.id),\n                    artifact.key,\n                    artifact.type,\n                    pendulum.instance(artifact.updated).diff_for_humans(),\n                )\n\n        else:\n            artifacts = await client.read_latest_artifacts(\n                sort=ArtifactCollectionSort.KEY_ASC,\n                limit=limit,\n            )\n\n            for artifact in sorted(artifacts, key=lambda x: f\"{x.key}\"):\n                table.add_row(\n                    str(artifact.latest_id),\n                    artifact.key,\n                    artifact.type,\n                    pendulum.instance(artifact.updated).diff_for_humans(),\n                )\n\n        app.console.print(table)\n
    ","tags":["Python API","artifacts","CLI"]},{"location":"api-ref/prefect/cli/block/","title":"block","text":"","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block","title":"prefect.cli.block","text":"

    Command line interface for working with blocks.

    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.block_create","title":"block_create async","text":"

    Generate a link to the Prefect UI to create a block.

    Source code in prefect/cli/block.py
    @blocks_app.command(\"create\")\nasync def block_create(\n    block_type_slug: str = typer.Argument(\n        ...,\n        help=\"A block type slug. View available types with: prefect block type ls\",\n        show_default=False,\n    ),\n):\n    \"\"\"\n    Generate a link to the Prefect UI to create a block.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            block_type = await client.read_block_type_by_slug(block_type_slug)\n        except ObjectNotFound:\n            app.console.print(f\"[red]Block type {block_type_slug!r} not found![/red]\")\n            block_types = await client.read_block_types()\n            slugs = {block_type.slug for block_type in block_types}\n            app.console.print(f\"Available block types: {', '.join(slugs)}\")\n            raise typer.Exit(1)\n\n        if not PREFECT_UI_URL:\n            exit_with_error(\n                \"Prefect must be configured to use a hosted Prefect server or \"\n                \"Prefect Cloud to display the Prefect UI\"\n            )\n\n        block_link = f\"{PREFECT_UI_URL.value()}/blocks/catalog/{block_type.slug}/create\"\n        app.console.print(\n            f\"Create a {block_type_slug} block: {block_link}\",\n        )\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.block_delete","title":"block_delete async","text":"

    Delete a configured block.

    Source code in prefect/cli/block.py
    @blocks_app.command(\"delete\")\nasync def block_delete(\n    slug: Optional[str] = typer.Argument(\n        None, help=\"A block slug. Formatted as '<BLOCK_TYPE_SLUG>/<BLOCK_NAME>'\"\n    ),\n    block_id: Optional[str] = typer.Option(None, \"--id\", help=\"A block id.\"),\n):\n    \"\"\"\n    Delete a configured block.\n    \"\"\"\n    async with get_client() as client:\n        if slug is None and block_id is not None:\n            try:\n                await client.delete_block_document(block_id)\n                exit_with_success(f\"Deleted Block '{block_id}'.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {block_id!r} not found!\")\n        elif slug is not None:\n            block_type_slug, block_document_name = slug.split(\"/\")\n            try:\n                block_document = await client.read_block_document_by_name(\n                    block_document_name, block_type_slug, include_secrets=False\n                )\n                await client.delete_block_document(block_document.id)\n                exit_with_success(f\"Deleted Block '{slug}'.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Block {slug!r} not found!\")\n        else:\n            exit_with_error(\"Must provide a block slug or id\")\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.block_inspect","title":"block_inspect async","text":"

    Displays details about a configured block.

    Source code in prefect/cli/block.py
    @blocks_app.command(\"inspect\")\nasync def block_inspect(\n    slug: Optional[str] = typer.Argument(\n        None, help=\"A Block slug: <BLOCK_TYPE_SLUG>/<BLOCK_NAME>\"\n    ),\n    block_id: Optional[str] = typer.Option(\n        None, \"--id\", help=\"A Block id to search for if no slug is given\"\n    ),\n):\n    \"\"\"\n    Displays details about a configured block.\n    \"\"\"\n    async with get_client() as client:\n        if slug is None and block_id is not None:\n            try:\n                block_document = await client.read_block_document(\n                    block_id, include_secrets=False\n                )\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {block_id!r} not found!\")\n        elif slug is not None:\n            block_type_slug, block_document_name = slug.split(\"/\")\n            try:\n                block_document = await client.read_block_document_by_name(\n                    block_document_name, block_type_slug, include_secrets=False\n                )\n            except ObjectNotFound:\n                exit_with_error(f\"Block {slug!r} not found!\")\n        else:\n            exit_with_error(\"Must provide a block slug or id\")\n        app.console.print(display_block(block_document))\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.block_ls","title":"block_ls async","text":"

    View all configured blocks.

    Source code in prefect/cli/block.py
    @blocks_app.command(\"ls\")\nasync def block_ls():\n    \"\"\"\n    View all configured blocks.\n    \"\"\"\n    async with get_client() as client:\n        blocks = await client.read_block_documents()\n\n    table = Table(\n        title=\"Blocks\", caption=\"List Block Types using `prefect block type ls`\"\n    )\n    table.add_column(\"ID\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Type\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Name\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Slug\", style=\"blue\", no_wrap=True)\n\n    for block in sorted(blocks, key=lambda x: f\"{x.block_type.slug}/{x.name}\"):\n        table.add_row(\n            str(block.id),\n            block.block_type.name,\n            str(block.name),\n            f\"{block.block_type.slug}/{block.name}\",\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.blocktype_delete","title":"blocktype_delete async","text":"

    Delete an unprotected Block Type.

    Source code in prefect/cli/block.py
    @blocktypes_app.command(\"delete\")\nasync def blocktype_delete(\n    slug: str = typer.Argument(..., help=\"A Block type slug\"),\n):\n    \"\"\"\n    Delete an unprotected Block Type.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            block_type = await client.read_block_type_by_slug(slug)\n            await client.delete_block_type(block_type.id)\n            exit_with_success(f\"Deleted Block Type '{slug}'.\")\n        except ObjectNotFound:\n            exit_with_error(f\"Block Type {slug!r} not found!\")\n        except ProtectedBlockError:\n            exit_with_error(f\"Block Type {slug!r} is a protected block!\")\n        except PrefectHTTPStatusError:\n            exit_with_error(f\"Cannot delete Block Type {slug!r}!\")\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.blocktype_inspect","title":"blocktype_inspect async","text":"

    Display details about a block type.

    Source code in prefect/cli/block.py
    @blocktypes_app.command(\"inspect\")\nasync def blocktype_inspect(\n    slug: str = typer.Argument(..., help=\"A block type slug\"),\n):\n    \"\"\"\n    Display details about a block type.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            block_type = await client.read_block_type_by_slug(slug)\n        except ObjectNotFound:\n            exit_with_error(f\"Block type {slug!r} not found!\")\n\n        app.console.print(display_block_type(block_type))\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.list_types","title":"list_types async","text":"

    List all block types.

    Source code in prefect/cli/block.py
    @blocktypes_app.command(\"ls\")\nasync def list_types():\n    \"\"\"\n    List all block types.\n    \"\"\"\n    async with get_client() as client:\n        block_types = await client.read_block_types()\n\n    table = Table(\n        title=\"Block Types\",\n        show_lines=True,\n    )\n\n    table.add_column(\"Block Type Slug\", style=\"italic cyan\", no_wrap=True)\n    table.add_column(\"Description\", style=\"blue\", no_wrap=False, justify=\"left\")\n    table.add_column(\n        \"Generate creation link\", style=\"italic cyan\", no_wrap=False, justify=\"left\"\n    )\n\n    for blocktype in sorted(block_types, key=lambda x: x.name):\n        table.add_row(\n            str(blocktype.slug),\n            (\n                str(blocktype.description.splitlines()[0].partition(\".\")[0])\n                if blocktype.description is not None\n                else \"\"\n            ),\n            f\"prefect block create {blocktype.slug}\",\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/block/#prefect.cli.block.register","title":"register async","text":"

    Register blocks types within a module or file.

    This makes the blocks available for configuration via the UI. If a block type has already been registered, its registration will be updated to match the block's current definition.

    \b Examples: \b Register block types in a Python module: $ prefect block register -m prefect_aws.credentials \b Register block types in a .py file: $ prefect block register -f my_blocks.py

    Source code in prefect/cli/block.py
    @blocks_app.command()\nasync def register(\n    module_name: Optional[str] = typer.Option(\n        None,\n        \"--module\",\n        \"-m\",\n        help=\"Python module containing block types to be registered\",\n    ),\n    file_path: Optional[Path] = typer.Option(\n        None,\n        \"--file\",\n        \"-f\",\n        help=\"Path to .py file containing block types to be registered\",\n    ),\n):\n    \"\"\"\n    Register blocks types within a module or file.\n\n    This makes the blocks available for configuration via the UI.\n    If a block type has already been registered, its registration will be updated to\n    match the block's current definition.\n\n    \\b\n    Examples:\n        \\b\n        Register block types in a Python module:\n        $ prefect block register -m prefect_aws.credentials\n        \\b\n        Register block types in a .py file:\n        $ prefect block register -f my_blocks.py\n    \"\"\"\n    # Handles if both options are specified or if neither are specified\n    if not (bool(file_path) ^ bool(module_name)):\n        exit_with_error(\n            \"Please specify either a module or a file containing blocks to be\"\n            \" registered, but not both.\"\n        )\n\n    if module_name:\n        try:\n            imported_module = import_module(name=module_name)\n        except ModuleNotFoundError:\n            exit_with_error(\n                f\"Unable to load {module_name}. Please make sure the module is \"\n                \"installed in your current environment.\"\n            )\n\n    if file_path:\n        if file_path.suffix != \".py\":\n            exit_with_error(\n                f\"{file_path} is not a .py file. Please specify a \"\n                \".py that contains blocks to be registered.\"\n            )\n        try:\n            imported_module = await run_sync_in_worker_thread(\n                load_script_as_module, str(file_path)\n            )\n        except ScriptError as exc:\n            app.console.print(exc)\n            app.console.print(exception_traceback(exc.user_exc))\n            exit_with_error(\n                f\"Unable to load file at {file_path}. Please make sure the file path \"\n                \"is correct and the file contains valid Python.\"\n            )\n\n    registered_blocks = await _register_blocks_in_module(imported_module)\n    number_of_registered_blocks = len(registered_blocks)\n    block_text = \"block\" if 0 < number_of_registered_blocks < 2 else \"blocks\"\n    app.console.print(\n        f\"[green]Successfully registered {number_of_registered_blocks} {block_text}\\n\"\n    )\n    app.console.print(_build_registered_blocks_table(registered_blocks))\n    msg = (\n        \"\\n To configure the newly registered blocks, \"\n        \"go to the Blocks page in the Prefect UI.\\n\"\n    )\n\n    ui_url = PREFECT_UI_URL.value()\n    if ui_url is not None:\n        block_catalog_url = f\"{ui_url}/blocks/catalog\"\n        msg = f\"{msg.rstrip().rstrip('.')}: {block_catalog_url}\\n\"\n\n    app.console.print(msg)\n
    ","tags":["Python API","blocks","CLI"]},{"location":"api-ref/prefect/cli/cloud-webhook/","title":"Cloud webhook","text":"","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook","title":"prefect.cli.cloud.webhook","text":"

    Command line interface for working with webhooks

    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.create","title":"create async","text":"

    Create a new Cloud webhook

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def create(\n    webhook_name: str,\n    description: str = typer.Option(\n        \"\", \"--description\", \"-d\", help=\"Description of the webhook\"\n    ),\n    template: str = typer.Option(\n        None, \"--template\", \"-t\", help=\"Jinja2 template expression\"\n    ),\n):\n    \"\"\"\n    Create a new Cloud webhook\n    \"\"\"\n    if not template:\n        exit_with_error(\n            \"Please provide a Jinja2 template expression in the --template flag \\nwhich\"\n            ' should define (at minimum) the following attributes: \\n{ \"event\":'\n            ' \"your.event.name\", \"resource\": { \"prefect.resource.id\":'\n            ' \"your.resource.id\" } }'\n            \" \\nhttps://docs.prefect.io/latest/cloud/webhooks/#webhook-templates\"\n        )\n\n    confirm_logged_in()\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        response = await client.request(\n            \"POST\",\n            \"/webhooks/\",\n            json={\n                \"name\": webhook_name,\n                \"description\": description,\n                \"template\": template,\n            },\n        )\n        app.console.print(f'Successfully created webhook {response[\"name\"]}')\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.delete","title":"delete async","text":"

    Delete an existing Cloud webhook

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def delete(webhook_id: UUID):\n    \"\"\"\n    Delete an existing Cloud webhook\n    \"\"\"\n    confirm_logged_in()\n\n    confirm_delete = typer.confirm(\n        \"Are you sure you want to delete it? This cannot be undone.\"\n    )\n\n    if not confirm_delete:\n        return\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        await client.request(\"DELETE\", f\"/webhooks/{webhook_id}\")\n        app.console.print(f\"Successfully deleted webhook {webhook_id}\")\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.get","title":"get async","text":"

    Retrieve a webhook by ID.

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def get(webhook_id: UUID):\n    \"\"\"\n    Retrieve a webhook by ID.\n    \"\"\"\n    confirm_logged_in()\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        webhook = await client.request(\"GET\", f\"/webhooks/{webhook_id}\")\n        display_table = _render_webhooks_into_table([webhook])\n        app.console.print(display_table)\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.ls","title":"ls async","text":"

    Fetch and list all webhooks in your workspace

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def ls():\n    \"\"\"\n    Fetch and list all webhooks in your workspace\n    \"\"\"\n    confirm_logged_in()\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        retrieved_webhooks = await client.request(\"POST\", \"/webhooks/filter\")\n        display_table = _render_webhooks_into_table(retrieved_webhooks)\n        app.console.print(display_table)\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.rotate","title":"rotate async","text":"

    Rotate url for an existing Cloud webhook, in case it has been compromised

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def rotate(webhook_id: UUID):\n    \"\"\"\n    Rotate url for an existing Cloud webhook, in case it has been compromised\n    \"\"\"\n    confirm_logged_in()\n\n    confirm_rotate = typer.confirm(\n        \"Are you sure you want to rotate? This will invalidate the old URL.\"\n    )\n\n    if not confirm_rotate:\n        return\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        response = await client.request(\"POST\", f\"/webhooks/{webhook_id}/rotate\")\n        app.console.print(f'Successfully rotated webhook URL to {response[\"slug\"]}')\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.toggle","title":"toggle async","text":"

    Toggle the enabled status of an existing Cloud webhook

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def toggle(\n    webhook_id: UUID,\n):\n    \"\"\"\n    Toggle the enabled status of an existing Cloud webhook\n    \"\"\"\n    confirm_logged_in()\n\n    status_lookup = {True: \"enabled\", False: \"disabled\"}\n\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        response = await client.request(\"GET\", f\"/webhooks/{webhook_id}\")\n        current_status = response[\"enabled\"]\n        new_status = not current_status\n\n        await client.request(\n            \"PATCH\", f\"/webhooks/{webhook_id}\", json={\"enabled\": new_status}\n        )\n        app.console.print(f\"Webhook is now {status_lookup[new_status]}\")\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud-webhook/#prefect.cli.cloud.webhook.update","title":"update async","text":"

    Partially update an existing Cloud webhook

    Source code in prefect/cli/cloud/webhook.py
    @webhook_app.command()\nasync def update(\n    webhook_id: UUID,\n    webhook_name: str = typer.Option(None, \"--name\", \"-n\", help=\"Webhook name\"),\n    description: str = typer.Option(\n        None, \"--description\", \"-d\", help=\"Description of the webhook\"\n    ),\n    template: str = typer.Option(\n        None, \"--template\", \"-t\", help=\"Jinja2 template expression\"\n    ),\n):\n    \"\"\"\n    Partially update an existing Cloud webhook\n    \"\"\"\n    confirm_logged_in()\n\n    # The /webhooks API lives inside the /accounts/{id}/workspaces/{id} routing tree\n    async with get_cloud_client(host=PREFECT_API_URL.value()) as client:\n        response = await client.request(\"GET\", f\"/webhooks/{webhook_id}\")\n        update_payload = {\n            \"name\": webhook_name or response[\"name\"],\n            \"description\": description or response[\"description\"],\n            \"template\": template or response[\"template\"],\n        }\n\n        await client.request(\"PUT\", f\"/webhooks/{webhook_id}\", json=update_payload)\n        app.console.print(f\"Successfully updated webhook {webhook_id}\")\n
    ","tags":["Python API","CLI","events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"api-ref/prefect/cli/cloud/","title":"cloud","text":"","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud","title":"prefect.cli.cloud","text":"

    Command line interface for interacting with Prefect Cloud

    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.login_api","title":"login_api = FastAPI(lifespan=lifespan) module-attribute","text":"

    This small API server is used for data transmission for browser-based log in.

    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.check_key_is_valid_for_login","title":"check_key_is_valid_for_login async","text":"

    Attempt to use a key to see if it is valid

    Source code in prefect/cli/cloud/__init__.py
    async def check_key_is_valid_for_login(key: str):\n    \"\"\"\n    Attempt to use a key to see if it is valid\n    \"\"\"\n    async with get_cloud_client(api_key=key) as client:\n        try:\n            await client.read_workspaces()\n            return True\n        except CloudUnauthorizedError:\n            return False\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.login","title":"login async","text":"

    Log in to Prefect Cloud. Creates a new profile configured to use the specified PREFECT_API_KEY. Uses a previously configured profile if it exists.

    Source code in prefect/cli/cloud/__init__.py
    @cloud_app.command()\nasync def login(\n    key: Optional[str] = typer.Option(\n        None, \"--key\", \"-k\", help=\"API Key to authenticate with Prefect\"\n    ),\n    workspace_handle: Optional[str] = typer.Option(\n        None,\n        \"--workspace\",\n        \"-w\",\n        help=(\n            \"Full handle of workspace, in format '<account_handle>/<workspace_handle>'\"\n        ),\n    ),\n):\n    \"\"\"\n    Log in to Prefect Cloud.\n    Creates a new profile configured to use the specified PREFECT_API_KEY.\n    Uses a previously configured profile if it exists.\n    \"\"\"\n    if not is_interactive() and (not key or not workspace_handle):\n        exit_with_error(\n            \"When not using an interactive terminal, you must supply a `--key` and\"\n            \" `--workspace`.\"\n        )\n\n    profiles = load_profiles()\n    current_profile = get_settings_context().profile\n    env_var_api_key = PREFECT_API_KEY.value()\n\n    if env_var_api_key and key and env_var_api_key != key:\n        exit_with_error(\n            \"Cannot log in with a key when a different PREFECT_API_KEY is present as an\"\n            \" environment variable that will override it.\"\n        )\n\n    if env_var_api_key and env_var_api_key == key:\n        is_valid_key = await check_key_is_valid_for_login(key)\n        is_correct_key_format = key.startswith(\"pnu_\") or key.startswith(\"pnb_\")\n        if not is_valid_key:\n            help_message = \"Please ensure your credentials are correct and unexpired.\"\n            if not is_correct_key_format:\n                help_message = \"Your key is not in our expected format.\"\n            exit_with_error(\n                f\"Unable to authenticate with Prefect Cloud. {help_message}\"\n            )\n\n    already_logged_in_profiles = []\n    for name, profile in profiles.items():\n        profile_key = profile.settings.get(PREFECT_API_KEY)\n        if (\n            # If a key is provided, only show profiles with the same key\n            (key and profile_key == key)\n            # Otherwise, show all profiles with a key set\n            or (not key and profile_key is not None)\n            # Check that the key is usable to avoid suggesting unauthenticated profiles\n            and await check_key_is_valid_for_login(profile_key)\n        ):\n            already_logged_in_profiles.append(name)\n\n    current_profile_is_logged_in = current_profile.name in already_logged_in_profiles\n\n    if current_profile_is_logged_in:\n        app.console.print(\"It looks like you're already authenticated on this profile.\")\n        should_reauth = typer.confirm(\n            \"? Would you like to reauthenticate?\", default=False\n        )\n        if not should_reauth:\n            app.console.print(\"Using the existing authentication on this profile.\")\n            key = PREFECT_API_KEY.value()\n\n    elif already_logged_in_profiles:\n        app.console.print(\n            \"It looks like you're already authenticated with another profile.\"\n        )\n        if not typer.confirm(\n            \"? Would you like to reauthenticate with this profile?\", default=False\n        ):\n            if typer.confirm(\n                \"? Would you like to switch to an authenticated profile?\", default=True\n            ):\n                profile_name = prompt_select_from_list(\n                    app.console,\n                    \"Which authenticated profile would you like to switch to?\",\n                    already_logged_in_profiles,\n                )\n\n                profiles.set_active(profile_name)\n                save_profiles(profiles)\n                exit_with_success(\n                    f\"Switched to authenticated profile {profile_name!r}.\"\n                )\n            else:\n                return\n\n    if not key:\n        choice = prompt_select_from_list(\n            app.console,\n            \"How would you like to authenticate?\",\n            [\n                (\"browser\", \"Log in with a web browser\"),\n                (\"key\", \"Paste an API key\"),\n            ],\n        )\n\n        if choice == \"key\":\n            key = typer.prompt(\"Paste your API key\", hide_input=True)\n        elif choice == \"browser\":\n            key = await login_with_browser()\n\n    async with get_cloud_client(api_key=key) as client:\n        try:\n            workspaces = await client.read_workspaces()\n        except CloudUnauthorizedError:\n            if key.startswith(\"pcu\"):\n                help_message = (\n                    \"It looks like you're using API key from Cloud 1\"\n                    \" (https://cloud.prefect.io). Make sure that you generate API key\"\n                    \" using Cloud 2 (https://app.prefect.cloud)\"\n                )\n            elif not key.startswith(\"pnu_\") and not key.startswith(\"pnb_\"):\n                help_message = (\n                    \"Your key is not in our expected format: 'pnu_' or 'pnb_'.\"\n                )\n            else:\n                help_message = (\n                    \"Please ensure your credentials are correct and unexpired.\"\n                )\n            exit_with_error(\n                f\"Unable to authenticate with Prefect Cloud. {help_message}\"\n            )\n        except httpx.HTTPStatusError as exc:\n            exit_with_error(f\"Error connecting to Prefect Cloud: {exc!r}\")\n\n    if workspace_handle:\n        # Search for the given workspace\n        for workspace in workspaces:\n            if workspace.handle == workspace_handle:\n                break\n        else:\n            if workspaces:\n                hint = (\n                    \" Available workspaces:\"\n                    f\" {listrepr((w.handle for w in workspaces), ', ')}\"\n                )\n            else:\n                hint = \"\"\n\n            exit_with_error(f\"Workspace {workspace_handle!r} not found.\" + hint)\n    else:\n        # Prompt a switch if the number of workspaces is greater than one\n        prompt_switch_workspace = len(workspaces) > 1\n\n        current_workspace = get_current_workspace(workspaces)\n\n        # Confirm that we want to switch if the current profile is already logged in\n        if (\n            current_profile_is_logged_in and current_workspace is not None\n        ) and prompt_switch_workspace:\n            app.console.print(\n                f\"You are currently using workspace {current_workspace.handle!r}.\"\n            )\n            prompt_switch_workspace = typer.confirm(\n                \"? Would you like to switch workspaces?\", default=False\n            )\n\n        if prompt_switch_workspace:\n            workspace = prompt_select_from_list(\n                app.console,\n                \"Which workspace would you like to use?\",\n                [(workspace, workspace.handle) for workspace in workspaces],\n            )\n        else:\n            if current_workspace:\n                workspace = current_workspace\n            elif len(workspaces) > 0:\n                workspace = workspaces[0]\n            else:\n                exit_with_error(\n                    \"No workspaces found! Create a workspace at\"\n                    f\" {PREFECT_CLOUD_UI_URL.value()} and try again.\"\n                )\n\n    update_current_profile(\n        {\n            PREFECT_API_KEY: key,\n            PREFECT_API_URL: workspace.api_url(),\n        }\n    )\n\n    exit_with_success(\n        f\"Authenticated with Prefect Cloud! Using workspace {workspace.handle!r}.\"\n    )\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.login_with_browser","title":"login_with_browser async","text":"

    Perform login using the browser.

    On failure, this function will exit the process. On success, it will return an API key.

    Source code in prefect/cli/cloud/__init__.py
    async def login_with_browser() -> str:\n    \"\"\"\n    Perform login using the browser.\n\n    On failure, this function will exit the process.\n    On success, it will return an API key.\n    \"\"\"\n\n    # Set up an event that the login API will toggle on startup\n    ready_event = login_api.extra[\"ready-event\"] = anyio.Event()\n\n    # Set up an event that the login API will set when a response comes from the UI\n    result_event = login_api.extra[\"result-event\"] = anyio.Event()\n\n    timeout_scope = None\n    async with anyio.create_task_group() as tg:\n        # Run a server in the background to get payload from the browser\n        server = await tg.start(serve_login_api, tg.cancel_scope)\n\n        # Wait for the login server to be ready\n        with anyio.fail_after(10):\n            await ready_event.wait()\n\n            # The server may not actually be serving as the lifespan is started first\n            while not server.started:\n                await anyio.sleep(0)\n\n        # Get the port the server is using\n        server_port = server.servers[0].sockets[0].getsockname()[1]\n        callback = urllib.parse.quote(f\"http://localhost:{server_port}\")\n        ui_login_url = (\n            PREFECT_CLOUD_UI_URL.value() + f\"/auth/client?callback={callback}\"\n        )\n\n        # Then open the authorization page in a new browser tab\n        app.console.print(\"Opening browser...\")\n        await run_sync_in_worker_thread(webbrowser.open_new_tab, ui_login_url)\n\n        # Wait for the response from the browser,\n        with anyio.move_on_after(120) as timeout_scope:\n            app.console.print(\"Waiting for response...\")\n            await result_event.wait()\n\n        # Uvicorn installs signal handlers, this is the cleanest way to shutdown the\n        # login API\n        raise_signal(signal.SIGINT)\n\n    result = login_api.extra.get(\"result\")\n    if not result:\n        if timeout_scope and timeout_scope.cancel_called:\n            exit_with_error(\"Timed out while waiting for authorization.\")\n        else:\n            exit_with_error(\"Aborted.\")\n\n    if result.type == \"success\":\n        return result.content.api_key\n    elif result.type == \"failure\":\n        exit_with_error(f\"Failed to log in. {result.content.reason}\")\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.logout","title":"logout async","text":"

    Logout the current workspace. Reset PREFECT_API_KEY and PREFECT_API_URL to default.

    Source code in prefect/cli/cloud/__init__.py
    @cloud_app.command()\nasync def logout():\n    \"\"\"\n    Logout the current workspace.\n    Reset PREFECT_API_KEY and PREFECT_API_URL to default.\n    \"\"\"\n    current_profile = prefect.context.get_settings_context().profile\n    if current_profile is None:\n        exit_with_error(\"There is no current profile set.\")\n\n    if current_profile.settings.get(PREFECT_API_KEY) is None:\n        exit_with_error(\"Current profile is not logged into Prefect Cloud.\")\n\n    update_current_profile(\n        {\n            PREFECT_API_URL: None,\n            PREFECT_API_KEY: None,\n        },\n    )\n\n    exit_with_success(\"Logged out from Prefect Cloud.\")\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.ls","title":"ls async","text":"

    List available workspaces.

    Source code in prefect/cli/cloud/__init__.py
    @workspace_app.command()\nasync def ls():\n    \"\"\"List available workspaces.\"\"\"\n\n    confirm_logged_in()\n\n    async with get_cloud_client() as client:\n        try:\n            workspaces = await client.read_workspaces()\n        except CloudUnauthorizedError:\n            exit_with_error(\n                \"Unable to authenticate. Please ensure your credentials are correct.\"\n            )\n\n    current_workspace = get_current_workspace(workspaces)\n\n    table = Table(caption=\"* active workspace\")\n    table.add_column(\n        \"[#024dfd]Workspaces:\", justify=\"left\", style=\"#8ea0ae\", no_wrap=True\n    )\n\n    for workspace_handle in sorted(workspace.handle for workspace in workspaces):\n        if workspace_handle == current_workspace.handle:\n            table.add_row(f\"[green]* {workspace_handle}[/green]\")\n        else:\n            table.add_row(f\"  {workspace_handle}\")\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.open","title":"open async","text":"

    Open the Prefect Cloud UI in the browser.

    Source code in prefect/cli/cloud/__init__.py
    @cloud_app.command()\nasync def open():\n    \"\"\"\n    Open the Prefect Cloud UI in the browser.\n    \"\"\"\n    confirm_logged_in()\n\n    current_profile = prefect.context.get_settings_context().profile\n    if current_profile is None:\n        exit_with_error(\n            \"There is no current profile set - set one with `prefect profile create\"\n            \" <name>` and `prefect profile use <name>`.\"\n        )\n\n    current_workspace = get_current_workspace(\n        await prefect.get_cloud_client().read_workspaces()\n    )\n    if current_workspace is None:\n        exit_with_error(\n            \"There is no current workspace set - set one with `prefect cloud workspace\"\n            \" set --workspace <workspace>`.\"\n        )\n\n    ui_url = current_workspace.ui_url()\n\n    await run_sync_in_worker_thread(webbrowser.open_new_tab, ui_url)\n\n    exit_with_success(f\"Opened {current_workspace.handle!r} in browser.\")\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.prompt_select_from_list","title":"prompt_select_from_list","text":"

    Given a list of options, display the values to user in a table and prompt them to select one.

    Parameters:

    Name Type Description Default options Union[List[str], List[Tuple[Hashable, str]]]

    A list of options to present to the user. A list of tuples can be passed as key value pairs. If a value is chosen, the key will be returned.

    required

    Returns:

    Name Type Description str str

    the selected option

    Source code in prefect/cli/cloud/__init__.py
    def prompt_select_from_list(\n    console, prompt: str, options: Union[List[str], List[Tuple[Hashable, str]]]\n) -> str:\n    \"\"\"\n    Given a list of options, display the values to user in a table and prompt them\n    to select one.\n\n    Args:\n        options: A list of options to present to the user.\n            A list of tuples can be passed as key value pairs. If a value is chosen, the\n            key will be returned.\n\n    Returns:\n        str: the selected option\n    \"\"\"\n\n    current_idx = 0\n    selected_option = None\n\n    def build_table() -> Table:\n        \"\"\"\n        Generate a table of options. The `current_idx` will be highlighted.\n        \"\"\"\n\n        table = Table(box=False, header_style=None, padding=(0, 0))\n        table.add_column(\n            f\"? [bold]{prompt}[/] [bright_blue][Use arrows to move; enter to select]\",\n            justify=\"left\",\n            no_wrap=True,\n        )\n\n        for i, option in enumerate(options):\n            if isinstance(option, tuple):\n                option = option[1]\n\n            if i == current_idx:\n                # Use blue for selected options\n                table.add_row(\"[bold][blue]> \" + option)\n            else:\n                table.add_row(\"  \" + option)\n        return table\n\n    with Live(build_table(), auto_refresh=False, console=console) as live:\n        while selected_option is None:\n            key = readchar.readkey()\n\n            if key == readchar.key.UP:\n                current_idx = current_idx - 1\n                # wrap to bottom if at the top\n                if current_idx < 0:\n                    current_idx = len(options) - 1\n            elif key == readchar.key.DOWN:\n                current_idx = current_idx + 1\n                # wrap to top if at the bottom\n                if current_idx >= len(options):\n                    current_idx = 0\n            elif key == readchar.key.CTRL_C:\n                # gracefully exit with no message\n                exit_with_error(\"\")\n            elif key == readchar.key.ENTER or key == readchar.key.CR:\n                selected_option = options[current_idx]\n                if isinstance(selected_option, tuple):\n                    selected_option = selected_option[0]\n\n            live.update(build_table(), refresh=True)\n\n        return selected_option\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/cloud/#prefect.cli.cloud.set","title":"set async","text":"

    Set current workspace. Shows a workspace picker if no workspace is specified.

    Source code in prefect/cli/cloud/__init__.py
    @workspace_app.command()\nasync def set(\n    workspace_handle: str = typer.Option(\n        None,\n        \"--workspace\",\n        \"-w\",\n        help=(\n            \"Full handle of workspace, in format '<account_handle>/<workspace_handle>'\"\n        ),\n    ),\n):\n    \"\"\"Set current workspace. Shows a workspace picker if no workspace is specified.\"\"\"\n    confirm_logged_in()\n\n    async with get_cloud_client() as client:\n        try:\n            workspaces = await client.read_workspaces()\n        except CloudUnauthorizedError:\n            exit_with_error(\n                \"Unable to authenticate. Please ensure your credentials are correct.\"\n            )\n\n    if workspace_handle:\n        # Search for the given workspace\n        for workspace in workspaces:\n            if workspace.handle == workspace_handle:\n                break\n        else:\n            exit_with_error(f\"Workspace {workspace_handle!r} not found.\")\n    else:\n        workspace = prompt_select_from_list(\n            app.console,\n            \"Which workspace would you like to use?\",\n            [(workspace, workspace.handle) for workspace in workspaces],\n        )\n\n    profile = update_current_profile({PREFECT_API_URL: workspace.api_url()})\n\n    exit_with_success(\n        f\"Successfully set workspace to {workspace.handle!r} in profile\"\n        f\" {profile.name!r}.\"\n    )\n
    ","tags":["Python API","CLI","authentication","Cloud"]},{"location":"api-ref/prefect/cli/concurrency_limit/","title":"concurrency_limit","text":"","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit","title":"prefect.cli.concurrency_limit","text":"

    Command line interface for working with concurrency limits.

    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.create","title":"create async","text":"

    Create a concurrency limit against a tag.

    This limit controls how many task runs with that tag may simultaneously be in a Running state.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def create(tag: str, concurrency_limit: int):\n    \"\"\"\n    Create a concurrency limit against a tag.\n\n    This limit controls how many task runs with that tag may simultaneously be in a\n    Running state.\n    \"\"\"\n\n    async with get_client() as client:\n        await client.create_concurrency_limit(\n            tag=tag, concurrency_limit=concurrency_limit\n        )\n        await client.read_concurrency_limit_by_tag(tag)\n\n    app.console.print(\n        textwrap.dedent(\n            f\"\"\"\n            Created concurrency limit with properties:\n                tag - {tag!r}\n                concurrency_limit - {concurrency_limit}\n\n            Delete the concurrency limit:\n                prefect concurrency-limit delete {tag!r}\n\n            Inspect the concurrency limit:\n                prefect concurrency-limit inspect {tag!r}\n        \"\"\"\n        )\n    )\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.delete","title":"delete async","text":"

    Delete the concurrency limit set on the specified tag.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def delete(tag: str):\n    \"\"\"\n    Delete the concurrency limit set on the specified tag.\n    \"\"\"\n\n    async with get_client() as client:\n        try:\n            await client.delete_concurrency_limit_by_tag(tag=tag)\n        except ObjectNotFound:\n            exit_with_error(f\"No concurrency limit found for the tag: {tag}\")\n\n    exit_with_success(f\"Deleted concurrency limit set on the tag: {tag}\")\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.inspect","title":"inspect async","text":"

    View details about a concurrency limit. active_slots shows a list of TaskRun IDs which are currently using a concurrency slot.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def inspect(tag: str):\n    \"\"\"\n    View details about a concurrency limit. `active_slots` shows a list of TaskRun IDs\n    which are currently using a concurrency slot.\n    \"\"\"\n\n    async with get_client() as client:\n        try:\n            result = await client.read_concurrency_limit_by_tag(tag=tag)\n        except ObjectNotFound:\n            exit_with_error(f\"No concurrency limit found for the tag: {tag}\")\n\n    trid_table = Table()\n    trid_table.add_column(\"Active Task Run IDs\", style=\"cyan\", no_wrap=True)\n\n    cl_table = Table(title=f\"Concurrency Limit ID: [red]{str(result.id)}\")\n    cl_table.add_column(\"Tag\", style=\"green\", no_wrap=True)\n    cl_table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n    cl_table.add_column(\"Created\", style=\"magenta\", no_wrap=True)\n    cl_table.add_column(\"Updated\", style=\"magenta\", no_wrap=True)\n\n    for trid in sorted(result.active_slots):\n        trid_table.add_row(str(trid))\n\n    cl_table.add_row(\n        str(result.tag),\n        str(result.concurrency_limit),\n        Pretty(pendulum.instance(result.created).diff_for_humans()),\n        Pretty(pendulum.instance(result.updated).diff_for_humans()),\n    )\n\n    group = Group(\n        cl_table,\n        trid_table,\n    )\n    app.console.print(Panel(group, expand=False))\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.ls","title":"ls async","text":"

    View all concurrency limits.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def ls(limit: int = 15, offset: int = 0):\n    \"\"\"\n    View all concurrency limits.\n    \"\"\"\n    table = Table(\n        title=\"Concurrency Limits\",\n        caption=\"inspect a concurrency limit to show active task run IDs\",\n    )\n    table.add_column(\"Tag\", style=\"green\", no_wrap=True)\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Active Task Runs\", style=\"magenta\", no_wrap=True)\n\n    async with get_client() as client:\n        concurrency_limits = await client.read_concurrency_limits(\n            limit=limit, offset=offset\n        )\n\n    for cl in sorted(concurrency_limits, key=lambda c: c.updated, reverse=True):\n        table.add_row(\n            str(cl.tag),\n            str(cl.id),\n            str(cl.concurrency_limit),\n            str(len(cl.active_slots)),\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/concurrency_limit/#prefect.cli.concurrency_limit.reset","title":"reset async","text":"

    Resets the concurrency limit slots set on the specified tag.

    Source code in prefect/cli/concurrency_limit.py
    @concurrency_limit_app.command()\nasync def reset(tag: str):\n    \"\"\"\n    Resets the concurrency limit slots set on the specified tag.\n    \"\"\"\n\n    async with get_client() as client:\n        try:\n            await client.reset_concurrency_limit_by_tag(tag=tag)\n        except ObjectNotFound:\n            exit_with_error(f\"No concurrency limit found for the tag: {tag}\")\n\n    exit_with_success(f\"Reset concurrency limit set on the tag: {tag}\")\n
    ","tags":["Python API","CLI","concurrency"]},{"location":"api-ref/prefect/cli/config/","title":"config","text":"","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config","title":"prefect.cli.config","text":"

    Command line interface for working with profiles

    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config.set_","title":"set_","text":"

    Change the value for a setting by setting the value in the current profile.

    Source code in prefect/cli/config.py
    @config_app.command(\"set\")\ndef set_(settings: List[str]):\n    \"\"\"\n    Change the value for a setting by setting the value in the current profile.\n    \"\"\"\n    parsed_settings = {}\n    for item in settings:\n        try:\n            setting, value = item.split(\"=\", maxsplit=1)\n        except ValueError:\n            exit_with_error(\n                f\"Failed to parse argument {item!r}. Use the format 'VAR=VAL'.\"\n            )\n\n        if setting not in prefect.settings.SETTING_VARIABLES:\n            exit_with_error(f\"Unknown setting name {setting!r}.\")\n\n        # Guard against changing settings that tweak config locations\n        if setting in {\"PREFECT_HOME\", \"PREFECT_PROFILES_PATH\"}:\n            exit_with_error(\n                f\"Setting {setting!r} cannot be changed with this command. \"\n                \"Use an environment variable instead.\"\n            )\n\n        parsed_settings[setting] = value\n\n    try:\n        new_profile = prefect.settings.update_current_profile(parsed_settings)\n    except pydantic.ValidationError as exc:\n        for error in exc.errors():\n            setting = error[\"loc\"][0]\n            message = error[\"msg\"]\n            app.console.print(f\"Validation error for setting {setting!r}: {message}\")\n        exit_with_error(\"Invalid setting value.\")\n\n    for setting, value in parsed_settings.items():\n        app.console.print(f\"Set {setting!r} to {value!r}.\")\n        if setting in os.environ:\n            app.console.print(\n                f\"[yellow]{setting} is also set by an environment variable which will \"\n                f\"override your config value. Run `unset {setting}` to clear it.\"\n            )\n\n        if prefect.settings.SETTING_VARIABLES[setting].deprecated:\n            app.console.print(\n                f\"[yellow]{prefect.settings.SETTING_VARIABLES[setting].deprecated_message}.\"\n            )\n\n    exit_with_success(f\"Updated profile {new_profile.name!r}.\")\n
    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config.unset","title":"unset","text":"

    Restore the default value for a setting.

    Removes the setting from the current profile.

    Source code in prefect/cli/config.py
    @config_app.command()\ndef unset(settings: List[str]):\n    \"\"\"\n    Restore the default value for a setting.\n\n    Removes the setting from the current profile.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    profile = profiles[prefect.context.get_settings_context().profile.name]\n    parsed = set()\n\n    for setting in settings:\n        if setting not in prefect.settings.SETTING_VARIABLES:\n            exit_with_error(f\"Unknown setting name {setting!r}.\")\n        # Cast to settings objects\n        parsed.add(prefect.settings.SETTING_VARIABLES[setting])\n\n    for setting in parsed:\n        if setting not in profile.settings:\n            exit_with_error(f\"{setting.name!r} is not set in profile {profile.name!r}.\")\n\n    profiles.update_profile(\n        name=profile.name, settings={setting: None for setting in parsed}\n    )\n\n    for setting in settings:\n        app.console.print(f\"Unset {setting!r}.\")\n\n        if setting in os.environ:\n            app.console.print(\n                f\"[yellow]{setting!r} is also set by an environment variable. \"\n                f\"Use `unset {setting}` to clear it.\"\n            )\n\n    prefect.settings.save_profiles(profiles)\n    exit_with_success(f\"Updated profile {profile.name!r}.\")\n
    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config.validate","title":"validate","text":"

    Read and validate the current profile.

    Deprecated settings will be automatically converted to new names unless both are set.

    Source code in prefect/cli/config.py
    @config_app.command()\ndef validate():\n    \"\"\"\n    Read and validate the current profile.\n\n    Deprecated settings will be automatically converted to new names unless both are\n    set.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    profile = profiles[prefect.context.get_settings_context().profile.name]\n    changed = profile.convert_deprecated_renamed_settings()\n    for old, new in changed:\n        app.console.print(f\"Updated {old.name!r} to {new.name!r}.\")\n\n    for setting in profile.settings.keys():\n        if setting.deprecated:\n            app.console.print(f\"Found deprecated setting {setting.name!r}.\")\n\n    profile.validate_settings()\n\n    prefect.settings.save_profiles(profiles)\n    exit_with_success(\"Configuration valid!\")\n
    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/config/#prefect.cli.config.view","title":"view","text":"

    Display the current settings.

    Source code in prefect/cli/config.py
    @config_app.command()\ndef view(\n    show_defaults: Optional[bool] = typer.Option(\n        False, \"--show-defaults/--hide-defaults\", help=(show_defaults_help)\n    ),\n    show_sources: Optional[bool] = typer.Option(\n        True,\n        \"--show-sources/--hide-sources\",\n        help=(show_sources_help),\n    ),\n    show_secrets: Optional[bool] = typer.Option(\n        False,\n        \"--show-secrets/--hide-secrets\",\n        help=\"Toggle display of secrets setting values.\",\n    ),\n):\n    \"\"\"\n    Display the current settings.\n    \"\"\"\n    context = prefect.context.get_settings_context()\n\n    # Get settings at each level, converted to a flat dictionary for easy comparison\n    default_settings = prefect.settings.get_default_settings()\n    env_settings = prefect.settings.get_settings_from_env()\n    current_profile_settings = context.settings\n\n    # Obfuscate secrets\n    if not show_secrets:\n        default_settings = default_settings.with_obfuscated_secrets()\n        env_settings = env_settings.with_obfuscated_secrets()\n        current_profile_settings = current_profile_settings.with_obfuscated_secrets()\n\n    # Display the profile first\n    app.console.print(f\"PREFECT_PROFILE={context.profile.name!r}\")\n\n    settings_output = []\n\n    # The combination of environment variables and profile settings that are in use\n    profile_overrides = current_profile_settings.dict(exclude_unset=True)\n\n    # Used to see which settings in current_profile_settings came from env vars\n    env_overrides = env_settings.dict(exclude_unset=True)\n\n    for key, value in profile_overrides.items():\n        source = \"env\" if env_overrides.get(key) is not None else \"profile\"\n        source_blurb = f\" (from {source})\" if show_sources else \"\"\n        settings_output.append(f\"{key}='{value}'{source_blurb}\")\n\n    if show_defaults:\n        for key, value in default_settings.dict().items():\n            if key not in profile_overrides:\n                source_blurb = \" (from defaults)\" if show_sources else \"\"\n                settings_output.append(f\"{key}='{value}'{source_blurb}\")\n\n    app.console.print(\"\\n\".join(sorted(settings_output)))\n
    ","tags":["Python API","CLI","config","settings"]},{"location":"api-ref/prefect/cli/deploy/","title":"deploy","text":"","tags":["Python API","deploy","deployment","CLI"]},{"location":"api-ref/prefect/cli/deploy/#prefect.cli.deploy","title":"prefect.cli.deploy","text":"

    Module containing implementation for deploying projects.

    ","tags":["Python API","deploy","deployment","CLI"]},{"location":"api-ref/prefect/cli/deploy/#prefect.cli.deploy.deploy","title":"deploy async","text":"

    Deploy a flow from this project by creating a deployment.

    Should be run from a project root directory.

    Source code in prefect/cli/deploy.py
    @app.command()\nasync def deploy(\n    entrypoint: str = typer.Argument(\n        None,\n        help=(\n            \"The path to a flow entrypoint within a project, in the form of\"\n            \" `./path/to/file.py:flow_func_name`\"\n        ),\n    ),\n    flow_name: str = typer.Option(\n        None,\n        \"--flow\",\n        \"-f\",\n        help=\"DEPRECATED: The name of a registered flow to create a deployment for.\",\n    ),\n    names: List[str] = typer.Option(\n        None,\n        \"--name\",\n        \"-n\",\n        help=(\n            \"The name to give the deployment. Can be a pattern. Examples:\"\n            \" 'my-deployment', 'my-flow/my-deployment', 'my-deployment-*',\"\n            \" '*-flow-name/deployment*'\"\n        ),\n    ),\n    description: str = typer.Option(\n        None,\n        \"--description\",\n        \"-d\",\n        help=(\n            \"The description to give the deployment. If not provided, the description\"\n            \" will be populated from the flow's description.\"\n        ),\n    ),\n    version: str = typer.Option(\n        None, \"--version\", help=\"A version to give the deployment.\"\n    ),\n    tags: List[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=(\n            \"One or more optional tags to apply to the deployment. Note: tags are used\"\n            \" only for organizational purposes. For delegating work to agents, use the\"\n            \" --work-queue flag.\"\n        ),\n    ),\n    work_pool_name: str = SettingsOption(\n        PREFECT_DEFAULT_WORK_POOL_NAME,\n        \"-p\",\n        \"--pool\",\n        help=\"The work pool that will handle this deployment's runs.\",\n    ),\n    work_queue_name: str = typer.Option(\n        None,\n        \"-q\",\n        \"--work-queue\",\n        help=(\n            \"The work queue that will handle this deployment's runs. \"\n            \"It will be created if it doesn't already exist. Defaults to `None`.\"\n        ),\n    ),\n    variables: List[str] = typer.Option(\n        None,\n        \"-v\",\n        \"--variable\",\n        help=(\n            \"One or more job variable overrides for the work pool provided in the\"\n            \" format of key=value string or a JSON object\"\n        ),\n    ),\n    cron: str = typer.Option(\n        None,\n        \"--cron\",\n        help=\"A cron string that will be used to set a CronSchedule on the deployment.\",\n    ),\n    interval: int = typer.Option(\n        None,\n        \"--interval\",\n        help=(\n            \"An integer specifying an interval (in seconds) that will be used to set an\"\n            \" IntervalSchedule on the deployment.\"\n        ),\n    ),\n    interval_anchor: Optional[str] = typer.Option(\n        None, \"--anchor-date\", help=\"The anchor date for an interval schedule\"\n    ),\n    rrule: str = typer.Option(\n        None,\n        \"--rrule\",\n        help=\"An RRule that will be used to set an RRuleSchedule on the deployment.\",\n    ),\n    timezone: str = typer.Option(\n        None,\n        \"--timezone\",\n        help=\"Deployment schedule timezone string e.g. 'America/New_York'\",\n    ),\n    trigger: List[str] = typer.Option(\n        None,\n        \"--trigger\",\n        help=(\n            \"Specifies a trigger for the deployment. The value can be a\"\n            \" json string or path to `.yaml`/`.json` file. This flag can be used\"\n            \" multiple times.\"\n        ),\n    ),\n    param: List[str] = typer.Option(\n        None,\n        \"--param\",\n        help=(\n            \"An optional parameter override, values are parsed as JSON strings e.g.\"\n            \" --param question=ultimate --param answer=42\"\n        ),\n    ),\n    params: str = typer.Option(\n        None,\n        \"--params\",\n        help=(\n            \"An optional parameter override in a JSON string format e.g.\"\n            ' --params=\\'{\"question\": \"ultimate\", \"answer\": 42}\\''\n        ),\n    ),\n    enforce_parameter_schema: bool = typer.Option(\n        False,\n        \"--enforce-parameter-schema\",\n        help=(\n            \"Whether to enforce the parameter schema on this deployment. If set to\"\n            \" True, any parameters passed to this deployment must match the signature\"\n            \" of the flow.\"\n        ),\n    ),\n    deploy_all: bool = typer.Option(\n        False,\n        \"--all\",\n        help=(\n            \"Deploy all flows in the project. If a flow name or entrypoint is also\"\n            \" provided, this flag will be ignored.\"\n        ),\n    ),\n    ci: bool = typer.Option(\n        False,\n        \"--ci\",\n        help=(\n            \"DEPRECATED: Please use the global '--no-prompt' flag instead: 'prefect\"\n            \" --no-prompt deploy'.\\n\\nRun this command in CI mode. This will disable\"\n            \" interactive prompts and will error if any required arguments are not\"\n            \" provided.\"\n        ),\n    ),\n):\n    \"\"\"\n    Deploy a flow from this project by creating a deployment.\n\n    Should be run from a project root directory.\n    \"\"\"\n    if ci:\n        app.console.print(\n            generate_deprecation_message(\n                name=\"The `--ci` flag\",\n                start_date=\"Jun 2023\",\n                help=(\n                    \"Please use the global `--no-prompt` flag instead: `prefect\"\n                    \" --no-prompt deploy`.\"\n                ),\n            ),\n            style=\"yellow\",\n        )\n\n    options = {\n        \"entrypoint\": entrypoint,\n        \"flow_name\": flow_name,\n        \"description\": description,\n        \"version\": version,\n        \"tags\": tags,\n        \"work_pool_name\": work_pool_name,\n        \"work_queue_name\": work_queue_name,\n        \"variables\": variables,\n        \"cron\": cron,\n        \"interval\": interval,\n        \"anchor_date\": interval_anchor,\n        \"rrule\": rrule,\n        \"timezone\": timezone,\n        \"triggers\": trigger,\n        \"param\": param,\n        \"params\": params,\n        \"enforce_parameter_schema\": enforce_parameter_schema,\n    }\n    try:\n        deploy_configs, actions = _load_deploy_configs_and_actions(ci=ci)\n\n        parsed_names = []\n        for name in names:\n            if \"*\" in name:\n                parsed_names.extend(_parse_name_from_pattern(deploy_configs, name))\n            else:\n                parsed_names.append(name)\n        deploy_configs = _pick_deploy_configs(\n            deploy_configs, parsed_names, deploy_all, ci\n        )\n\n        if len(deploy_configs) > 1:\n            if any(options.values()):\n                app.console.print(\n                    (\n                        \"You have passed options to the deploy command, but you are\"\n                        \" creating or updating multiple deployments. These options\"\n                        \" will be ignored.\"\n                    ),\n                    style=\"yellow\",\n                )\n            await _run_multi_deploy(\n                deploy_configs=deploy_configs,\n                actions=actions,\n                deploy_all=deploy_all,\n                ci=ci,\n            )\n        else:\n            # Accommodate passing in -n flow-name/deployment-name as well as -n deployment-name\n            options[\"names\"] = [\n                name.split(\"/\", 1)[-1] if \"/\" in name else name for name in parsed_names\n            ]\n\n            await _run_single_deploy(\n                deploy_config=deploy_configs[0] if deploy_configs else {},\n                actions=actions,\n                options=options,\n                ci=ci,\n            )\n    except ValueError as exc:\n        exit_with_error(str(exc))\n
    ","tags":["Python API","deploy","deployment","CLI"]},{"location":"api-ref/prefect/cli/deployment/","title":"deployment","text":"","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment","title":"prefect.cli.deployment","text":"

    Command line interface for working with deployments.

    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.apply","title":"apply async","text":"

    Create or update a deployment from a YAML file.

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def apply(\n    paths: List[str] = typer.Argument(\n        ...,\n        help=\"One or more paths to deployment YAML files.\",\n    ),\n    upload: bool = typer.Option(\n        False,\n        \"--upload\",\n        help=(\n            \"A flag that, when provided, uploads this deployment's files to remote\"\n            \" storage.\"\n        ),\n    ),\n    work_queue_concurrency: int = typer.Option(\n        None,\n        \"--limit\",\n        \"-l\",\n        help=(\n            \"Sets the concurrency limit on the work queue that handles this\"\n            \" deployment's runs\"\n        ),\n    ),\n):\n    \"\"\"\n    Create or update a deployment from a YAML file.\n    \"\"\"\n    deployment = None\n    async with get_client() as client:\n        for path in paths:\n            try:\n                deployment = await Deployment.load_from_yaml(path)\n                app.console.print(\n                    f\"Successfully loaded {deployment.name!r}\", style=\"green\"\n                )\n            except Exception as exc:\n                exit_with_error(\n                    f\"'{path!s}' did not conform to deployment spec: {exc!r}\"\n                )\n\n            assert deployment\n\n            await create_work_queue_and_set_concurrency_limit(\n                deployment.work_queue_name,\n                deployment.work_pool_name,\n                work_queue_concurrency,\n            )\n\n            if upload:\n                if (\n                    deployment.storage\n                    and \"put-directory\" in deployment.storage.get_block_capabilities()\n                ):\n                    file_count = await deployment.upload_to_storage()\n                    if file_count:\n                        app.console.print(\n                            (\n                                f\"Successfully uploaded {file_count} files to\"\n                                f\" {deployment.location}\"\n                            ),\n                            style=\"green\",\n                        )\n                else:\n                    app.console.print(\n                        (\n                            f\"Deployment storage {deployment.storage} does not have\"\n                            \" upload capabilities; no files uploaded.\"\n                        ),\n                        style=\"red\",\n                    )\n            await check_work_pool_exists(\n                work_pool_name=deployment.work_pool_name, client=client\n            )\n\n            if client.server_type != ServerType.CLOUD and deployment.triggers:\n                app.console.print(\n                    (\n                        \"Deployment triggers are only supported on \"\n                        f\"Prefect Cloud. Triggers defined in {path!r} will be \"\n                        \"ignored.\"\n                    ),\n                    style=\"red\",\n                )\n\n            deployment_id = await deployment.apply()\n            app.console.print(\n                (\n                    f\"Deployment '{deployment.flow_name}/{deployment.name}'\"\n                    f\" successfully created with id '{deployment_id}'.\"\n                ),\n                style=\"green\",\n            )\n\n            if PREFECT_UI_URL:\n                app.console.print(\n                    \"View Deployment in UI:\"\n                    f\" {PREFECT_UI_URL.value()}/deployments/deployment/{deployment_id}\"\n                )\n\n            if deployment.work_pool_name is not None:\n                await _print_deployment_work_pool_instructions(\n                    work_pool_name=deployment.work_pool_name, client=client\n                )\n            elif deployment.work_queue_name is not None:\n                app.console.print(\n                    \"\\nTo execute flow runs from this deployment, start an agent that\"\n                    f\" pulls work from the {deployment.work_queue_name!r} work queue:\"\n                )\n                app.console.print(\n                    f\"$ prefect agent start -q {deployment.work_queue_name!r}\",\n                    style=\"blue\",\n                )\n            else:\n                app.console.print(\n                    (\n                        \"\\nThis deployment does not specify a work queue name, which\"\n                        \" means agents will not be able to pick up its runs. To add a\"\n                        \" work queue, edit the deployment spec and re-run this command,\"\n                        \" or visit the deployment in the UI.\"\n                    ),\n                    style=\"red\",\n                )\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.build","title":"build async","text":"

    Generate a deployment YAML from /path/to/file.py:flow_function

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def build(\n    entrypoint: str = typer.Argument(\n        ...,\n        help=(\n            \"The path to a flow entrypoint, in the form of\"\n            \" `./path/to/file.py:flow_func_name`\"\n        ),\n    ),\n    name: str = typer.Option(\n        None, \"--name\", \"-n\", help=\"The name to give the deployment.\"\n    ),\n    description: str = typer.Option(\n        None,\n        \"--description\",\n        \"-d\",\n        help=(\n            \"The description to give the deployment. If not provided, the description\"\n            \" will be populated from the flow's description.\"\n        ),\n    ),\n    version: str = typer.Option(\n        None, \"--version\", \"-v\", help=\"A version to give the deployment.\"\n    ),\n    tags: List[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=(\n            \"One or more optional tags to apply to the deployment. Note: tags are used\"\n            \" only for organizational purposes. For delegating work to agents, use the\"\n            \" --work-queue flag.\"\n        ),\n    ),\n    work_queue_name: str = typer.Option(\n        None,\n        \"-q\",\n        \"--work-queue\",\n        help=(\n            \"The work queue that will handle this deployment's runs. \"\n            \"It will be created if it doesn't already exist. Defaults to `None`. \"\n            \"Note that if a work queue is not set, work will not be scheduled.\"\n        ),\n    ),\n    work_pool_name: str = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The work pool that will handle this deployment's runs.\",\n    ),\n    work_queue_concurrency: int = typer.Option(\n        None,\n        \"--limit\",\n        \"-l\",\n        help=(\n            \"Sets the concurrency limit on the work queue that handles this\"\n            \" deployment's runs\"\n        ),\n    ),\n    infra_type: str = typer.Option(\n        None,\n        \"--infra\",\n        \"-i\",\n        help=\"The infrastructure type to use, prepopulated with defaults. For example: \"\n        + listrepr(builtin_infrastructure_types, sep=\", \"),\n    ),\n    infra_block: str = typer.Option(\n        None,\n        \"--infra-block\",\n        \"-ib\",\n        help=\"The slug of the infrastructure block to use as a template.\",\n    ),\n    overrides: List[str] = typer.Option(\n        None,\n        \"--override\",\n        help=(\n            \"One or more optional infrastructure overrides provided as a dot delimited\"\n            \" path, e.g., `env.env_key=env_value`\"\n        ),\n    ),\n    storage_block: str = typer.Option(\n        None,\n        \"--storage-block\",\n        \"-sb\",\n        help=(\n            \"The slug of a remote storage block. Use the syntax:\"\n            \" 'block_type/block_name', where block_type is one of 'github', 's3',\"\n            \" 'gcs', 'azure', 'smb', or a registered block from a library that\"\n            \" implements the WritableDeploymentStorage interface such as\"\n            \" 'gitlab-repository', 'bitbucket-repository', 's3-bucket',\"\n            \" 'gcs-bucket'\"\n        ),\n    ),\n    skip_upload: bool = typer.Option(\n        False,\n        \"--skip-upload\",\n        help=(\n            \"A flag that, when provided, skips uploading this deployment's files to\"\n            \" remote storage.\"\n        ),\n    ),\n    cron: str = typer.Option(\n        None,\n        \"--cron\",\n        help=\"A cron string that will be used to set a CronSchedule on the deployment.\",\n    ),\n    interval: int = typer.Option(\n        None,\n        \"--interval\",\n        help=(\n            \"An integer specifying an interval (in seconds) that will be used to set an\"\n            \" IntervalSchedule on the deployment.\"\n        ),\n    ),\n    interval_anchor: Optional[str] = typer.Option(\n        None, \"--anchor-date\", help=\"The anchor date for an interval schedule\"\n    ),\n    rrule: str = typer.Option(\n        None,\n        \"--rrule\",\n        help=\"An RRule that will be used to set an RRuleSchedule on the deployment.\",\n    ),\n    timezone: str = typer.Option(\n        None,\n        \"--timezone\",\n        help=\"Deployment schedule timezone string e.g. 'America/New_York'\",\n    ),\n    path: str = typer.Option(\n        None,\n        \"--path\",\n        help=(\n            \"An optional path to specify a subdirectory of remote storage to upload to,\"\n            \" or to point to a subdirectory of a locally stored flow.\"\n        ),\n    ),\n    output: str = typer.Option(\n        None,\n        \"--output\",\n        \"-o\",\n        help=\"An optional filename to write the deployment file to.\",\n    ),\n    _apply: bool = typer.Option(\n        False,\n        \"--apply\",\n        \"-a\",\n        help=(\n            \"An optional flag to automatically register the resulting deployment with\"\n            \" the API.\"\n        ),\n    ),\n    param: List[str] = typer.Option(\n        None,\n        \"--param\",\n        help=(\n            \"An optional parameter override, values are parsed as JSON strings e.g.\"\n            \" --param question=ultimate --param answer=42\"\n        ),\n    ),\n    params: str = typer.Option(\n        None,\n        \"--params\",\n        help=(\n            \"An optional parameter override in a JSON string format e.g.\"\n            ' --params=\\'{\"question\": \"ultimate\", \"answer\": 42}\\''\n        ),\n    ),\n    no_schedule: bool = typer.Option(\n        False,\n        \"--no-schedule\",\n        help=\"An optional flag to disable scheduling for this deployment.\",\n    ),\n):\n    \"\"\"\n    Generate a deployment YAML from /path/to/file.py:flow_function\n    \"\"\"\n    # validate inputs\n    if not name:\n        exit_with_error(\n            \"A name for this deployment must be provided with the '--name' flag.\"\n        )\n\n    if (\n        len([value for value in (cron, rrule, interval) if value is not None])\n        + (1 if no_schedule else 0)\n        > 1\n    ):\n        exit_with_error(\"Only one schedule type can be provided.\")\n\n    if infra_block and infra_type:\n        exit_with_error(\n            \"Only one of `infra` or `infra_block` can be provided, please choose one.\"\n        )\n\n    output_file = None\n    if output:\n        output_file = Path(output)\n        if output_file.suffix and output_file.suffix != \".yaml\":\n            exit_with_error(\"Output file must be a '.yaml' file.\")\n        else:\n            output_file = output_file.with_suffix(\".yaml\")\n\n    # validate flow\n    try:\n        fpath, obj_name = entrypoint.rsplit(\":\", 1)\n    except ValueError as exc:\n        if str(exc) == \"not enough values to unpack (expected 2, got 1)\":\n            missing_flow_name_msg = (\n                \"Your flow entrypoint must include the name of the function that is\"\n                f\" the entrypoint to your flow.\\nTry {entrypoint}:<flow_name>\"\n            )\n            exit_with_error(missing_flow_name_msg)\n        else:\n            raise exc\n    try:\n        flow = await run_sync_in_worker_thread(load_flow_from_entrypoint, entrypoint)\n    except Exception as exc:\n        exit_with_error(exc)\n    app.console.print(f\"Found flow {flow.name!r}\", style=\"green\")\n    infra_overrides = {}\n    for override in overrides or []:\n        key, value = override.split(\"=\", 1)\n        infra_overrides[key] = value\n\n    if infra_block:\n        infrastructure = await Block.load(infra_block)\n    elif infra_type:\n        # Create an instance of the given type\n        infrastructure = Block.get_block_class_from_key(infra_type)()\n    else:\n        # will reset to a default of Process is no infra is present on the\n        # server-side definition of this deployment\n        infrastructure = None\n\n    if interval_anchor and not interval:\n        exit_with_error(\"An anchor date can only be provided with an interval schedule\")\n\n    schedule = None\n    if cron:\n        cron_kwargs = {\"cron\": cron, \"timezone\": timezone}\n        schedule = CronSchedule(\n            **{k: v for k, v in cron_kwargs.items() if v is not None}\n        )\n    elif interval:\n        interval_kwargs = {\n            \"interval\": timedelta(seconds=interval),\n            \"anchor_date\": interval_anchor,\n            \"timezone\": timezone,\n        }\n        schedule = IntervalSchedule(\n            **{k: v for k, v in interval_kwargs.items() if v is not None}\n        )\n    elif rrule:\n        try:\n            schedule = RRuleSchedule(**json.loads(rrule))\n            if timezone:\n                # override timezone if specified via CLI argument\n                schedule.timezone = timezone\n        except json.JSONDecodeError:\n            schedule = RRuleSchedule(rrule=rrule, timezone=timezone)\n\n    # parse storage_block\n    if storage_block:\n        block_type, block_name, *block_path = storage_block.split(\"/\")\n        if block_path and path:\n            exit_with_error(\n                \"Must provide a `path` explicitly or provide one on the storage block\"\n                \" specification, but not both.\"\n            )\n        elif not path:\n            path = \"/\".join(block_path)\n        storage_block = f\"{block_type}/{block_name}\"\n        storage = await Block.load(storage_block)\n    else:\n        storage = None\n\n    if create_default_ignore_file(path=\".\"):\n        app.console.print(\n            (\n                \"Default '.prefectignore' file written to\"\n                f\" {(Path('.') / '.prefectignore').absolute()}\"\n            ),\n            style=\"green\",\n        )\n\n    if param and (params is not None):\n        exit_with_error(\"Can only pass one of `param` or `params` options\")\n\n    parameters = dict()\n\n    if param:\n        for p in param or []:\n            k, unparsed_value = p.split(\"=\", 1)\n            try:\n                v = json.loads(unparsed_value)\n                app.console.print(\n                    f\"The parameter value {unparsed_value} is parsed as a JSON string\"\n                )\n            except json.JSONDecodeError:\n                v = unparsed_value\n            parameters[k] = v\n\n    if params is not None:\n        parameters = json.loads(params)\n\n    # set up deployment object\n    entrypoint = (\n        f\"{Path(fpath).absolute().relative_to(Path('.').absolute())}:{obj_name}\"\n    )\n\n    init_kwargs = dict(\n        path=path,\n        entrypoint=entrypoint,\n        version=version,\n        storage=storage,\n        infra_overrides=infra_overrides or {},\n    )\n\n    if parameters:\n        init_kwargs[\"parameters\"] = parameters\n\n    if description:\n        init_kwargs[\"description\"] = description\n\n    # if a schedule, tags, work_queue_name, or infrastructure are not provided via CLI,\n    # we let `build_from_flow` load them from the server\n    if schedule or no_schedule:\n        init_kwargs.update(schedule=schedule)\n    if tags:\n        init_kwargs.update(tags=tags)\n    if infrastructure:\n        init_kwargs.update(infrastructure=infrastructure)\n    if work_queue_name:\n        init_kwargs.update(work_queue_name=work_queue_name)\n    if work_pool_name:\n        init_kwargs.update(work_pool_name=work_pool_name)\n\n    deployment_loc = output_file or f\"{obj_name}-deployment.yaml\"\n    deployment = await Deployment.build_from_flow(\n        flow=flow,\n        name=name,\n        output=deployment_loc,\n        skip_upload=skip_upload,\n        apply=False,\n        **init_kwargs,\n    )\n    app.console.print(\n        f\"Deployment YAML created at '{Path(deployment_loc).absolute()!s}'.\",\n        style=\"green\",\n    )\n\n    await create_work_queue_and_set_concurrency_limit(\n        deployment.work_queue_name, deployment.work_pool_name, work_queue_concurrency\n    )\n\n    # we process these separately for informative output\n    if not skip_upload:\n        if (\n            deployment.storage\n            and \"put-directory\" in deployment.storage.get_block_capabilities()\n        ):\n            file_count = await deployment.upload_to_storage()\n            if file_count:\n                app.console.print(\n                    (\n                        f\"Successfully uploaded {file_count} files to\"\n                        f\" {deployment.location}\"\n                    ),\n                    style=\"green\",\n                )\n        else:\n            app.console.print(\n                (\n                    f\"Deployment storage {deployment.storage} does not have upload\"\n                    \" capabilities; no files uploaded.  Pass --skip-upload to suppress\"\n                    \" this warning.\"\n                ),\n                style=\"green\",\n            )\n\n    if _apply:\n        async with get_client() as client:\n            await check_work_pool_exists(\n                work_pool_name=deployment.work_pool_name, client=client\n            )\n            deployment_id = await deployment.apply()\n            app.console.print(\n                (\n                    f\"Deployment '{deployment.flow_name}/{deployment.name}'\"\n                    f\" successfully created with id '{deployment_id}'.\"\n                ),\n                style=\"green\",\n            )\n            if deployment.work_pool_name is not None:\n                await _print_deployment_work_pool_instructions(\n                    work_pool_name=deployment.work_pool_name, client=client\n                )\n\n            elif deployment.work_queue_name is not None:\n                app.console.print(\n                    \"\\nTo execute flow runs from this deployment, start an agent that\"\n                    f\" pulls work from the {deployment.work_queue_name!r} work queue:\"\n                )\n                app.console.print(\n                    f\"$ prefect agent start -q {deployment.work_queue_name!r}\",\n                    style=\"blue\",\n                )\n            else:\n                app.console.print(\n                    (\n                        \"\\nThis deployment does not specify a work queue name, which\"\n                        \" means agents will not be able to pick up its runs. To add a\"\n                        \" work queue, edit the deployment spec and re-run this command,\"\n                        \" or visit the deployment in the UI.\"\n                    ),\n                    style=\"red\",\n                )\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.delete","title":"delete async","text":"

    Delete a deployment.

    \b Examples: \b $ prefect deployment delete test_flow/test_deployment $ prefect deployment delete --id dfd3e220-a130-4149-9af6-8d487e02fea6

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def delete(\n    name: Optional[str] = typer.Argument(\n        None, help=\"A deployed flow's name: <FLOW_NAME>/<DEPLOYMENT_NAME>\"\n    ),\n    deployment_id: Optional[str] = typer.Option(\n        None, \"--id\", help=\"A deployment id to search for if no name is given\"\n    ),\n):\n    \"\"\"\n    Delete a deployment.\n\n    \\b\n    Examples:\n        \\b\n        $ prefect deployment delete test_flow/test_deployment\n        $ prefect deployment delete --id dfd3e220-a130-4149-9af6-8d487e02fea6\n    \"\"\"\n    async with get_client() as client:\n        if name is None and deployment_id is not None:\n            try:\n                await client.delete_deployment(deployment_id)\n                exit_with_success(f\"Deleted deployment '{deployment_id}'.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {deployment_id!r} not found!\")\n        elif name is not None:\n            try:\n                deployment = await client.read_deployment_by_name(name)\n                await client.delete_deployment(deployment.id)\n                exit_with_success(f\"Deleted deployment '{name}'.\")\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {name!r} not found!\")\n        else:\n            exit_with_error(\"Must provide a deployment name or id\")\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.inspect","title":"inspect async","text":"

    View details about a deployment.

    \b Example: \b $ prefect deployment inspect \"hello-world/my-deployment\" { 'id': '610df9c3-0fb4-4856-b330-67f588d20201', 'created': '2022-08-01T18:36:25.192102+00:00', 'updated': '2022-08-01T18:36:25.188166+00:00', 'name': 'my-deployment', 'description': None, 'flow_id': 'b57b0aa2-ef3a-479e-be49-381fb0483b4e', 'schedule': None, 'is_schedule_active': True, 'parameters': {'name': 'Marvin'}, 'tags': ['test'], 'parameter_openapi_schema': { 'title': 'Parameters', 'type': 'object', 'properties': { 'name': { 'title': 'name', 'type': 'string' } }, 'required': ['name'] }, 'storage_document_id': '63ef008f-1e5d-4e07-a0d4-4535731adb32', 'infrastructure_document_id': '6702c598-7094-42c8-9785-338d2ec3a028', 'infrastructure': { 'type': 'process', 'env': {}, 'labels': {}, 'name': None, 'command': ['python', '-m', 'prefect.engine'], 'stream_output': True } }

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def inspect(name: str):\n    \"\"\"\n    View details about a deployment.\n\n    \\b\n    Example:\n        \\b\n        $ prefect deployment inspect \"hello-world/my-deployment\"\n        {\n            'id': '610df9c3-0fb4-4856-b330-67f588d20201',\n            'created': '2022-08-01T18:36:25.192102+00:00',\n            'updated': '2022-08-01T18:36:25.188166+00:00',\n            'name': 'my-deployment',\n            'description': None,\n            'flow_id': 'b57b0aa2-ef3a-479e-be49-381fb0483b4e',\n            'schedule': None,\n            'is_schedule_active': True,\n            'parameters': {'name': 'Marvin'},\n            'tags': ['test'],\n            'parameter_openapi_schema': {\n                'title': 'Parameters',\n                'type': 'object',\n                'properties': {\n                    'name': {\n                        'title': 'name',\n                        'type': 'string'\n                    }\n                },\n                'required': ['name']\n            },\n            'storage_document_id': '63ef008f-1e5d-4e07-a0d4-4535731adb32',\n            'infrastructure_document_id': '6702c598-7094-42c8-9785-338d2ec3a028',\n            'infrastructure': {\n                'type': 'process',\n                'env': {},\n                'labels': {},\n                'name': None,\n                'command': ['python', '-m', 'prefect.engine'],\n                'stream_output': True\n            }\n        }\n\n    \"\"\"\n    assert_deployment_name_format(name)\n\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(name)\n        except ObjectNotFound:\n            exit_with_error(f\"Deployment {name!r} not found!\")\n\n        deployment_json = deployment.dict(json_compatible=True)\n\n        if deployment.infrastructure_document_id:\n            deployment_json[\"infrastructure\"] = Block._from_block_document(\n                await client.read_block_document(deployment.infrastructure_document_id)\n            ).dict(\n                exclude={\"_block_document_id\", \"_block_document_name\", \"_is_anonymous\"}\n            )\n\n        if client.server_type == ServerType.CLOUD:\n            deployment_json[\"automations\"] = [\n                a.dict()\n                for a in await client.read_resource_related_automations(\n                    f\"prefect.deployment.{deployment.id}\"\n                )\n            ]\n\n    app.console.print(Pretty(deployment_json))\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.ls","title":"ls async","text":"

    View all deployments or deployments for specific flows.

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def ls(flow_name: List[str] = None, by_created: bool = False):\n    \"\"\"\n    View all deployments or deployments for specific flows.\n    \"\"\"\n    async with get_client() as client:\n        deployments = await client.read_deployments(\n            flow_filter=FlowFilter(name={\"any_\": flow_name}) if flow_name else None\n        )\n        flows = {\n            flow.id: flow\n            for flow in await client.read_flows(\n                flow_filter=FlowFilter(id={\"any_\": [d.flow_id for d in deployments]})\n            )\n        }\n\n    def sort_by_name_keys(d):\n        return flows[d.flow_id].name, d.name\n\n    def sort_by_created_key(d):\n        return pendulum.now(\"utc\") - d.created\n\n    table = Table(\n        title=\"Deployments\",\n    )\n    table.add_column(\"Name\", style=\"blue\", no_wrap=True)\n    table.add_column(\"ID\", style=\"cyan\", no_wrap=True)\n\n    for deployment in sorted(\n        deployments, key=sort_by_created_key if by_created else sort_by_name_keys\n    ):\n        table.add_row(\n            f\"{flows[deployment.flow_id].name}/[bold]{deployment.name}[/]\",\n            str(deployment.id),\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.pause_schedule","title":"pause_schedule async","text":"

    Pause schedule of a given deployment.

    Source code in prefect/cli/deployment.py
    @deployment_app.command(\"pause-schedule\")\nasync def pause_schedule(\n    name: str,\n):\n    \"\"\"\n    Pause schedule of a given deployment.\n    \"\"\"\n    assert_deployment_name_format(name)\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(name)\n        except ObjectNotFound:\n            exit_with_error(f\"Deployment {name!r} not found!\")\n\n        await client.update_deployment(deployment, is_schedule_active=False)\n        exit_with_success(f\"Paused schedule for deployment {name}\")\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.resume_schedule","title":"resume_schedule async","text":"

    Resume schedule of a given deployment.

    Source code in prefect/cli/deployment.py
    @deployment_app.command(\"resume-schedule\")\nasync def resume_schedule(\n    name: str,\n):\n    \"\"\"\n    Resume schedule of a given deployment.\n    \"\"\"\n    assert_deployment_name_format(name)\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(name)\n        except ObjectNotFound:\n            exit_with_error(f\"Deployment {name!r} not found!\")\n\n        await client.update_deployment(deployment, is_schedule_active=True)\n        exit_with_success(f\"Resumed schedule for deployment {name}\")\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.run","title":"run async","text":"

    Create a flow run for the given flow and deployment.

    The flow run will be scheduled to run immediately unless --start-in or --start-at is specified. The flow run will not execute until an agent starts.

    Source code in prefect/cli/deployment.py
    @deployment_app.command()\nasync def run(\n    name: Optional[str] = typer.Argument(\n        None, help=\"A deployed flow's name: <FLOW_NAME>/<DEPLOYMENT_NAME>\"\n    ),\n    deployment_id: Optional[str] = typer.Option(\n        None, \"--id\", help=\"A deployment id to search for if no name is given\"\n    ),\n    params: List[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--param\",\n        help=(\n            \"A key, value pair (key=value) specifying a flow parameter. The value will\"\n            \" be interpreted as JSON. May be passed multiple times to specify multiple\"\n            \" parameter values.\"\n        ),\n    ),\n    multiparams: Optional[str] = typer.Option(\n        None,\n        \"--params\",\n        help=(\n            \"A mapping of parameters to values. To use a stdin, pass '-'. Any \"\n            \"parameters passed with `--param` will take precedence over these values.\"\n        ),\n    ),\n    start_in: Optional[str] = typer.Option(\n        None,\n        \"--start-in\",\n    ),\n    start_at: Optional[str] = typer.Option(\n        None,\n        \"--start-at\",\n    ),\n):\n    \"\"\"\n    Create a flow run for the given flow and deployment.\n\n    The flow run will be scheduled to run immediately unless `--start-in` or `--start-at` is specified.\n    The flow run will not execute until an agent starts.\n    \"\"\"\n    import dateparser\n\n    now = pendulum.now(\"UTC\")\n\n    multi_params = {}\n    if multiparams:\n        if multiparams == \"-\":\n            multiparams = sys.stdin.read()\n            if not multiparams:\n                exit_with_error(\"No data passed to stdin\")\n\n        try:\n            multi_params = json.loads(multiparams)\n        except ValueError as exc:\n            exit_with_error(f\"Failed to parse JSON: {exc}\")\n\n    cli_params = _load_json_key_values(params, \"parameter\")\n    conflicting_keys = set(cli_params.keys()).intersection(multi_params.keys())\n    if conflicting_keys:\n        app.console.print(\n            \"The following parameters were specified by `--param` and `--params`, the \"\n            f\"`--param` value will be used: {conflicting_keys}\"\n        )\n    parameters = {**multi_params, **cli_params}\n\n    if start_in and start_at:\n        exit_with_error(\n            \"Only one of `--start-in` or `--start-at` can be set, not both.\"\n        )\n\n    elif start_in is None and start_at is None:\n        scheduled_start_time = now\n        human_dt_diff = \" (now)\"\n    else:\n        if start_in:\n            start_time_raw = \"in \" + start_in\n        else:\n            start_time_raw = \"at \" + start_at\n        with warnings.catch_warnings():\n            # PyTZ throws a warning based on dateparser usage of the library\n            # See https://github.com/scrapinghub/dateparser/issues/1089\n            warnings.filterwarnings(\"ignore\", module=\"dateparser\")\n\n            try:\n                start_time_parsed = dateparser.parse(\n                    start_time_raw,\n                    settings={\n                        \"TO_TIMEZONE\": \"UTC\",\n                        \"RETURN_AS_TIMEZONE_AWARE\": False,\n                        \"PREFER_DATES_FROM\": \"future\",\n                        \"RELATIVE_BASE\": datetime.fromtimestamp(\n                            now.timestamp(), tz=pendulum.tz.UTC\n                        ),\n                    },\n                )\n\n            except Exception as exc:\n                exit_with_error(f\"Failed to parse '{start_time_raw!r}': {exc!s}\")\n\n        if start_time_parsed is None:\n            exit_with_error(f\"Unable to parse scheduled start time {start_time_raw!r}.\")\n\n        scheduled_start_time = pendulum.instance(start_time_parsed)\n        human_dt_diff = (\n            \" (\" + pendulum.format_diff(scheduled_start_time.diff(now)) + \")\"\n        )\n\n    async with get_client() as client:\n        deployment = await get_deployment(client, name, deployment_id)\n        flow = await client.read_flow(deployment.flow_id)\n\n        deployment_parameters = deployment.parameter_openapi_schema[\"properties\"].keys()\n        unknown_keys = set(parameters.keys()).difference(deployment_parameters)\n        if unknown_keys:\n            available_parameters = (\n                (\n                    \"The following parameters are available on the deployment: \"\n                    + listrepr(deployment_parameters, sep=\", \")\n                )\n                if deployment_parameters\n                else \"This deployment does not accept parameters.\"\n            )\n\n            exit_with_error(\n                \"The following parameters were specified but not found on the \"\n                f\"deployment: {listrepr(unknown_keys, sep=', ')}\"\n                f\"\\n{available_parameters}\"\n            )\n\n        app.console.print(\n            f\"Creating flow run for deployment '{flow.name}/{deployment.name}'...\",\n        )\n\n        try:\n            flow_run = await client.create_flow_run_from_deployment(\n                deployment.id,\n                parameters=parameters,\n                state=Scheduled(scheduled_time=scheduled_start_time),\n            )\n        except PrefectHTTPStatusError as exc:\n            detail = exc.response.json().get(\"detail\")\n            if detail:\n                exit_with_error(\n                    exc.response.json()[\"detail\"],\n                )\n            else:\n                raise\n\n    if PREFECT_UI_URL:\n        run_url = f\"{PREFECT_UI_URL.value()}/flow-runs/flow-run/{flow_run.id}\"\n    else:\n        run_url = \"<no dashboard available>\"\n\n    datetime_local_tz = scheduled_start_time.in_tz(pendulum.tz.local_timezone())\n    scheduled_display = (\n        datetime_local_tz.to_datetime_string()\n        + \" \"\n        + datetime_local_tz.tzname()\n        + human_dt_diff\n    )\n\n    app.console.print(f\"Created flow run {flow_run.name!r}.\")\n    app.console.print(\n        textwrap.dedent(\n            f\"\"\"\n        \u2514\u2500\u2500 UUID: {flow_run.id}\n        \u2514\u2500\u2500 Parameters: {flow_run.parameters}\n        \u2514\u2500\u2500 Scheduled start time: {scheduled_display}\n        \u2514\u2500\u2500 URL: {run_url}\n        \"\"\"\n        ).strip()\n    )\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.set_schedule","title":"set_schedule async","text":"

    Set schedule for a given deployment.

    Source code in prefect/cli/deployment.py
    @deployment_app.command(\"set-schedule\")\nasync def set_schedule(\n    name: str,\n    interval: Optional[float] = typer.Option(\n        None,\n        \"--interval\",\n        help=\"An interval to schedule on, specified in seconds\",\n        min=0.0001,\n    ),\n    interval_anchor: Optional[str] = typer.Option(\n        None,\n        \"--anchor-date\",\n        help=\"The anchor date for an interval schedule\",\n    ),\n    rrule_string: Optional[str] = typer.Option(\n        None, \"--rrule\", help=\"Deployment schedule rrule string\"\n    ),\n    cron_string: Optional[str] = typer.Option(\n        None, \"--cron\", help=\"Deployment schedule cron string\"\n    ),\n    cron_day_or: Optional[str] = typer.Option(\n        None,\n        \"--day_or\",\n        help=\"Control how croniter handles `day` and `day_of_week` entries\",\n    ),\n    timezone: Optional[str] = typer.Option(\n        None,\n        \"--timezone\",\n        help=\"Deployment schedule timezone string e.g. 'America/New_York'\",\n    ),\n    no_schedule: bool = typer.Option(\n        False,\n        \"--no-schedule\",\n        help=\"An optional flag to disable scheduling for this deployment.\",\n    ),\n):\n    \"\"\"\n    Set schedule for a given deployment.\n    \"\"\"\n    assert_deployment_name_format(name)\n\n    if (\n        sum(option is not None for option in [interval, rrule_string, cron_string])\n        + (1 if no_schedule else 0)\n        != 1\n    ):\n        exit_with_error(\n            \"Exactly one of `--interval`, `--rrule`, `--cron` or `--no-schedule` must\"\n            \" be provided.\"\n        )\n\n    if interval_anchor and not interval:\n        exit_with_error(\"An anchor date can only be provided with an interval schedule\")\n\n    if interval is not None:\n        if interval_anchor:\n            try:\n                pendulum.parse(interval_anchor)\n            except ValueError:\n                exit_with_error(\"The anchor date must be a valid date string.\")\n        interval_schedule = {\n            \"interval\": interval,\n            \"anchor_date\": interval_anchor,\n            \"timezone\": timezone,\n        }\n        updated_schedule = IntervalSchedule(\n            **{k: v for k, v in interval_schedule.items() if v is not None}\n        )\n\n    if cron_string is not None:\n        cron_schedule = {\n            \"cron\": cron_string,\n            \"day_or\": cron_day_or,\n            \"timezone\": timezone,\n        }\n        updated_schedule = CronSchedule(\n            **{k: v for k, v in cron_schedule.items() if v is not None}\n        )\n\n    if rrule_string is not None:\n        # a timezone in the `rrule_string` gets ignored by the RRuleSchedule constructor\n        if \"TZID\" in rrule_string and not timezone:\n            exit_with_error(\n                \"You can provide a timezone by providing a dict with a `timezone` key\"\n                \" to the --rrule option. E.g. {'rrule': 'FREQ=MINUTELY;INTERVAL=5',\"\n                \" 'timezone': 'America/New_York'}.\\nAlternatively, you can provide a\"\n                \" timezone by passing in a --timezone argument.\"\n            )\n        try:\n            updated_schedule = RRuleSchedule(**json.loads(rrule_string))\n            if timezone:\n                # override timezone if specified via CLI argument\n                updated_schedule.timezone = timezone\n        except json.JSONDecodeError:\n            updated_schedule = RRuleSchedule(rrule=rrule_string, timezone=timezone)\n\n    if no_schedule:\n        updated_schedule = NoSchedule()\n\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(name)\n        except ObjectNotFound:\n            exit_with_error(f\"Deployment {name!r} not found!\")\n\n        await client.update_deployment(deployment, schedule=updated_schedule)\n        exit_with_success(\"Updated deployment schedule!\")\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/deployment/#prefect.cli.deployment.str_presenter","title":"str_presenter","text":"

    configures yaml for dumping multiline strings Ref: https://stackoverflow.com/questions/8640959/how-can-i-control-what-scalar-form-pyyaml-uses-for-my-data

    Source code in prefect/cli/deployment.py
    def str_presenter(dumper, data):\n    \"\"\"\n    configures yaml for dumping multiline strings\n    Ref: https://stackoverflow.com/questions/8640959/how-can-i-control-what-scalar-form-pyyaml-uses-for-my-data\n    \"\"\"\n    if len(data.splitlines()) > 1:  # check for multiline string\n        return dumper.represent_scalar(\"tag:yaml.org,2002:str\", data, style=\"|\")\n    return dumper.represent_scalar(\"tag:yaml.org,2002:str\", data)\n
    ","tags":["Python API","CLI","deployments"]},{"location":"api-ref/prefect/cli/dev/","title":"dev","text":"","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev","title":"prefect.cli.dev","text":"

    Command line interface for working with Prefect Server

    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.agent","title":"agent async","text":"

    Starts a hot-reloading development agent process.

    Source code in prefect/cli/dev.py
    @dev_app.command()\nasync def agent(\n    api_url: str = SettingsOption(PREFECT_API_URL),\n    work_queues: List[str] = typer.Option(\n        [\"default\"],\n        \"-q\",\n        \"--work-queue\",\n        help=\"One or more work queue names for the agent to pull from.\",\n    ),\n):\n    \"\"\"\n    Starts a hot-reloading development agent process.\n    \"\"\"\n    # Delayed import since this is only a 'dev' dependency\n    import watchfiles\n\n    app.console.print(\"Creating hot-reloading agent process...\")\n\n    try:\n        await watchfiles.arun_process(\n            prefect.__module_path__,\n            target=agent_process_entrypoint,\n            kwargs=dict(api=api_url, work_queues=work_queues),\n        )\n    except RuntimeError as err:\n        # a bug in watchfiles causes an 'Already borrowed' error from Rust when\n        # exiting: https://github.com/samuelcolvin/watchfiles/issues/200\n        if str(err).strip() != \"Already borrowed\":\n            raise\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.agent_process_entrypoint","title":"agent_process_entrypoint","text":"

    An entrypoint for starting an agent in a subprocess. Adds a Rich console to the Typer app, processes Typer default parameters, then starts an agent. All kwargs are forwarded to prefect.cli.agent.start.

    Source code in prefect/cli/dev.py
    def agent_process_entrypoint(**kwargs):\n    \"\"\"\n    An entrypoint for starting an agent in a subprocess. Adds a Rich console\n    to the Typer app, processes Typer default parameters, then starts an agent.\n    All kwargs are forwarded to  `prefect.cli.agent.start`.\n    \"\"\"\n    import inspect\n\n    import rich\n\n    # import locally so only the `dev` command breaks if Typer internals change\n    from typer.models import ParameterInfo\n\n    # Typer does not process default parameters when calling a function\n    # directly, so we must set `start_agent`'s default parameters manually.\n    # get the signature of the `start_agent` function\n    start_agent_signature = inspect.signature(start_agent)\n\n    # for any arguments not present in kwargs, use the default value.\n    for name, param in start_agent_signature.parameters.items():\n        if name not in kwargs:\n            # All `param.default` values for start_agent are Typer params that store the\n            # actual default value in their `default` attribute and we must call\n            # `param.default.default` to get the actual default value. We should also\n            # ensure we extract the right default if non-Typer defaults are added\n            # to `start_agent` in the future.\n            if isinstance(param.default, ParameterInfo):\n                default = param.default.default\n            else:\n                default = param.default\n\n            # Some defaults are Prefect `SettingsOption.value` methods\n            # that must be called to get the actual value.\n            kwargs[name] = default() if callable(default) else default\n\n    # add a console, because calling the agent start function directly\n    # instead of via CLI call means `app` has no `console` attached.\n    app.console = (\n        rich.console.Console(\n            highlight=False,\n            color_system=\"auto\" if PREFECT_CLI_COLORS else None,\n            soft_wrap=not PREFECT_CLI_WRAP_LINES.value(),\n        )\n        if not getattr(app, \"console\", None)\n        else app.console\n    )\n\n    try:\n        start_agent(**kwargs)  # type: ignore\n    except KeyboardInterrupt:\n        # expected when watchfiles kills the process\n        pass\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.api","title":"api async","text":"

    Starts a hot-reloading development API.

    Source code in prefect/cli/dev.py
    @dev_app.command()\nasync def api(\n    host: str = SettingsOption(PREFECT_SERVER_API_HOST),\n    port: int = SettingsOption(PREFECT_SERVER_API_PORT),\n    log_level: str = \"DEBUG\",\n    services: bool = True,\n):\n    \"\"\"\n    Starts a hot-reloading development API.\n    \"\"\"\n    import watchfiles\n\n    server_env = os.environ.copy()\n    server_env[\"PREFECT_API_SERVICES_RUN_IN_APP\"] = str(services)\n    server_env[\"PREFECT_API_SERVICES_UI\"] = \"False\"\n    server_env[\"PREFECT_UI_API_URL\"] = f\"http://{host}:{port}/api\"\n\n    command = [\n        sys.executable,\n        \"-m\",\n        \"uvicorn\",\n        \"--factory\",\n        \"prefect.server.api.server:create_app\",\n        \"--host\",\n        str(host),\n        \"--port\",\n        str(port),\n        \"--log-level\",\n        log_level.lower(),\n    ]\n\n    app.console.print(f\"Running: {' '.join(command)}\")\n    import signal\n\n    stop_event = anyio.Event()\n    start_command = partial(\n        run_process, command=command, env=server_env, stream_output=True\n    )\n\n    async with anyio.create_task_group() as tg:\n        try:\n            server_pid = await tg.start(start_command)\n            async for _ in watchfiles.awatch(\n                prefect.__module_path__, stop_event=stop_event  # type: ignore\n            ):\n                # when any watched files change, restart the server\n                app.console.print(\"Restarting Prefect Server...\")\n                os.kill(server_pid, signal.SIGTERM)  # type: ignore\n                # start a new server\n                server_pid = await tg.start(start_command)\n        except RuntimeError as err:\n            # a bug in watchfiles causes an 'Already borrowed' error from Rust when\n            # exiting: https://github.com/samuelcolvin/watchfiles/issues/200\n            if str(err).strip() != \"Already borrowed\":\n                raise\n        except KeyboardInterrupt:\n            # exit cleanly on ctrl-c by killing the server process if it's\n            # still running\n            try:\n                os.kill(server_pid, signal.SIGTERM)  # type: ignore\n            except ProcessLookupError:\n                # process already exited\n                pass\n\n            stop_event.set()\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.build_docs","title":"build_docs","text":"

    Builds REST API reference documentation for static display.

    Source code in prefect/cli/dev.py
    @dev_app.command()\ndef build_docs(\n    schema_path: str = None,\n):\n    \"\"\"\n    Builds REST API reference documentation for static display.\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n\n    from prefect.server.api.server import create_app\n\n    schema = create_app(ephemeral=True).openapi()\n\n    if not schema_path:\n        schema_path = (\n            prefect.__development_base_path__ / \"docs\" / \"api-ref\" / \"schema.json\"\n        ).absolute()\n    # overwrite info for display purposes\n    schema[\"info\"] = {}\n    with open(schema_path, \"w\") as f:\n        json.dump(schema, f)\n    app.console.print(f\"OpenAPI schema written to {schema_path}\")\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.build_image","title":"build_image","text":"

    Build a docker image for development.

    Source code in prefect/cli/dev.py
    @dev_app.command()\ndef build_image(\n    arch: str = typer.Option(\n        None,\n        help=(\n            \"The architecture to build the container for. \"\n            \"Defaults to the architecture of the host Python. \"\n            f\"[default: {platform.machine()}]\"\n        ),\n    ),\n    python_version: str = typer.Option(\n        None,\n        help=(\n            \"The Python version to build the container for. \"\n            \"Defaults to the version of the host Python. \"\n            f\"[default: {python_version_minor()}]\"\n        ),\n    ),\n    flavor: str = typer.Option(\n        None,\n        help=(\n            \"An alternative flavor to build, for example 'conda'. \"\n            \"Defaults to the standard Python base image\"\n        ),\n    ),\n    dry_run: bool = False,\n):\n    \"\"\"\n    Build a docker image for development.\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n    # TODO: Once https://github.com/tiangolo/typer/issues/354 is addressed, the\n    #       default can be set in the function signature\n    arch = arch or platform.machine()\n    python_version = python_version or python_version_minor()\n\n    tag = get_prefect_image_name(python_version=python_version, flavor=flavor)\n\n    # Here we use a subprocess instead of the docker-py client to easily stream output\n    # as it comes\n    command = [\n        \"docker\",\n        \"build\",\n        str(prefect.__development_base_path__),\n        \"--tag\",\n        tag,\n        \"--platform\",\n        f\"linux/{arch}\",\n        \"--build-arg\",\n        \"PREFECT_EXTRAS=[dev]\",\n        \"--build-arg\",\n        f\"PYTHON_VERSION={python_version}\",\n    ]\n\n    if flavor:\n        command += [\"--build-arg\", f\"BASE_IMAGE=prefect-{flavor}\"]\n\n    if dry_run:\n        print(\" \".join(command))\n        return\n\n    try:\n        subprocess.check_call(command, shell=sys.platform == \"win32\")\n    except subprocess.CalledProcessError:\n        exit_with_error(\"Failed to build image!\")\n    else:\n        exit_with_success(f\"Built image {tag!r} for linux/{arch}\")\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.container","title":"container","text":"

    Run a docker container with local code mounted and installed.

    Source code in prefect/cli/dev.py
    @dev_app.command()\ndef container(bg: bool = False, name=\"prefect-dev\", api: bool = True, tag: str = None):\n    \"\"\"\n    Run a docker container with local code mounted and installed.\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n    import docker\n    from docker.models.containers import Container\n\n    client = docker.from_env()\n\n    containers = client.containers.list()\n    container_names = {container.name for container in containers}\n    if name in container_names:\n        exit_with_error(\n            f\"Container {name!r} already exists. Specify a different name or stop \"\n            \"the existing container.\"\n        )\n\n    blocking_cmd = \"prefect dev api\" if api else \"sleep infinity\"\n    tag = tag or get_prefect_image_name()\n\n    container: Container = client.containers.create(\n        image=tag,\n        command=[\n            \"/bin/bash\",\n            \"-c\",\n            (  # noqa\n                \"pip install -e /opt/prefect/repo\\\\[dev\\\\] && touch /READY &&\"\n                f\" {blocking_cmd}\"\n            ),\n        ],\n        name=name,\n        auto_remove=True,\n        working_dir=\"/opt/prefect/repo\",\n        volumes=[f\"{prefect.__development_base_path__}:/opt/prefect/repo\"],\n        shm_size=\"4G\",\n    )\n\n    print(f\"Starting container for image {tag!r}...\")\n    container.start()\n\n    print(\"Waiting for installation to complete\", end=\"\", flush=True)\n    try:\n        ready = False\n        while not ready:\n            print(\".\", end=\"\", flush=True)\n            result = container.exec_run(\"test -f /READY\")\n            ready = result.exit_code == 0\n            if not ready:\n                time.sleep(3)\n    except BaseException:\n        print(\"\\nInterrupted. Stopping container...\")\n        container.stop()\n        raise\n\n    print(\n        textwrap.dedent(\n            f\"\"\"\n            Container {container.name!r} is ready! To connect to the container, run:\n\n                docker exec -it {container.name} /bin/bash\n            \"\"\"\n        )\n    )\n\n    if bg:\n        print(\n            textwrap.dedent(\n                f\"\"\"\n                The container will run forever. Stop the container with:\n\n                    docker stop {container.name}\n                \"\"\"\n            )\n        )\n        # Exit without stopping\n        return\n\n    try:\n        print(\"Send a keyboard interrupt to exit...\")\n        container.wait()\n    except KeyboardInterrupt:\n        pass  # Avoid showing \"Abort\"\n    finally:\n        print(\"\\nStopping container...\")\n        container.stop()\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.kubernetes_manifest","title":"kubernetes_manifest","text":"

    Generates a Kubernetes manifest for development.

    Example

    $ prefect dev kubernetes-manifest | kubectl apply -f -

    Source code in prefect/cli/dev.py
    @dev_app.command()\ndef kubernetes_manifest():\n    \"\"\"\n    Generates a Kubernetes manifest for development.\n\n    Example:\n        $ prefect dev kubernetes-manifest | kubectl apply -f -\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n\n    template = Template(\n        (\n            prefect.__module_path__ / \"cli\" / \"templates\" / \"kubernetes-dev.yaml\"\n        ).read_text()\n    )\n    manifest = template.substitute(\n        {\n            \"prefect_root_directory\": prefect.__development_base_path__,\n            \"image_name\": get_prefect_image_name(),\n        }\n    )\n    print(manifest)\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.start","title":"start async","text":"

    Starts a hot-reloading development server with API, UI, and agent processes.

    Each service has an individual command if you wish to start them separately. Each service can be excluded here as well.

    Source code in prefect/cli/dev.py
    @dev_app.command()\nasync def start(\n    exclude_api: bool = typer.Option(False, \"--no-api\"),\n    exclude_ui: bool = typer.Option(False, \"--no-ui\"),\n    exclude_agent: bool = typer.Option(False, \"--no-agent\"),\n    work_queues: List[str] = typer.Option(\n        [\"default\"],\n        \"-q\",\n        \"--work-queue\",\n        help=\"One or more work queue names for the dev agent to pull from.\",\n    ),\n):\n    \"\"\"\n    Starts a hot-reloading development server with API, UI, and agent processes.\n\n    Each service has an individual command if you wish to start them separately.\n    Each service can be excluded here as well.\n    \"\"\"\n    async with anyio.create_task_group() as tg:\n        if not exclude_api:\n            tg.start_soon(\n                partial(\n                    api,\n                    host=PREFECT_SERVER_API_HOST.value(),\n                    port=PREFECT_SERVER_API_PORT.value(),\n                )\n            )\n        if not exclude_ui:\n            tg.start_soon(ui)\n        if not exclude_agent:\n            # Hook the agent to the hosted API if running\n            if not exclude_api:\n                host = f\"http://{PREFECT_SERVER_API_HOST.value()}:{PREFECT_SERVER_API_PORT.value()}/api\"  # noqa\n            else:\n                host = PREFECT_API_URL.value()\n            tg.start_soon(agent, host, work_queues)\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/dev/#prefect.cli.dev.ui","title":"ui async","text":"

    Starts a hot-reloading development UI.

    Source code in prefect/cli/dev.py
    @dev_app.command()\nasync def ui():\n    \"\"\"\n    Starts a hot-reloading development UI.\n    \"\"\"\n    exit_with_error_if_not_editable_install()\n    with tmpchdir(prefect.__development_base_path__):\n        with tmpchdir(prefect.__development_base_path__ / \"ui\"):\n            app.console.print(\"Installing npm packages...\")\n            await run_process([\"npm\", \"install\"], stream_output=True)\n\n            app.console.print(\"Starting UI development server...\")\n            await run_process(command=[\"npm\", \"run\", \"serve\"], stream_output=True)\n
    ","tags":["Python API","CLI","development"]},{"location":"api-ref/prefect/cli/flow/","title":"flow","text":"","tags":["Python API","flows","CLI"]},{"location":"api-ref/prefect/cli/flow/#prefect.cli.flow","title":"prefect.cli.flow","text":"

    Command line interface for working with flows.

    ","tags":["Python API","flows","CLI"]},{"location":"api-ref/prefect/cli/flow/#prefect.cli.flow.ls","title":"ls async","text":"

    View flows.

    Source code in prefect/cli/flow.py
    @flow_app.command()\nasync def ls(\n    limit: int = 15,\n):\n    \"\"\"\n    View flows.\n    \"\"\"\n    async with get_client() as client:\n        flows = await client.read_flows(\n            limit=limit,\n            sort=FlowSort.CREATED_DESC,\n        )\n\n    table = Table(title=\"Flows\")\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Created\", no_wrap=True)\n\n    for flow in flows:\n        table.add_row(\n            str(flow.id),\n            str(flow.name),\n            str(flow.created),\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","flows","CLI"]},{"location":"api-ref/prefect/cli/flow/#prefect.cli.flow.serve","title":"serve async","text":"

    Serve a flow via an entrypoint.

    Source code in prefect/cli/flow.py
    @flow_app.command()\nasync def serve(\n    entrypoint: str = typer.Argument(\n        ...,\n        help=(\n            \"The path to a file containing a flow and the name of the flow function in\"\n            \" the format `./path/to/file.py:flow_func_name`.\"\n        ),\n    ),\n    name: str = typer.Option(\n        ...,\n        \"--name\",\n        \"-n\",\n        help=\"The name to give the deployment created for the flow.\",\n    ),\n    description: Optional[str] = typer.Option(\n        None,\n        \"--description\",\n        \"-d\",\n        help=(\n            \"The description to give the created deployment. If not provided, the\"\n            \" description will be populated from the flow's description.\"\n        ),\n    ),\n    version: Optional[str] = typer.Option(\n        None, \"-v\", \"--version\", help=\"A version to give the created deployment.\"\n    ),\n    tags: Optional[List[str]] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=\"One or more optional tags to apply to the created deployment.\",\n    ),\n    cron: Optional[str] = typer.Option(\n        None,\n        \"--cron\",\n        help=(\n            \"A cron string that will be used to set a schedule for the created\"\n            \" deployment.\"\n        ),\n    ),\n    interval: Optional[int] = typer.Option(\n        None,\n        \"--interval\",\n        help=(\n            \"An integer specifying an interval (in seconds) between scheduled runs of\"\n            \" the flow.\"\n        ),\n    ),\n    interval_anchor: Optional[str] = typer.Option(\n        None, \"--anchor-date\", help=\"The start date for an interval schedule.\"\n    ),\n    rrule: Optional[str] = typer.Option(\n        None,\n        \"--rrule\",\n        help=\"An RRule that will be used to set a schedule for the created deployment.\",\n    ),\n    timezone: Optional[str] = typer.Option(\n        None,\n        \"--timezone\",\n        help=\"Timezone to used scheduling flow runs e.g. 'America/New_York'\",\n    ),\n    pause_on_shutdown: bool = typer.Option(\n        True,\n        help=(\n            \"If set, provided schedule will be paused when the serve command is\"\n            \" stopped. If not set, the schedules will continue running.\"\n        ),\n    ),\n):\n    \"\"\"\n    Serve a flow via an entrypoint.\n    \"\"\"\n    runner = Runner(name=name, pause_on_shutdown=pause_on_shutdown)\n    try:\n        schedule = None\n        if interval or cron or rrule:\n            schedule = construct_schedule(\n                interval=interval,\n                cron=cron,\n                rrule=rrule,\n                timezone=timezone,\n                anchor_date=interval_anchor,\n            )\n        runner_deployment = RunnerDeployment.from_entrypoint(\n            entrypoint=entrypoint,\n            name=name,\n            schedule=schedule,\n            description=description,\n            tags=tags or [],\n            version=version,\n        )\n    except (MissingFlowError, ValueError) as exc:\n        exit_with_error(str(exc))\n    deployment_id = await runner.add_deployment(runner_deployment)\n\n    help_message = (\n        f\"[green]Your flow {runner_deployment.flow_name!r} is being served and polling\"\n        \" for scheduled runs!\\n[/]\\nTo trigger a run for this flow, use the following\"\n        \" command:\\n[blue]\\n\\t$ prefect deployment run\"\n        f\" '{runner_deployment.flow_name}/{name}'\\n[/]\"\n    )\n    if PREFECT_UI_URL:\n        help_message += (\n            \"\\nYou can also run your flow via the Prefect UI:\"\n            f\" [blue]{PREFECT_UI_URL.value()}/deployments/deployment/{deployment_id}[/]\\n\"\n        )\n    app.console.print(Panel(help_message))\n    await runner.start()\n
    ","tags":["Python API","flows","CLI"]},{"location":"api-ref/prefect/cli/flow_run/","title":"flow_run","text":"","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run","title":"prefect.cli.flow_run","text":"

    Command line interface for working with flow runs

    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.cancel","title":"cancel async","text":"

    Cancel a flow run by ID.

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def cancel(id: UUID):\n    \"\"\"Cancel a flow run by ID.\"\"\"\n    async with get_client() as client:\n        cancelling_state = State(type=StateType.CANCELLING)\n        try:\n            result = await client.set_flow_run_state(\n                flow_run_id=id, state=cancelling_state\n            )\n        except ObjectNotFound:\n            exit_with_error(f\"Flow run '{id}' not found!\")\n\n    if result.status == SetStateStatus.ABORT:\n        exit_with_error(\n            f\"Flow run '{id}' was unable to be cancelled. Reason:\"\n            f\" '{result.details.reason}'\"\n        )\n\n    exit_with_success(f\"Flow run '{id}' was successfully scheduled for cancellation.\")\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.delete","title":"delete async","text":"

    Delete a flow run by ID.

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def delete(id: UUID):\n    \"\"\"\n    Delete a flow run by ID.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.delete_flow_run(id)\n        except ObjectNotFound:\n            exit_with_error(f\"Flow run '{id}' not found!\")\n\n    exit_with_success(f\"Successfully deleted flow run '{id}'.\")\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.inspect","title":"inspect async","text":"

    View details about a flow run.

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def inspect(id: UUID):\n    \"\"\"\n    View details about a flow run.\n    \"\"\"\n    async with get_client() as client:\n        try:\n            flow_run = await client.read_flow_run(id)\n        except httpx.HTTPStatusError as exc:\n            if exc.response.status_code == status.HTTP_404_NOT_FOUND:\n                exit_with_error(f\"Flow run {id!r} not found!\")\n            else:\n                raise\n\n    app.console.print(Pretty(flow_run))\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.logs","title":"logs async","text":"

    View logs for a flow run.

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def logs(\n    id: UUID,\n    head: bool = typer.Option(\n        False,\n        \"--head\",\n        \"-h\",\n        help=(\n            f\"Show the first {LOGS_WITH_LIMIT_FLAG_DEFAULT_NUM_LOGS} logs instead of\"\n            \" all logs.\"\n        ),\n    ),\n    num_logs: int = typer.Option(\n        None,\n        \"--num-logs\",\n        \"-n\",\n        help=(\n            \"Number of logs to show when using the --head or --tail flag. If None,\"\n            f\" defaults to {LOGS_WITH_LIMIT_FLAG_DEFAULT_NUM_LOGS}.\"\n        ),\n        min=1,\n    ),\n    reverse: bool = typer.Option(\n        False,\n        \"--reverse\",\n        \"-r\",\n        help=\"Reverse the logs order to print the most recent logs first\",\n    ),\n    tail: bool = typer.Option(\n        False,\n        \"--tail\",\n        \"-t\",\n        help=(\n            f\"Show the last {LOGS_WITH_LIMIT_FLAG_DEFAULT_NUM_LOGS} logs instead of\"\n            \" all logs.\"\n        ),\n    ),\n):\n    \"\"\"\n    View logs for a flow run.\n    \"\"\"\n    # Pagination - API returns max 200 (LOGS_DEFAULT_PAGE_SIZE) logs at a time\n    offset = 0\n    more_logs = True\n    num_logs_returned = 0\n\n    # if head and tail flags are being used together\n    if head and tail:\n        exit_with_error(\"Please provide either a `head` or `tail` option but not both.\")\n\n    user_specified_num_logs = (\n        num_logs or LOGS_WITH_LIMIT_FLAG_DEFAULT_NUM_LOGS\n        if head or tail or num_logs\n        else None\n    )\n\n    # if using tail update offset according to LOGS_DEFAULT_PAGE_SIZE\n    if tail:\n        offset = max(0, user_specified_num_logs - LOGS_DEFAULT_PAGE_SIZE)\n\n    log_filter = LogFilter(flow_run_id={\"any_\": [id]})\n\n    async with get_client() as client:\n        # Get the flow run\n        try:\n            flow_run = await client.read_flow_run(id)\n        except ObjectNotFound:\n            exit_with_error(f\"Flow run {str(id)!r} not found!\")\n\n        while more_logs:\n            num_logs_to_return_from_page = (\n                LOGS_DEFAULT_PAGE_SIZE\n                if user_specified_num_logs is None\n                else min(\n                    LOGS_DEFAULT_PAGE_SIZE, user_specified_num_logs - num_logs_returned\n                )\n            )\n\n            # Get the next page of logs\n            page_logs = await client.read_logs(\n                log_filter=log_filter,\n                limit=num_logs_to_return_from_page,\n                offset=offset,\n                sort=(\n                    LogSort.TIMESTAMP_DESC if reverse or tail else LogSort.TIMESTAMP_ASC\n                ),\n            )\n\n            for log in reversed(page_logs) if tail and not reverse else page_logs:\n                app.console.print(\n                    # Print following the flow run format (declared in logging.yml)\n                    (\n                        f\"{pendulum.instance(log.timestamp).to_datetime_string()}.{log.timestamp.microsecond // 1000:03d} |\"\n                        f\" {logging.getLevelName(log.level):7s} | Flow run\"\n                        f\" {flow_run.name!r} - {log.message}\"\n                    ),\n                    soft_wrap=True,\n                )\n\n            # Update the number of logs retrieved\n            num_logs_returned += num_logs_to_return_from_page\n\n            if tail:\n                #  If the current offset is not 0, update the offset for the next page\n                if offset != 0:\n                    offset = (\n                        0\n                        # Reset the offset to 0 if there are less logs than the LOGS_DEFAULT_PAGE_SIZE to get the remaining log\n                        if offset < LOGS_DEFAULT_PAGE_SIZE\n                        else offset - LOGS_DEFAULT_PAGE_SIZE\n                    )\n                else:\n                    more_logs = False\n            else:\n                if len(page_logs) == LOGS_DEFAULT_PAGE_SIZE:\n                    offset += LOGS_DEFAULT_PAGE_SIZE\n                else:\n                    # No more logs to show, exit\n                    more_logs = False\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/flow_run/#prefect.cli.flow_run.ls","title":"ls async","text":"

    View recent flow runs or flow runs for specific flows

    Source code in prefect/cli/flow_run.py
    @flow_run_app.command()\nasync def ls(\n    flow_name: List[str] = typer.Option(None, help=\"Name of the flow\"),\n    limit: int = typer.Option(15, help=\"Maximum number of flow runs to list\"),\n    state: List[str] = typer.Option(None, help=\"Name of the flow run's state\"),\n    state_type: List[StateType] = typer.Option(\n        None, help=\"Type of the flow run's state\"\n    ),\n):\n    \"\"\"\n    View recent flow runs or flow runs for specific flows\n    \"\"\"\n\n    state_filter = {}\n    if state:\n        state_filter[\"name\"] = {\"any_\": state}\n    if state_type:\n        state_filter[\"type\"] = {\"any_\": state_type}\n\n    async with get_client() as client:\n        flow_runs = await client.read_flow_runs(\n            flow_filter=FlowFilter(name={\"any_\": flow_name}) if flow_name else None,\n            flow_run_filter=FlowRunFilter(state=state_filter) if state_filter else None,\n            limit=limit,\n            sort=FlowRunSort.EXPECTED_START_TIME_DESC,\n        )\n        flows_by_id = {\n            flow.id: flow\n            for flow in await client.read_flows(\n                flow_filter=FlowFilter(id={\"any_\": [run.flow_id for run in flow_runs]})\n            )\n        }\n\n    table = Table(title=\"Flow Runs\")\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Flow\", style=\"blue\", no_wrap=True)\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"State\", no_wrap=True)\n    table.add_column(\"When\", style=\"bold\", no_wrap=True)\n\n    for flow_run in sorted(flow_runs, key=lambda d: d.created, reverse=True):\n        flow = flows_by_id[flow_run.flow_id]\n        timestamp = (\n            flow_run.state.state_details.scheduled_time\n            if flow_run.state.is_scheduled()\n            else flow_run.state.timestamp\n        )\n        table.add_row(\n            str(flow_run.id),\n            str(flow.name),\n            str(flow_run.name),\n            str(flow_run.state.type.value),\n            pendulum.instance(timestamp).diff_for_humans(),\n        )\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","flow runs"]},{"location":"api-ref/prefect/cli/kubernetes/","title":"kubernetes","text":"","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/kubernetes/#prefect.cli.kubernetes","title":"prefect.cli.kubernetes","text":"

    Command line interface for working with Prefect on Kubernetes

    ","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/kubernetes/#prefect.cli.kubernetes.manifest_agent","title":"manifest_agent","text":"

    Generates a manifest for deploying Agent on Kubernetes.

    Example

    $ prefect kubernetes manifest agent | kubectl apply -f -

    Source code in prefect/cli/kubernetes.py
    @manifest_app.command(\"agent\")\ndef manifest_agent(\n    api_url: str = SettingsOption(PREFECT_API_URL),\n    api_key: str = SettingsOption(PREFECT_API_KEY),\n    image_tag: str = typer.Option(\n        get_prefect_image_name(),\n        \"-i\",\n        \"--image-tag\",\n        help=\"The tag of a Docker image to use for the Agent.\",\n    ),\n    namespace: str = typer.Option(\n        \"default\",\n        \"-n\",\n        \"--namespace\",\n        help=\"A Kubernetes namespace to create agent in.\",\n    ),\n    work_queue: str = typer.Option(\n        \"kubernetes\",\n        \"-q\",\n        \"--work-queue\",\n        help=\"A work queue name for the agent to pull from.\",\n    ),\n):\n    \"\"\"\n    Generates a manifest for deploying Agent on Kubernetes.\n\n    Example:\n        $ prefect kubernetes manifest agent | kubectl apply -f -\n    \"\"\"\n\n    template = Template(\n        (\n            prefect.__module_path__ / \"cli\" / \"templates\" / \"kubernetes-agent.yaml\"\n        ).read_text()\n    )\n    manifest = template.substitute(\n        {\n            \"api_url\": api_url,\n            \"api_key\": api_key,\n            \"image_name\": image_tag,\n            \"namespace\": namespace,\n            \"work_queue\": work_queue,\n        }\n    )\n    print(manifest)\n
    ","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/kubernetes/#prefect.cli.kubernetes.manifest_flow_run_job","title":"manifest_flow_run_job async","text":"

    Prints the default KubernetesJob Job manifest.

    Use this file to fully customize your KubernetesJob deployments.

    \b Example: \b $ prefect kubernetes manifest flow-run-job

    \b Output, a YAML file: \b apiVersion: batch/v1 kind: Job ...

    Source code in prefect/cli/kubernetes.py
    @manifest_app.command(\"flow-run-job\")\nasync def manifest_flow_run_job():\n    \"\"\"\n    Prints the default KubernetesJob Job manifest.\n\n    Use this file to fully customize your `KubernetesJob` deployments.\n\n    \\b\n    Example:\n        \\b\n        $ prefect kubernetes manifest flow-run-job\n\n    \\b\n    Output, a YAML file:\n        \\b\n        apiVersion: batch/v1\n        kind: Job\n        ...\n    \"\"\"\n\n    KubernetesJob.base_job_manifest()\n\n    output = yaml.dump(KubernetesJob.base_job_manifest())\n\n    # add some commentary where appropriate\n    output = output.replace(\n        \"metadata:\\n  labels:\",\n        \"metadata:\\n  # labels are required, even if empty\\n  labels:\",\n    )\n    output = output.replace(\n        \"containers:\\n\",\n        \"containers:  # the first container is required\\n\",\n    )\n    output = output.replace(\n        \"env: []\\n\",\n        \"env: []  # env is required, even if empty\\n\",\n    )\n\n    print(output)\n
    ","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/kubernetes/#prefect.cli.kubernetes.manifest_server","title":"manifest_server","text":"

    Generates a manifest for deploying Prefect on Kubernetes.

    Example

    $ prefect kubernetes manifest server | kubectl apply -f -

    Source code in prefect/cli/kubernetes.py
    @manifest_app.command(\"server\")\ndef manifest_server(\n    image_tag: str = typer.Option(\n        get_prefect_image_name(),\n        \"-i\",\n        \"--image-tag\",\n        help=\"The tag of a Docker image to use for the server.\",\n    ),\n    namespace: str = typer.Option(\n        \"default\",\n        \"-n\",\n        \"--namespace\",\n        help=\"A Kubernetes namespace to create the server in.\",\n    ),\n    log_level: str = SettingsOption(PREFECT_LOGGING_SERVER_LEVEL),\n):\n    \"\"\"\n    Generates a manifest for deploying Prefect on Kubernetes.\n\n    Example:\n        $ prefect kubernetes manifest server | kubectl apply -f -\n    \"\"\"\n\n    template = Template(\n        (\n            prefect.__module_path__ / \"cli\" / \"templates\" / \"kubernetes-server.yaml\"\n        ).read_text()\n    )\n    manifest = template.substitute(\n        {\n            \"image_name\": image_tag,\n            \"namespace\": namespace,\n            \"log_level\": log_level,\n        }\n    )\n    print(manifest)\n
    ","tags":["Python API","kubernetes","CLI"]},{"location":"api-ref/prefect/cli/profile/","title":"profile","text":"","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile","title":"prefect.cli.profile","text":"

    Command line interface for working with profiles.

    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.create","title":"create","text":"

    Create a new profile.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef create(\n    name: str,\n    from_name: str = typer.Option(None, \"--from\", help=\"Copy an existing profile.\"),\n):\n    \"\"\"\n    Create a new profile.\n    \"\"\"\n\n    profiles = prefect.settings.load_profiles()\n    if name in profiles:\n        app.console.print(\n            textwrap.dedent(\n                f\"\"\"\n                [red]Profile {name!r} already exists.[/red]\n                To create a new profile, remove the existing profile first:\n\n                    prefect profile delete {name!r}\n                \"\"\"\n            ).strip()\n        )\n        raise typer.Exit(1)\n\n    if from_name:\n        if from_name not in profiles:\n            exit_with_error(f\"Profile {from_name!r} not found.\")\n\n        # Create a copy of the profile with a new name and add to the collection\n        profiles.add_profile(profiles[from_name].copy(update={\"name\": name}))\n    else:\n        profiles.add_profile(prefect.settings.Profile(name=name, settings={}))\n\n    prefect.settings.save_profiles(profiles)\n\n    app.console.print(\n        textwrap.dedent(\n            f\"\"\"\n            Created profile with properties:\n                name - {name!r}\n                from name - {from_name or None}\n\n            Use created profile for future, subsequent commands:\n                prefect profile use {name!r}\n\n            Use created profile temporarily for a single command:\n                prefect -p {name!r} config view\n            \"\"\"\n        )\n    )\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.delete","title":"delete","text":"

    Delete the given profile.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef delete(name: str):\n    \"\"\"\n    Delete the given profile.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    if name not in profiles:\n        exit_with_error(f\"Profile {name!r} not found.\")\n\n    current_profile = prefect.context.get_settings_context().profile\n    if current_profile.name == name:\n        exit_with_error(\n            f\"Profile {name!r} is the active profile. You must switch profiles before\"\n            \" it can be deleted.\"\n        )\n\n    profiles.remove_profile(name)\n\n    verb = \"Removed\"\n    if name == \"default\":\n        verb = \"Reset\"\n\n    prefect.settings.save_profiles(profiles)\n    exit_with_success(f\"{verb} profile {name!r}.\")\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.inspect","title":"inspect","text":"

    Display settings from a given profile; defaults to active.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef inspect(\n    name: Optional[str] = typer.Argument(\n        None, help=\"Name of profile to inspect; defaults to active profile.\"\n    )\n):\n    \"\"\"\n    Display settings from a given profile; defaults to active.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    if name is None:\n        current_profile = prefect.context.get_settings_context().profile\n        if not current_profile:\n            exit_with_error(\"No active profile set - please provide a name to inspect.\")\n        name = current_profile.name\n        print(f\"No name provided, defaulting to {name!r}\")\n    if name not in profiles:\n        exit_with_error(f\"Profile {name!r} not found.\")\n\n    if not profiles[name].settings:\n        # TODO: Consider instructing on how to add settings.\n        print(f\"Profile {name!r} is empty.\")\n\n    for setting, value in profiles[name].settings.items():\n        app.console.print(f\"{setting.name}='{value}'\")\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.ls","title":"ls","text":"

    List profile names.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef ls():\n    \"\"\"\n    List profile names.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    current_profile = prefect.context.get_settings_context().profile\n    current_name = current_profile.name if current_profile is not None else None\n\n    table = Table(caption=\"* active profile\")\n    table.add_column(\n        \"[#024dfd]Available Profiles:\", justify=\"right\", style=\"#8ea0ae\", no_wrap=True\n    )\n\n    for name in profiles:\n        if name == current_name:\n            table.add_row(f\"[green]  * {name}[/green]\")\n        else:\n            table.add_row(f\"  {name}\")\n    app.console.print(table)\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.rename","title":"rename","text":"

    Change the name of a profile.

    Source code in prefect/cli/profile.py
    @profile_app.command()\ndef rename(name: str, new_name: str):\n    \"\"\"\n    Change the name of a profile.\n    \"\"\"\n    profiles = prefect.settings.load_profiles()\n    if name not in profiles:\n        exit_with_error(f\"Profile {name!r} not found.\")\n\n    if new_name in profiles:\n        exit_with_error(f\"Profile {new_name!r} already exists.\")\n\n    profiles.add_profile(profiles[name].copy(update={\"name\": new_name}))\n    profiles.remove_profile(name)\n\n    # If the active profile was renamed switch the active profile to the new name.\n    prefect.context.get_settings_context().profile\n    if profiles.active_name == name:\n        profiles.set_active(new_name)\n    if os.environ.get(\"PREFECT_PROFILE\") == name:\n        app.console.print(\n            f\"You have set your current profile to {name!r} with the \"\n            \"PREFECT_PROFILE environment variable. You must update this variable to \"\n            f\"{new_name!r} to continue using the profile.\"\n        )\n\n    prefect.settings.save_profiles(profiles)\n    exit_with_success(f\"Renamed profile {name!r} to {new_name!r}.\")\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/profile/#prefect.cli.profile.use","title":"use async","text":"

    Set the given profile to active.

    Source code in prefect/cli/profile.py
    @profile_app.command()\nasync def use(name: str):\n    \"\"\"\n    Set the given profile to active.\n    \"\"\"\n    status_messages = {\n        ConnectionStatus.CLOUD_CONNECTED: (\n            exit_with_success,\n            f\"Connected to Prefect Cloud using profile {name!r}\",\n        ),\n        ConnectionStatus.CLOUD_ERROR: (\n            exit_with_error,\n            f\"Error connecting to Prefect Cloud using profile {name!r}\",\n        ),\n        ConnectionStatus.CLOUD_UNAUTHORIZED: (\n            exit_with_error,\n            f\"Error authenticating with Prefect Cloud using profile {name!r}\",\n        ),\n        ConnectionStatus.ORION_CONNECTED: (\n            exit_with_success,\n            f\"Connected to Prefect server using profile {name!r}\",\n        ),\n        ConnectionStatus.ORION_ERROR: (\n            exit_with_error,\n            f\"Error connecting to Prefect server using profile {name!r}\",\n        ),\n        ConnectionStatus.EPHEMERAL: (\n            exit_with_success,\n            (\n                f\"No Prefect server specified using profile {name!r} - the API will run\"\n                \" in ephemeral mode.\"\n            ),\n        ),\n        ConnectionStatus.INVALID_API: (\n            exit_with_error,\n            \"Error connecting to Prefect API URL\",\n        ),\n    }\n\n    profiles = prefect.settings.load_profiles()\n    if name not in profiles.names:\n        exit_with_error(f\"Profile {name!r} not found.\")\n\n    profiles.set_active(name)\n    prefect.settings.save_profiles(profiles)\n\n    with Progress(\n        SpinnerColumn(),\n        TextColumn(\"[progress.description]{task.description}\"),\n        transient=False,\n    ) as progress:\n        progress.add_task(\n            description=\"Checking API connectivity...\",\n            total=None,\n        )\n\n        with use_profile(name, include_current_context=False):\n            connection_status = await check_orion_connection()\n\n        exit_method, msg = status_messages[connection_status]\n\n    exit_method(msg)\n
    ","tags":["Python API","CLI","settings","configuration","profiles"]},{"location":"api-ref/prefect/cli/project/","title":"project","text":"","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project","title":"prefect.cli.project","text":"

    Deprecated - Command line interface for working with projects.

    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project.clone","title":"clone async","text":"

    Clone an existing project for a given deployment.

    Source code in prefect/cli/project.py
    @project_app.command()\nasync def clone(\n    deployment_name: str = typer.Option(\n        None,\n        \"--deployment\",\n        \"-d\",\n        help=\"The name of the deployment to clone a project for.\",\n    ),\n    deployment_id: str = typer.Option(\n        None,\n        \"--id\",\n        \"-i\",\n        help=\"The id of the deployment to clone a project for.\",\n    ),\n):\n    \"\"\"\n    Clone an existing project for a given deployment.\n    \"\"\"\n    app.console.print(\n        generate_deprecation_message(\n            \"The `prefect project clone` command\",\n            start_date=\"Jun 2023\",\n        )\n    )\n    if deployment_name and deployment_id:\n        exit_with_error(\n            \"Can only pass one of deployment name or deployment ID options.\"\n        )\n\n    if not deployment_name and not deployment_id:\n        exit_with_error(\"Must pass either a deployment name or deployment ID.\")\n\n    if deployment_name:\n        async with get_client() as client:\n            try:\n                deployment = await client.read_deployment_by_name(deployment_name)\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {deployment_name!r} not found!\")\n    else:\n        async with get_client() as client:\n            try:\n                deployment = await client.read_deployment(deployment_id)\n            except ObjectNotFound:\n                exit_with_error(f\"Deployment {deployment_id!r} not found!\")\n\n    if deployment.pull_steps:\n        output = await run_steps(deployment.pull_steps)\n        app.console.out(output[\"directory\"])\n    else:\n        exit_with_error(\"No pull steps found, exiting early.\")\n
    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project.init","title":"init async","text":"

    Initialize a new project.

    Source code in prefect/cli/project.py
    @project_app.command()\n@app.command()\nasync def init(\n    name: str = None,\n    recipe: str = None,\n    fields: List[str] = typer.Option(\n        None,\n        \"-f\",\n        \"--field\",\n        help=(\n            \"One or more fields to pass to the recipe (e.g., image_name) in the format\"\n            \" of key=value.\"\n        ),\n    ),\n):\n    \"\"\"\n    Initialize a new project.\n    \"\"\"\n    inputs = {}\n    fields = fields or []\n    recipe_paths = prefect.__module_path__ / \"deployments\" / \"recipes\"\n\n    for field in fields:\n        key, value = field.split(\"=\")\n        inputs[key] = value\n\n    if not recipe and is_interactive():\n        recipe_paths = prefect.__module_path__ / \"deployments\" / \"recipes\"\n        recipes = []\n\n        for r in recipe_paths.iterdir():\n            if r.is_dir() and (r / \"prefect.yaml\").exists():\n                with open(r / \"prefect.yaml\") as f:\n                    recipe_data = yaml.safe_load(f)\n                    recipe_name = r.name\n                    recipe_description = recipe_data.get(\n                        \"description\", \"(no description available)\"\n                    )\n                    recipe_dict = {\n                        \"name\": recipe_name,\n                        \"description\": recipe_description,\n                    }\n                    recipes.append(recipe_dict)\n\n        selected_recipe = prompt_select_from_table(\n            app.console,\n            \"Would you like to initialize your deployment configuration with a recipe?\",\n            columns=[\n                {\"header\": \"Name\", \"key\": \"name\"},\n                {\"header\": \"Description\", \"key\": \"description\"},\n            ],\n            data=recipes,\n            opt_out_message=\"No, I'll use the default deployment configuration.\",\n            opt_out_response={},\n        )\n        if selected_recipe != {}:\n            recipe = selected_recipe[\"name\"]\n\n    if recipe and (recipe_paths / recipe / \"prefect.yaml\").exists():\n        with open(recipe_paths / recipe / \"prefect.yaml\") as f:\n            recipe_inputs = yaml.safe_load(f).get(\"required_inputs\") or {}\n\n        if recipe_inputs:\n            if set(recipe_inputs.keys()) < set(inputs.keys()):\n                # message to user about extra fields\n                app.console.print(\n                    (\n                        f\"Warning: extra fields provided for {recipe!r} recipe:\"\n                        f\" '{', '.join(set(inputs.keys()) - set(recipe_inputs.keys()))}'\"\n                    ),\n                    style=\"red\",\n                )\n            elif set(recipe_inputs.keys()) > set(inputs.keys()):\n                table = Table(\n                    title=f\"[red]Required inputs for {recipe!r} recipe[/red]\",\n                )\n                table.add_column(\"Field Name\", style=\"green\", no_wrap=True)\n                table.add_column(\n                    \"Description\", justify=\"left\", style=\"white\", no_wrap=False\n                )\n                for field, description in recipe_inputs.items():\n                    if field not in inputs:\n                        table.add_row(field, description)\n\n                app.console.print(table)\n\n                for key, description in recipe_inputs.items():\n                    if key not in inputs:\n                        inputs[key] = typer.prompt(key)\n\n            app.console.print(\"-\" * 15)\n\n    try:\n        files = [\n            f\"[green]{fname}[/green]\"\n            for fname in initialize_project(name=name, recipe=recipe, inputs=inputs)\n        ]\n    except ValueError as exc:\n        if \"Unknown recipe\" in str(exc):\n            exit_with_error(\n                f\"Unknown recipe {recipe!r} provided - run [yellow]`prefect init\"\n                \"`[/yellow] to see all available recipes.\"\n            )\n        else:\n            raise\n\n    files = \"\\n\".join(files)\n    empty_msg = (\n        f\"Created project in [green]{Path('.').resolve()}[/green]; no new files\"\n        \" created.\"\n    )\n    file_msg = (\n        f\"Created project in [green]{Path('.').resolve()}[/green] with the following\"\n        f\" new files:\\n{files}\"\n    )\n    app.console.print(file_msg if files else empty_msg)\n
    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project.ls","title":"ls async","text":"

    List available recipes.

    Source code in prefect/cli/project.py
    @recipe_app.command()\nasync def ls():\n    \"\"\"\n    List available recipes.\n    \"\"\"\n\n    recipe_paths = prefect.__module_path__ / \"deployments\" / \"recipes\"\n    recipes = {}\n\n    for recipe in recipe_paths.iterdir():\n        if recipe.is_dir() and (recipe / \"prefect.yaml\").exists():\n            with open(recipe / \"prefect.yaml\") as f:\n                recipes[recipe.name] = yaml.safe_load(f).get(\n                    \"description\", \"(no description available)\"\n                )\n\n    table = Table(\n        title=\"Available project recipes\",\n        caption=(\n            \"Run `prefect project init --recipe <recipe>` to initialize a project with\"\n            \" a recipe.\"\n        ),\n        caption_style=\"red\",\n    )\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Description\", justify=\"left\", style=\"white\", no_wrap=False)\n    for name, description in sorted(recipes.items(), key=lambda x: x[0]):\n        table.add_row(name, description)\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/project/#prefect.cli.project.register_flow","title":"register_flow async","text":"

    Register a flow with this project.

    Source code in prefect/cli/project.py
    @project_app.command()\nasync def register_flow(\n    entrypoint: str = typer.Argument(\n        ...,\n        help=(\n            \"The path to a flow entrypoint, in the form of\"\n            \" `./path/to/file.py:flow_func_name`\"\n        ),\n    ),\n    force: bool = typer.Option(\n        False,\n        \"--force\",\n        \"-f\",\n        help=(\n            \"An optional flag to force register this flow and overwrite any existing\"\n            \" entry\"\n        ),\n    ),\n):\n    \"\"\"\n    Register a flow with this project.\n    \"\"\"\n    try:\n        flow = await register(entrypoint, force=force)\n    except Exception as exc:\n        exit_with_error(exc)\n\n    app.console.print(\n        (\n            f\"Registered flow {flow.name!r} in\"\n            f\" {(find_prefect_directory()/'flows.json').resolve()!s}\"\n        ),\n        style=\"green\",\n    )\n
    ","tags":["Python API","CLI","projects","deployments","storage"]},{"location":"api-ref/prefect/cli/root/","title":"root","text":"","tags":["Python API","CLI"]},{"location":"api-ref/prefect/cli/root/#prefect.cli.root","title":"prefect.cli.root","text":"

    Base prefect command-line application

    ","tags":["Python API","CLI"]},{"location":"api-ref/prefect/cli/root/#prefect.cli.root.version","title":"version async","text":"

    Get the current Prefect version.

    Source code in prefect/cli/root.py
    @app.command()\nasync def version():\n    \"\"\"Get the current Prefect version.\"\"\"\n    import sqlite3\n\n    from prefect.server.utilities.database import get_dialect\n    from prefect.settings import PREFECT_API_DATABASE_CONNECTION_URL\n\n    version_info = {\n        \"Version\": prefect.__version__,\n        \"API version\": SERVER_API_VERSION,\n        \"Python version\": platform.python_version(),\n        \"Git commit\": prefect.__version_info__[\"full-revisionid\"][:8],\n        \"Built\": pendulum.parse(\n            prefect.__version_info__[\"date\"]\n        ).to_day_datetime_string(),\n        \"OS/Arch\": f\"{sys.platform}/{platform.machine()}\",\n        \"Profile\": prefect.context.get_settings_context().profile.name,\n    }\n\n    server_type: str\n\n    try:\n        # We do not context manage the client because when using an ephemeral app we do not\n        # want to create the database or run migrations\n        client = prefect.get_client()\n        server_type = client.server_type.value\n    except Exception:\n        server_type = \"<client error>\"\n\n    version_info[\"Server type\"] = server_type.lower()\n\n    # TODO: Consider adding an API route to retrieve this information?\n    if server_type == ServerType.EPHEMERAL.value:\n        database = get_dialect(PREFECT_API_DATABASE_CONNECTION_URL.value()).name\n        version_info[\"Server\"] = {\"Database\": database}\n        if database == \"sqlite\":\n            version_info[\"Server\"][\"SQLite version\"] = sqlite3.sqlite_version\n\n    def display(object: dict, nesting: int = 0):\n        # Recursive display of a dictionary with nesting\n        for key, value in object.items():\n            key += \":\"\n            if isinstance(value, dict):\n                app.console.print(key)\n                return display(value, nesting + 2)\n            prefix = \" \" * nesting\n            app.console.print(f\"{prefix}{key.ljust(20 - len(prefix))} {value}\")\n\n    display(version_info)\n
    ","tags":["Python API","CLI"]},{"location":"api-ref/prefect/cli/server/","title":"server","text":"","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server","title":"prefect.cli.server","text":"

    Command line interface for working with Prefect

    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.downgrade","title":"downgrade async","text":"

    Downgrade the Prefect database

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def downgrade(\n    yes: bool = typer.Option(False, \"--yes\", \"-y\"),\n    revision: str = typer.Option(\n        \"base\",\n        \"-r\",\n        help=(\n            \"The revision to pass to `alembic downgrade`. If not provided, runs all\"\n            \" migrations.\"\n        ),\n    ),\n    dry_run: bool = typer.Option(\n        False,\n        help=(\n            \"Flag to show what migrations would be made without applying them. Will\"\n            \" emit sql statements to stdout.\"\n        ),\n    ),\n):\n    \"\"\"Downgrade the Prefect database\"\"\"\n    from prefect.server.database.alembic_commands import alembic_downgrade\n    from prefect.server.database.dependencies import provide_database_interface\n\n    db = provide_database_interface()\n\n    engine = await db.engine()\n\n    if not yes:\n        confirm = typer.confirm(\n            \"Are you sure you want to downgrade the Prefect \"\n            f\"database at {engine.url!r}?\"\n        )\n        if not confirm:\n            exit_with_error(\"Database downgrade aborted!\")\n\n    app.console.print(\"Running downgrade migrations ...\")\n    await run_sync_in_worker_thread(\n        alembic_downgrade, revision=revision, dry_run=dry_run\n    )\n    app.console.print(\"Migrations succeeded!\")\n    exit_with_success(f\"Prefect database at {engine.url!r} downgraded!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.reset","title":"reset async","text":"

    Drop and recreate all Prefect database tables

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def reset(yes: bool = typer.Option(False, \"--yes\", \"-y\")):\n    \"\"\"Drop and recreate all Prefect database tables\"\"\"\n    from prefect.server.database.dependencies import provide_database_interface\n\n    db = provide_database_interface()\n    engine = await db.engine()\n    if not yes:\n        confirm = typer.confirm(\n            \"Are you sure you want to reset the Prefect database located \"\n            f'at \"{engine.url!r}\"? This will drop and recreate all tables.'\n        )\n        if not confirm:\n            exit_with_error(\"Database reset aborted\")\n    app.console.print(\"Downgrading database...\")\n    await db.drop_db()\n    app.console.print(\"Upgrading database...\")\n    await db.create_db()\n    exit_with_success(f'Prefect database \"{engine.url!r}\" reset!')\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.revision","title":"revision async","text":"

    Create a new migration for the Prefect database

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def revision(\n    message: str = typer.Option(\n        None,\n        \"--message\",\n        \"-m\",\n        help=\"A message to describe the migration.\",\n    ),\n    autogenerate: bool = False,\n):\n    \"\"\"Create a new migration for the Prefect database\"\"\"\n    from prefect.server.database.alembic_commands import alembic_revision\n\n    app.console.print(\"Running migration file creation ...\")\n    await run_sync_in_worker_thread(\n        alembic_revision,\n        message=message,\n        autogenerate=autogenerate,\n    )\n    exit_with_success(\"Creating new migration file succeeded!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.stamp","title":"stamp async","text":"

    Stamp the revision table with the given revision; don't run any migrations

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def stamp(revision: str):\n    \"\"\"Stamp the revision table with the given revision; don't run any migrations\"\"\"\n    from prefect.server.database.alembic_commands import alembic_stamp\n\n    app.console.print(\"Stamping database with revision ...\")\n    await run_sync_in_worker_thread(alembic_stamp, revision=revision)\n    exit_with_success(\"Stamping database with revision succeeded!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.start","title":"start async","text":"

    Start a Prefect server

    Source code in prefect/cli/server.py
    @server_app.command()\nasync def start(\n    host: str = SettingsOption(PREFECT_SERVER_API_HOST),\n    port: int = SettingsOption(PREFECT_SERVER_API_PORT),\n    keep_alive_timeout: int = SettingsOption(PREFECT_SERVER_API_KEEPALIVE_TIMEOUT),\n    log_level: str = SettingsOption(PREFECT_LOGGING_SERVER_LEVEL),\n    scheduler: bool = SettingsOption(PREFECT_API_SERVICES_SCHEDULER_ENABLED),\n    analytics: bool = SettingsOption(\n        PREFECT_SERVER_ANALYTICS_ENABLED, \"--analytics-on/--analytics-off\"\n    ),\n    late_runs: bool = SettingsOption(PREFECT_API_SERVICES_LATE_RUNS_ENABLED),\n    ui: bool = SettingsOption(PREFECT_UI_ENABLED),\n):\n    \"\"\"Start a Prefect server\"\"\"\n\n    server_env = os.environ.copy()\n    server_env[\"PREFECT_API_SERVICES_SCHEDULER_ENABLED\"] = str(scheduler)\n    server_env[\"PREFECT_SERVER_ANALYTICS_ENABLED\"] = str(analytics)\n    server_env[\"PREFECT_API_SERVICES_LATE_RUNS_ENABLED\"] = str(late_runs)\n    server_env[\"PREFECT_API_SERVICES_UI\"] = str(ui)\n    server_env[\"PREFECT_LOGGING_SERVER_LEVEL\"] = log_level\n\n    base_url = f\"http://{host}:{port}\"\n\n    async with anyio.create_task_group() as tg:\n        app.console.print(generate_welcome_blurb(base_url, ui_enabled=ui))\n        app.console.print(\"\\n\")\n\n        server_process_id = await tg.start(\n            partial(\n                run_process,\n                command=[\n                    get_sys_executable(),\n                    \"-m\",\n                    \"uvicorn\",\n                    \"--app-dir\",\n                    # quote wrapping needed for windows paths with spaces\n                    f'\"{prefect.__module_path__.parent}\"',\n                    \"--factory\",\n                    \"prefect.server.api.server:create_app\",\n                    \"--host\",\n                    str(host),\n                    \"--port\",\n                    str(port),\n                    \"--timeout-keep-alive\",\n                    str(keep_alive_timeout),\n                ],\n                env=server_env,\n                stream_output=True,\n            )\n        )\n\n        # Explicitly handle the interrupt signal here, as it will allow us to\n        # cleanly stop the uvicorn server. Failing to do that may cause a\n        # large amount of anyio error traces on the terminal, because the\n        # SIGINT is handled by Typer/Click in this process (the parent process)\n        # and will start shutting down subprocesses:\n        # https://github.com/PrefectHQ/server/issues/2475\n\n        setup_signal_handlers_server(\n            server_process_id, \"the Prefect server\", app.console.print\n        )\n\n    app.console.print(\"Server stopped!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/server/#prefect.cli.server.upgrade","title":"upgrade async","text":"

    Upgrade the Prefect database

    Source code in prefect/cli/server.py
    @database_app.command()\nasync def upgrade(\n    yes: bool = typer.Option(False, \"--yes\", \"-y\"),\n    revision: str = typer.Option(\n        \"head\",\n        \"-r\",\n        help=(\n            \"The revision to pass to `alembic upgrade`. If not provided, runs all\"\n            \" migrations.\"\n        ),\n    ),\n    dry_run: bool = typer.Option(\n        False,\n        help=(\n            \"Flag to show what migrations would be made without applying them. Will\"\n            \" emit sql statements to stdout.\"\n        ),\n    ),\n):\n    \"\"\"Upgrade the Prefect database\"\"\"\n    from prefect.server.database.alembic_commands import alembic_upgrade\n    from prefect.server.database.dependencies import provide_database_interface\n\n    db = provide_database_interface()\n    engine = await db.engine()\n\n    if not yes:\n        confirm = typer.confirm(\n            f\"Are you sure you want to upgrade the Prefect database at {engine.url!r}?\"\n        )\n        if not confirm:\n            exit_with_error(\"Database upgrade aborted!\")\n\n    app.console.print(\"Running upgrade migrations ...\")\n    await run_sync_in_worker_thread(alembic_upgrade, revision=revision, dry_run=dry_run)\n    app.console.print(\"Migrations succeeded!\")\n    exit_with_success(f\"Prefect database at {engine.url!r} upgraded!\")\n
    ","tags":["Python API","CLI","Kubernetes","database"]},{"location":"api-ref/prefect/cli/variable/","title":"variable","text":"","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/variable/#prefect.cli.variable","title":"prefect.cli.variable","text":"","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/variable/#prefect.cli.variable.delete","title":"delete async","text":"

    Delete a variable.

    Parameters:

    Name Type Description Default name str

    the name of the variable to delete

    required Source code in prefect/cli/variable.py
    @variable_app.command(\"delete\")\nasync def delete(\n    name: str,\n):\n    \"\"\"\n    Delete a variable.\n\n    Arguments:\n        name: the name of the variable to delete\n    \"\"\"\n\n    async with get_client() as client:\n        try:\n            await client.delete_variable_by_name(\n                name=name,\n            )\n        except ObjectNotFound:\n            exit_with_error(f\"Variable {name!r} not found.\")\n\n        exit_with_success(f\"Deleted variable {name!r}.\")\n
    ","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/variable/#prefect.cli.variable.inspect","title":"inspect async","text":"

    View details about a variable.

    Parameters:

    Name Type Description Default name str

    the name of the variable to inspect

    required Source code in prefect/cli/variable.py
    @variable_app.command(\"inspect\")\nasync def inspect(\n    name: str,\n):\n    \"\"\"\n    View details about a variable.\n\n    Arguments:\n        name: the name of the variable to inspect\n    \"\"\"\n\n    async with get_client() as client:\n        variable = await client.read_variable_by_name(\n            name=name,\n        )\n        if not variable:\n            exit_with_error(f\"Variable {name!r} not found.\")\n\n        app.console.print(Pretty(variable))\n
    ","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/variable/#prefect.cli.variable.list_variables","title":"list_variables async","text":"

    List variables.

    Source code in prefect/cli/variable.py
    @variable_app.command(\"ls\")\nasync def list_variables(\n    limit: int = typer.Option(\n        100,\n        \"--limit\",\n        help=\"The maximum number of variables to return.\",\n    ),\n):\n    \"\"\"\n    List variables.\n    \"\"\"\n    async with get_client() as client:\n        variables = await client.read_variables(\n            limit=limit,\n        )\n\n        table = Table(\n            title=\"Variables\",\n            caption=\"List Variables using `prefect variable ls`\",\n            show_header=True,\n        )\n\n        table.add_column(\"Name\", style=\"blue\", no_wrap=True)\n        # values can be up 5000 characters so truncate early\n        table.add_column(\"Value\", style=\"blue\", no_wrap=True, max_width=50)\n        table.add_column(\"Created\", style=\"blue\", no_wrap=True)\n        table.add_column(\"Updated\", style=\"blue\", no_wrap=True)\n\n        for variable in sorted(variables, key=lambda x: f\"{x.name}\"):\n            table.add_row(\n                variable.name,\n                variable.value,\n                pendulum.instance(variable.created).diff_for_humans(),\n                pendulum.instance(variable.updated).diff_for_humans(),\n            )\n\n        app.console.print(table)\n
    ","tags":["Python API","variables","CLI"]},{"location":"api-ref/prefect/cli/work_pool/","title":"work_pool","text":"","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool","title":"prefect.cli.work_pool","text":"

    Command line interface for working with work queues.

    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.clear_concurrency_limit","title":"clear_concurrency_limit async","text":"

    Clear the concurrency limit for a work pool.

    \b Examples: $ prefect work-pool clear-concurrency-limit \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def clear_concurrency_limit(\n    name: str = typer.Argument(..., help=\"The name of the work pool to update.\"),\n):\n    \"\"\"\n    Clear the concurrency limit for a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool clear-concurrency-limit \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    concurrency_limit=None,\n                ),\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(f\"Cleared concurrency limit for work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.create","title":"create async","text":"

    Create a new work pool.

    \b Examples: \b Create a Kubernetes work pool in a paused state: \b $ prefect work-pool create \"my-pool\" --type kubernetes --paused \b Create a Docker work pool with a custom base job template: \b $ prefect work-pool create \"my-pool\" --type docker --base-job-template ./base-job-template.json

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def create(\n    name: str = typer.Argument(..., help=\"The name of the work pool.\"),\n    base_job_template: typer.FileText = typer.Option(\n        None,\n        \"--base-job-template\",\n        help=(\n            \"The path to a JSON file containing the base job template to use. If\"\n            \" unspecified, Prefect will use the default base job template for the given\"\n            \" worker type.\"\n        ),\n    ),\n    paused: bool = typer.Option(\n        False,\n        \"--paused\",\n        help=\"Whether or not to create the work pool in a paused state.\",\n    ),\n    type: str = typer.Option(\n        None, \"-t\", \"--type\", help=\"The type of work pool to create.\"\n    ),\n    set_as_default: bool = typer.Option(\n        False,\n        \"--set-as-default\",\n        help=(\n            \"Whether or not to use the created work pool as the local default for\"\n            \" deployment.\"\n        ),\n    ),\n    provision_infrastructure: bool = typer.Option(\n        False,\n        \"--provision-infrastructure\",\n        \"--provision-infra\",\n        help=(\n            \"Whether or not to provision infrastructure for the work pool if supported\"\n            \" for the given work pool type.\"\n        ),\n    ),\n):\n    \"\"\"\n    Create a new work pool.\n\n    \\b\n    Examples:\n        \\b\n        Create a Kubernetes work pool in a paused state:\n            \\b\n            $ prefect work-pool create \"my-pool\" --type kubernetes --paused\n        \\b\n        Create a Docker work pool with a custom base job template:\n            \\b\n            $ prefect work-pool create \"my-pool\" --type docker --base-job-template ./base-job-template.json\n\n    \"\"\"\n    if not name.lower().strip(\"'\\\" \"):\n        exit_with_error(\"Work pool name cannot be empty.\")\n    async with get_client() as client:\n        try:\n            await client.read_work_pool(work_pool_name=name)\n        except ObjectNotFound:\n            pass\n        else:\n            exit_with_error(\n                f\"Work pool named {name!r} already exists. Please try creating your\"\n                \" work pool again with a different name.\"\n            )\n\n        if type is None:\n            async with get_collections_metadata_client() as collections_client:\n                if not is_interactive():\n                    exit_with_error(\n                        \"When not using an interactive terminal, you must supply a\"\n                        \" `--type` value.\"\n                    )\n                worker_metadata = await collections_client.read_worker_metadata()\n\n                # Retrieve only push pools if provisioning infrastructure\n                data = [\n                    worker\n                    for collection in worker_metadata.values()\n                    for worker in collection.values()\n                    if provision_infrastructure\n                    and has_provisioner_for_type(worker[\"type\"])\n                    or not provision_infrastructure\n                ]\n                worker = prompt_select_from_table(\n                    app.console,\n                    \"What type of work pool infrastructure would you like to use?\",\n                    columns=[\n                        {\"header\": \"Infrastructure Type\", \"key\": \"display_name\"},\n                        {\"header\": \"Description\", \"key\": \"description\"},\n                    ],\n                    data=data,\n                    table_kwargs={\"show_lines\": True},\n                )\n                type = worker[\"type\"]\n\n        available_work_pool_types = await get_available_work_pool_types()\n        if type not in available_work_pool_types:\n            exit_with_error(\n                f\"Unknown work pool type {type!r}. \"\n                \"Please choose from\"\n                f\" {', '.join(available_work_pool_types)}.\"\n            )\n\n        if base_job_template is None:\n            template_contents = (\n                await get_default_base_job_template_for_infrastructure_type(type)\n            )\n        else:\n            template_contents = json.load(base_job_template)\n\n        if provision_infrastructure:\n            try:\n                provisioner = get_infrastructure_provisioner_for_work_pool_type(type)\n                provisioner.console = app.console\n                template_contents = await provisioner.provision(\n                    work_pool_name=name, base_job_template=template_contents\n                )\n            except ValueError as exc:\n                print(exc)\n                app.console.print(\n                    (\n                        \"Automatic infrastructure provisioning is not supported for\"\n                        f\" {type!r} work pools.\"\n                    ),\n                    style=\"yellow\",\n                )\n            except RuntimeError as exc:\n                exit_with_error(f\"Failed to provision infrastructure: {exc}\")\n\n        try:\n            wp = WorkPoolCreate(\n                name=name,\n                type=type,\n                base_job_template=template_contents,\n                is_paused=paused,\n            )\n            work_pool = await client.create_work_pool(work_pool=wp)\n            app.console.print(f\"Created work pool {work_pool.name!r}!\\n\", style=\"green\")\n            if (\n                not work_pool.is_paused\n                and not work_pool.is_managed_pool\n                and not work_pool.is_push_pool\n            ):\n                app.console.print(\"To start a worker for this work pool, run:\\n\")\n                app.console.print(\n                    f\"\\t[blue]prefect worker start --pool {work_pool.name}[/]\\n\"\n                )\n            if set_as_default:\n                set_work_pool_as_default(work_pool.name)\n            exit_with_success(\"\")\n        except ObjectAlreadyExists:\n            exit_with_error(\n                f\"Work pool named {name!r} already exists. Please try creating your\"\n                \" work pool again with a different name.\"\n            )\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.delete","title":"delete async","text":"

    Delete a work pool.

    \b Examples: $ prefect work-pool delete \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def delete(\n    name: str = typer.Argument(..., help=\"The name of the work pool to delete.\"),\n):\n    \"\"\"\n    Delete a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool delete \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.delete_work_pool(work_pool_name=name)\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(f\"Deleted work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.get_default_base_job_template","title":"get_default_base_job_template async","text":"

    Get the default base job template for a given work pool type.

    \b Examples: $ prefect work-pool get-default-base-job-template --type kubernetes

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def get_default_base_job_template(\n    type: str = typer.Option(\n        None,\n        \"-t\",\n        \"--type\",\n        help=\"The type of work pool for which to get the default base job template.\",\n    ),\n    file: str = typer.Option(\n        None, \"-f\", \"--file\", help=\"If set, write the output to a file.\"\n    ),\n):\n    \"\"\"\n    Get the default base job template for a given work pool type.\n\n    \\b\n    Examples:\n        $ prefect work-pool get-default-base-job-template --type kubernetes\n    \"\"\"\n    base_job_template = await get_default_base_job_template_for_infrastructure_type(\n        type\n    )\n    if base_job_template is None:\n        exit_with_error(\n            f\"Unknown work pool type {type!r}. \"\n            \"Please choose from\"\n            f\" {', '.join(await get_available_work_pool_types())}.\"\n        )\n\n    if file is None:\n        print(json.dumps(base_job_template, indent=2))\n    else:\n        with open(file, mode=\"w\") as f:\n            json.dump(base_job_template, fp=f, indent=2)\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.has_provisioner_for_type","title":"has_provisioner_for_type","text":"

    Check if there is a provisioner for the given work pool type.

    Parameters:

    Name Type Description Default work_pool_type str

    The type of the work pool.

    required

    Returns:

    Name Type Description bool bool

    True if a provisioner exists for the given type, False otherwise.

    Source code in prefect/cli/work_pool.py
    def has_provisioner_for_type(work_pool_type: str) -> bool:\n    \"\"\"\n    Check if there is a provisioner for the given work pool type.\n\n    Args:\n        work_pool_type (str): The type of the work pool.\n\n    Returns:\n        bool: True if a provisioner exists for the given type, False otherwise.\n    \"\"\"\n    return work_pool_type in _provisioners\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.inspect","title":"inspect async","text":"

    Inspect a work pool.

    \b Examples: $ prefect work-pool inspect \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def inspect(\n    name: str = typer.Argument(..., help=\"The name of the work pool to inspect.\"),\n):\n    \"\"\"\n    Inspect a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool inspect \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            pool = await client.read_work_pool(work_pool_name=name)\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        app.console.print(Pretty(pool))\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.ls","title":"ls async","text":"

    List work pools.

    \b Examples: $ prefect work-pool ls

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def ls(\n    verbose: bool = typer.Option(\n        False,\n        \"--verbose\",\n        \"-v\",\n        help=\"Show additional information about work pools.\",\n    ),\n):\n    \"\"\"\n    List work pools.\n\n    \\b\n    Examples:\n        $ prefect work-pool ls\n    \"\"\"\n    table = Table(\n        title=\"Work Pools\", caption=\"(**) denotes a paused pool\", caption_style=\"red\"\n    )\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Type\", style=\"magenta\", no_wrap=True)\n    table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n    if verbose:\n        table.add_column(\"Base Job Template\", style=\"magenta\", no_wrap=True)\n\n    async with get_client() as client:\n        pools = await client.read_work_pools()\n\n    def sort_by_created_key(q):\n        return pendulum.now(\"utc\") - q.created\n\n    for pool in sorted(pools, key=sort_by_created_key):\n        row = [\n            f\"{pool.name} [red](**)\" if pool.is_paused else pool.name,\n            str(pool.type),\n            str(pool.id),\n            (\n                f\"[red]{pool.concurrency_limit}\"\n                if pool.concurrency_limit\n                else \"[blue]None\"\n            ),\n        ]\n        if verbose:\n            row.append(str(pool.base_job_template))\n        table.add_row(*row)\n\n    app.console.print(table)\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.pause","title":"pause async","text":"

    Pause a work pool.

    \b Examples: $ prefect work-pool pause \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def pause(\n    name: str = typer.Argument(..., help=\"The name of the work pool to pause.\"),\n):\n    \"\"\"\n    Pause a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool pause \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    is_paused=True,\n                ),\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(f\"Paused work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.preview","title":"preview async","text":"

    Preview the work pool's scheduled work for all queues.

    \b Examples: $ prefect work-pool preview \"my-pool\" --hours 24

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def preview(\n    name: str = typer.Argument(None, help=\"The name or ID of the work pool to preview\"),\n    hours: int = typer.Option(\n        None,\n        \"-h\",\n        \"--hours\",\n        help=\"The number of hours to look ahead; defaults to 1 hour\",\n    ),\n):\n    \"\"\"\n    Preview the work pool's scheduled work for all queues.\n\n    \\b\n    Examples:\n        $ prefect work-pool preview \"my-pool\" --hours 24\n\n    \"\"\"\n    if hours is None:\n        hours = 1\n\n    async with get_client() as client:\n        try:\n            responses = await client.get_scheduled_flow_runs_for_work_pool(\n                work_pool_name=name,\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n    runs = [response.flow_run for response in responses]\n    table = Table(caption=\"(**) denotes a late run\", caption_style=\"red\")\n\n    table.add_column(\n        \"Scheduled Start Time\", justify=\"left\", style=\"yellow\", no_wrap=True\n    )\n    table.add_column(\"Run ID\", justify=\"left\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Deployment ID\", style=\"blue\", no_wrap=True)\n\n    pendulum.now(\"utc\").add(hours=hours or 1)\n\n    now = pendulum.now(\"utc\")\n\n    def sort_by_created_key(r):\n        return now - r.created\n\n    for run in sorted(runs, key=sort_by_created_key):\n        table.add_row(\n            (\n                f\"{run.expected_start_time} [red](**)\"\n                if run.expected_start_time < now\n                else f\"{run.expected_start_time}\"\n            ),\n            str(run.id),\n            run.name,\n            str(run.deployment_id),\n        )\n\n    if runs:\n        app.console.print(table)\n    else:\n        app.console.print(\n            (\n                \"No runs found - try increasing how far into the future you preview\"\n                \" with the --hours flag\"\n            ),\n            style=\"yellow\",\n        )\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.provision_infrastructure","title":"provision_infrastructure async","text":"

    Provision infrastructure for a work pool.

    \b Examples: $ prefect work-pool provision-infrastructure \"my-pool\"

    $ prefect work-pool provision-infra \"my-pool\"\n
    Source code in prefect/cli/work_pool.py
    @work_pool_app.command(aliases=[\"provision-infra\"])\nasync def provision_infrastructure(\n    name: str = typer.Argument(\n        ..., help=\"The name of the work pool to provision infrastructure for.\"\n    ),\n):\n    \"\"\"\n    Provision infrastructure for a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool provision-infrastructure \"my-pool\"\n\n        $ prefect work-pool provision-infra \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            work_pool = await client.read_work_pool(work_pool_name=name)\n            if not work_pool.is_push_pool:\n                exit_with_error(\n                    f\"Work pool {name!r} is not a push pool type. \"\n                    \"Please try provisioning infrastructure for a push pool.\"\n                )\n        except ObjectNotFound:\n            exit_with_error(f\"Work pool {name!r} does not exist.\")\n        except Exception as exc:\n            exit_with_error(f\"Failed to read work pool {name!r}: {exc}\")\n\n        try:\n            provisioner = get_infrastructure_provisioner_for_work_pool_type(\n                work_pool.type\n            )\n            provisioner.console = app.console\n            new_base_job_template = await provisioner.provision(\n                work_pool_name=name, base_job_template=work_pool.base_job_template\n            )\n\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    base_job_template=new_base_job_template,\n                ),\n            )\n\n        except ValueError as exc:\n            app.console.print(f\"Error: {exc}\")\n            app.console.print(\n                (\n                    \"Automatic infrastructure provisioning is not supported for\"\n                    f\" {work_pool.type!r} work pools.\"\n                ),\n                style=\"yellow\",\n            )\n        except RuntimeError as exc:\n            exit_with_error(\n                f\"Failed to provision infrastructure for '{name}' work pool: {exc}\"\n            )\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.resume","title":"resume async","text":"

    Resume a work pool.

    \b Examples: $ prefect work-pool resume \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def resume(\n    name: str = typer.Argument(..., help=\"The name of the work pool to resume.\"),\n):\n    \"\"\"\n    Resume a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool resume \"my-pool\"\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    is_paused=False,\n                ),\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(f\"Resumed work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.set_concurrency_limit","title":"set_concurrency_limit async","text":"

    Set the concurrency limit for a work pool.

    \b Examples: $ prefect work-pool set-concurrency-limit \"my-pool\" 10

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def set_concurrency_limit(\n    name: str = typer.Argument(..., help=\"The name of the work pool to update.\"),\n    concurrency_limit: int = typer.Argument(\n        ..., help=\"The new concurrency limit for the work pool.\"\n    ),\n):\n    \"\"\"\n    Set the concurrency limit for a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool set-concurrency-limit \"my-pool\" 10\n\n    \"\"\"\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=WorkPoolUpdate(\n                    concurrency_limit=concurrency_limit,\n                ),\n            )\n        except ObjectNotFound as exc:\n            exit_with_error(exc)\n\n        exit_with_success(\n            f\"Set concurrency limit for work pool {name!r} to {concurrency_limit}\"\n        )\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_pool/#prefect.cli.work_pool.update","title":"update async","text":"

    Update a work pool.

    \b Examples: $ prefect work-pool update \"my-pool\"

    Source code in prefect/cli/work_pool.py
    @work_pool_app.command()\nasync def update(\n    name: str = typer.Argument(..., help=\"The name of the work pool to update.\"),\n    base_job_template: typer.FileText = typer.Option(\n        None,\n        \"--base-job-template\",\n        help=(\n            \"The path to a JSON file containing the base job template to use. If\"\n            \" unspecified, Prefect will use the default base job template for the given\"\n            \" worker type. If None, the base job template will not be modified.\"\n        ),\n    ),\n    concurrency_limit: int = typer.Option(\n        None,\n        \"--concurrency-limit\",\n        help=(\n            \"The concurrency limit for the work pool. If None, the concurrency limit\"\n            \" will not be modified.\"\n        ),\n    ),\n    description: str = typer.Option(\n        None,\n        \"--description\",\n        help=(\n            \"The description for the work pool. If None, the description will not be\"\n            \" modified.\"\n        ),\n    ),\n):\n    \"\"\"\n    Update a work pool.\n\n    \\b\n    Examples:\n        $ prefect work-pool update \"my-pool\"\n\n    \"\"\"\n    wp = WorkPoolUpdate()\n    if base_job_template:\n        wp.base_job_template = json.load(base_job_template)\n    if concurrency_limit:\n        wp.concurrency_limit = concurrency_limit\n    if description:\n        wp.description = description\n\n    async with get_client() as client:\n        try:\n            await client.update_work_pool(\n                work_pool_name=name,\n                work_pool=wp,\n            )\n        except ObjectNotFound:\n            exit_with_error(\"Work pool named {name!r} does not exist.\")\n\n        exit_with_success(f\"Updated work pool {name!r}\")\n
    ","tags":["Python API","work pools","CLI"]},{"location":"api-ref/prefect/cli/work_queue/","title":"work_queue","text":"","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue","title":"prefect.cli.work_queue","text":"

    Command line interface for working with work queues.

    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.clear_concurrency_limit","title":"clear_concurrency_limit async","text":"

    Clear any concurrency limits from a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def clear_concurrency_limit(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue to clear\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Clear any concurrency limits from a work queue.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n    async with get_client() as client:\n        try:\n            await client.update_work_queue(\n                id=queue_id,\n                concurrency_limit=None,\n            )\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n\n    if pool:\n        success_message = (\n            f\"Concurrency limits removed on work queue {name!r} in work pool {pool!r}\"\n        )\n    else:\n        success_message = f\"Concurrency limits removed on work queue {name!r}\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.create","title":"create async","text":"

    Create a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def create(\n    name: str = typer.Argument(..., help=\"The unique name to assign this work queue\"),\n    limit: int = typer.Option(\n        None, \"-l\", \"--limit\", help=\"The concurrency limit to set on the queue.\"\n    ),\n    tags: List[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--tag\",\n        help=(\n            \"DEPRECATED: One or more optional tags. This option will be removed on\"\n            \" 2023-02-23.\"\n        ),\n    ),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool to create the work queue in.\",\n    ),\n    priority: Optional[int] = typer.Option(\n        None,\n        \"-q\",\n        \"--priority\",\n        help=\"The associated priority for the created work queue\",\n    ),\n):\n    \"\"\"\n    Create a work queue.\n    \"\"\"\n    if tags:\n        app.console.print(\n            (\n                \"Supplying `tags` for work queues is deprecated. This work \"\n                \"queue will use legacy tag-matching behavior. \"\n                \"This option will be removed on 2023-02-23.\"\n            ),\n            style=\"red\",\n        )\n\n    if pool and tags:\n        exit_with_error(\n            \"Work queues created with tags cannot specify work pools or set priorities.\"\n        )\n\n    async with get_client() as client:\n        try:\n            result = await client.create_work_queue(\n                name=name, tags=tags or None, work_pool_name=pool, priority=priority\n            )\n            if limit is not None:\n                await client.update_work_queue(\n                    id=result.id,\n                    concurrency_limit=limit,\n                )\n        except ObjectAlreadyExists:\n            exit_with_error(f\"Work queue with name: {name!r} already exists.\")\n        except ObjectNotFound:\n            exit_with_error(f\"Work pool with name: {pool!r} not found.\")\n\n    if tags:\n        tags_message = f\"tags - {', '.join(sorted(tags))}\\n\" or \"\"\n        output_msg = dedent(\n            f\"\"\"\n            Created work queue with properties:\n                name - {name!r}\n                id - {result.id}\n                concurrency limit - {limit}\n                {tags_message}\n            Start an agent to pick up flow runs from the work queue:\n                prefect agent start -q '{result.name}'\n\n            Inspect the work queue:\n                prefect work-queue inspect '{result.name}'\n            \"\"\"\n        )\n    else:\n        if not pool:\n            # specify the default work pool name after work queue creation to allow the server\n            # to handle a bunch of logic associated with agents without work pools\n            pool = DEFAULT_AGENT_WORK_POOL_NAME\n        output_msg = dedent(\n            f\"\"\"\n            Created work queue with properties:\n                name - {name!r}\n                work pool - {pool!r}\n                id - {result.id}\n                concurrency limit - {limit}\n            Start an agent to pick up flow runs from the work queue:\n                prefect agent start -q '{result.name} -p {pool}'\n\n            Inspect the work queue:\n                prefect work-queue inspect '{result.name}'\n            \"\"\"\n        )\n    exit_with_success(output_msg)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.delete","title":"delete async","text":"

    Delete a work queue by ID.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def delete(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue to delete\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool containing the work queue to delete.\",\n    ),\n):\n    \"\"\"\n    Delete a work queue by ID.\n    \"\"\"\n\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n    async with get_client() as client:\n        try:\n            await client.delete_work_queue_by_id(id=queue_id)\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n    if pool:\n        success_message = (\n            f\"Successfully deleted work queue {name!r} in work pool {pool!r}\"\n        )\n    else:\n        success_message = f\"Successfully deleted work queue {name!r}\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.inspect","title":"inspect async","text":"

    Inspect a work queue by ID.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def inspect(\n    name: str = typer.Argument(\n        None, help=\"The name or ID of the work queue to inspect\"\n    ),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Inspect a work queue by ID.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n    async with get_client() as client:\n        try:\n            result = await client.read_work_queue(id=queue_id)\n            app.console.print(Pretty(result))\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n\n        try:\n            status = await client.read_work_queue_status(id=queue_id)\n            app.console.print(Pretty(status))\n        except ObjectNotFound:\n            pass\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.ls","title":"ls async","text":"

    View all work queues.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def ls(\n    verbose: bool = typer.Option(\n        False, \"--verbose\", \"-v\", help=\"Display more information.\"\n    ),\n    work_queue_prefix: str = typer.Option(\n        None,\n        \"--match\",\n        \"-m\",\n        help=(\n            \"Will match work queues with names that start with the specified prefix\"\n            \" string\"\n        ),\n    ),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool containing the work queues to list.\",\n    ),\n):\n    \"\"\"\n    View all work queues.\n    \"\"\"\n    if not pool and not experiment_enabled(\"work_pools\"):\n        table = Table(\n            title=\"Work Queues\",\n            caption=\"(**) denotes a paused queue\",\n            caption_style=\"red\",\n        )\n        table.add_column(\"Name\", style=\"green\", no_wrap=True)\n        table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n        table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n        if verbose:\n            table.add_column(\"Filter (Deprecated)\", style=\"magenta\", no_wrap=True)\n\n        async with get_client() as client:\n            if work_queue_prefix is not None:\n                queues = await client.match_work_queues([work_queue_prefix])\n            else:\n                queues = await client.read_work_queues()\n\n            def sort_by_created_key(q):\n                return pendulum.now(\"utc\") - q.created\n\n            for queue in sorted(queues, key=sort_by_created_key):\n                row = [\n                    f\"{queue.name} [red](**)\" if queue.is_paused else queue.name,\n                    str(queue.id),\n                    (\n                        f\"[red]{queue.concurrency_limit}\"\n                        if queue.concurrency_limit\n                        else \"[blue]None\"\n                    ),\n                ]\n                if verbose and queue.filter is not None:\n                    row.append(queue.filter.json())\n                table.add_row(*row)\n    elif not pool:\n        table = Table(\n            title=\"Work Queues\",\n            caption=\"(**) denotes a paused queue\",\n            caption_style=\"red\",\n        )\n        table.add_column(\"Name\", style=\"green\", no_wrap=True)\n        table.add_column(\"Pool\", style=\"magenta\", no_wrap=True)\n        table.add_column(\"ID\", justify=\"right\", style=\"cyan\", no_wrap=True)\n        table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n        if verbose:\n            table.add_column(\"Filter (Deprecated)\", style=\"magenta\", no_wrap=True)\n\n        async with get_client() as client:\n            if work_queue_prefix is not None:\n                queues = await client.match_work_queues([work_queue_prefix])\n            else:\n                queues = await client.read_work_queues()\n\n            pool_ids = [q.work_pool_id for q in queues]\n            wp_filter = WorkPoolFilter(id=WorkPoolFilterId(any_=pool_ids))\n            pools = await client.read_work_pools(work_pool_filter=wp_filter)\n            pool_id_name_map = {p.id: p.name for p in pools}\n\n            def sort_by_created_key(q):\n                return pendulum.now(\"utc\") - q.created\n\n            for queue in sorted(queues, key=sort_by_created_key):\n                row = [\n                    f\"{queue.name} [red](**)\" if queue.is_paused else queue.name,\n                    pool_id_name_map[queue.work_pool_id],\n                    str(queue.id),\n                    (\n                        f\"[red]{queue.concurrency_limit}\"\n                        if queue.concurrency_limit\n                        else \"[blue]None\"\n                    ),\n                ]\n                if verbose and queue.filter is not None:\n                    row.append(queue.filter.json())\n                table.add_row(*row)\n\n    else:\n        table = Table(\n            title=f\"Work Queues in Work Pool {pool!r}\",\n            caption=\"(**) denotes a paused queue\",\n            caption_style=\"red\",\n        )\n        table.add_column(\"Name\", style=\"green\", no_wrap=True)\n        table.add_column(\"Priority\", style=\"magenta\", no_wrap=True)\n        table.add_column(\"Concurrency Limit\", style=\"blue\", no_wrap=True)\n        if verbose:\n            table.add_column(\"Description\", style=\"cyan\", no_wrap=False)\n\n        async with get_client() as client:\n            try:\n                queues = await client.read_work_queues(work_pool_name=pool)\n            except ObjectNotFound:\n                exit_with_error(f\"No work pool found: {pool!r}\")\n\n            def sort_by_created_key(q):\n                return pendulum.now(\"utc\") - q.created\n\n            for queue in sorted(queues, key=sort_by_created_key):\n                row = [\n                    f\"{queue.name} [red](**)\" if queue.is_paused else queue.name,\n                    f\"{queue.priority}\",\n                    (\n                        f\"[red]{queue.concurrency_limit}\"\n                        if queue.concurrency_limit\n                        else \"[blue]None\"\n                    ),\n                ]\n                if verbose:\n                    row.append(queue.description)\n                table.add_row(*row)\n\n    app.console.print(table)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.pause","title":"pause async","text":"

    Pause a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def pause(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue to pause\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Pause a work queue.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n\n    async with get_client() as client:\n        try:\n            await client.update_work_queue(\n                id=queue_id,\n                is_paused=True,\n            )\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n\n    if pool:\n        success_message = f\"Work queue {name!r} in work pool {pool!r} paused\"\n    else:\n        success_message = f\"Work queue {name!r} paused\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.preview","title":"preview async","text":"

    Preview a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def preview(\n    name: str = typer.Argument(\n        None, help=\"The name or ID of the work queue to preview\"\n    ),\n    hours: int = typer.Option(\n        None,\n        \"-h\",\n        \"--hours\",\n        help=\"The number of hours to look ahead; defaults to 1 hour\",\n    ),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Preview a work queue.\n    \"\"\"\n    if pool:\n        title = f\"Preview of Work Queue {name!r} in Work Pool {pool!r}\"\n    else:\n        title = f\"Preview of Work Queue {name!r}\"\n\n    table = Table(title=title, caption=\"(**) denotes a late run\", caption_style=\"red\")\n    table.add_column(\n        \"Scheduled Start Time\", justify=\"left\", style=\"yellow\", no_wrap=True\n    )\n    table.add_column(\"Run ID\", justify=\"left\", style=\"cyan\", no_wrap=True)\n    table.add_column(\"Name\", style=\"green\", no_wrap=True)\n    table.add_column(\"Deployment ID\", style=\"blue\", no_wrap=True)\n\n    window = pendulum.now(\"utc\").add(hours=hours or 1)\n\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name, work_pool_name=pool\n    )\n    async with get_client() as client:\n        if pool:\n            try:\n                responses = await client.get_scheduled_flow_runs_for_work_pool(\n                    work_pool_name=pool,\n                    work_queue_names=[name],\n                )\n                runs = [response.flow_run for response in responses]\n            except ObjectNotFound:\n                exit_with_error(f\"No work queue found: {name!r} in work pool {pool!r}\")\n        else:\n            try:\n                runs = await client.get_runs_in_work_queue(\n                    queue_id,\n                    limit=10,\n                    scheduled_before=window,\n                )\n            except ObjectNotFound:\n                exit_with_error(f\"No work queue found: {name!r}\")\n    now = pendulum.now(\"utc\")\n\n    def sort_by_created_key(r):\n        return now - r.created\n\n    for run in sorted(runs, key=sort_by_created_key):\n        table.add_row(\n            (\n                f\"{run.expected_start_time} [red](**)\"\n                if run.expected_start_time < now\n                else f\"{run.expected_start_time}\"\n            ),\n            str(run.id),\n            run.name,\n            str(run.deployment_id),\n        )\n\n    if runs:\n        app.console.print(table)\n    else:\n        app.console.print(\n            (\n                \"No runs found - try increasing how far into the future you preview\"\n                \" with the --hours flag\"\n            ),\n            style=\"yellow\",\n        )\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.resume","title":"resume async","text":"

    Resume a paused work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def resume(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue to resume\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Resume a paused work queue.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n\n    async with get_client() as client:\n        try:\n            await client.update_work_queue(\n                id=queue_id,\n                is_paused=False,\n            )\n        except ObjectNotFound:\n            if pool:\n                error_message = f\"No work queue found: {name!r} in work pool {pool!r}\"\n            else:\n                error_message = f\"No work queue found: {name!r}\"\n            exit_with_error(error_message)\n\n    if pool:\n        success_message = f\"Work queue {name!r} in work pool {pool!r} resumed\"\n    else:\n        success_message = f\"Work queue {name!r} resumed\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/work_queue/#prefect.cli.work_queue.set_concurrency_limit","title":"set_concurrency_limit async","text":"

    Set a concurrency limit on a work queue.

    Source code in prefect/cli/work_queue.py
    @work_app.command()\n@experimental_parameter(\"pool\", group=\"work_pools\", when=lambda y: y is not None)\nasync def set_concurrency_limit(\n    name: str = typer.Argument(..., help=\"The name or ID of the work queue\"),\n    limit: int = typer.Argument(..., help=\"The concurrency limit to set on the queue.\"),\n    pool: Optional[str] = typer.Option(\n        None,\n        \"-p\",\n        \"--pool\",\n        help=\"The name of the work pool that the work queue belongs to.\",\n    ),\n):\n    \"\"\"\n    Set a concurrency limit on a work queue.\n    \"\"\"\n    queue_id = await _get_work_queue_id_from_name_or_id(\n        name_or_id=name,\n        work_pool_name=pool,\n    )\n\n    async with get_client() as client:\n        try:\n            await client.update_work_queue(\n                id=queue_id,\n                concurrency_limit=limit,\n            )\n        except ObjectNotFound:\n            if pool:\n                error_message = (\n                    f\"No work queue named {name!r} found in work pool {pool!r}.\"\n                )\n            else:\n                error_message = f\"No work queue named {name!r} found.\"\n            exit_with_error(error_message)\n\n    if pool:\n        success_message = (\n            f\"Concurrency limit of {limit} set on work queue {name!r} in work pool\"\n            f\" {pool!r}\"\n        )\n    else:\n        success_message = f\"Concurrency limit of {limit} set on work queue {name!r}\"\n    exit_with_success(success_message)\n
    ","tags":["Python API","CLI","work-queue"]},{"location":"api-ref/prefect/cli/worker/","title":"worker","text":"","tags":["Python API","workers","CLI"]},{"location":"api-ref/prefect/cli/worker/#prefect.cli.worker","title":"prefect.cli.worker","text":"","tags":["Python API","workers","CLI"]},{"location":"api-ref/prefect/cli/worker/#prefect.cli.worker.start","title":"start async","text":"

    Start a worker process to poll a work pool for flow runs.

    Source code in prefect/cli/worker.py
    @worker_app.command()\nasync def start(\n    worker_name: str = typer.Option(\n        None,\n        \"-n\",\n        \"--name\",\n        help=(\n            \"The name to give to the started worker. If not provided, a unique name\"\n            \" will be generated.\"\n        ),\n    ),\n    work_pool_name: str = typer.Option(\n        ...,\n        \"-p\",\n        \"--pool\",\n        help=\"The work pool the started worker should poll.\",\n        prompt=True,\n    ),\n    work_queues: List[str] = typer.Option(\n        None,\n        \"-q\",\n        \"--work-queue\",\n        help=(\n            \"One or more work queue names for the worker to pull from. If not provided,\"\n            \" the worker will pull from all work queues in the work pool.\"\n        ),\n    ),\n    worker_type: Optional[str] = typer.Option(\n        None,\n        \"-t\",\n        \"--type\",\n        help=(\n            \"The type of worker to start. If not provided, the worker type will be\"\n            \" inferred from the work pool.\"\n        ),\n    ),\n    prefetch_seconds: int = SettingsOption(\n        PREFECT_WORKER_PREFETCH_SECONDS,\n        help=\"Number of seconds to look into the future for scheduled flow runs.\",\n    ),\n    run_once: bool = typer.Option(\n        False, help=\"Only run worker polling once. By default, the worker runs forever.\"\n    ),\n    limit: int = typer.Option(\n        None,\n        \"-l\",\n        \"--limit\",\n        help=\"Maximum number of flow runs to start simultaneously.\",\n    ),\n    with_healthcheck: bool = typer.Option(\n        False, help=\"Start a healthcheck server for the worker.\"\n    ),\n    install_policy: InstallPolicy = typer.Option(\n        InstallPolicy.PROMPT.value,\n        \"--install-policy\",\n        help=\"Install policy to use workers from Prefect integration packages.\",\n        case_sensitive=False,\n    ),\n    base_job_template: typer.FileText = typer.Option(\n        None,\n        \"--base-job-template\",\n        help=(\n            \"The path to a JSON file containing the base job template to use. If\"\n            \" unspecified, Prefect will use the default base job template for the given\"\n            \" worker type. If the work pool already exists, this will be ignored.\"\n        ),\n    ),\n):\n    \"\"\"\n    Start a worker process to poll a work pool for flow runs.\n    \"\"\"\n\n    is_paused = await _check_work_pool_paused(work_pool_name)\n    if is_paused:\n        app.console.print(\n            (\n                f\"The work pool {work_pool_name!r} is currently paused. This worker\"\n                \" will not execute any flow runs until the work pool is unpaused.\"\n            ),\n            style=\"yellow\",\n        )\n\n    worker_cls = await _get_worker_class(worker_type, work_pool_name, install_policy)\n\n    if worker_cls is None:\n        exit_with_error(\n            \"Unable to start worker. Please ensure you have the necessary dependencies\"\n            \" installed to run your desired worker type.\"\n        )\n\n    worker_process_id = os.getpid()\n    setup_signal_handlers_worker(\n        worker_process_id, f\"the {worker_type} worker\", app.console.print\n    )\n\n    template_contents = None\n    if base_job_template is not None:\n        template_contents = json.load(fp=base_job_template)\n\n    async with worker_cls(\n        name=worker_name,\n        work_pool_name=work_pool_name,\n        work_queues=work_queues,\n        limit=limit,\n        prefetch_seconds=prefetch_seconds,\n        heartbeat_interval_seconds=PREFECT_WORKER_HEARTBEAT_SECONDS.value(),\n        base_job_template=template_contents,\n    ) as worker:\n        app.console.print(f\"Worker {worker.name!r} started!\", style=\"green\")\n        async with anyio.create_task_group() as tg:\n            # wait for an initial heartbeat to configure the worker\n            await worker.sync_with_backend()\n            # schedule the scheduled flow run polling loop\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=worker.get_and_submit_flow_runs,\n                    interval=PREFECT_WORKER_QUERY_SECONDS.value(),\n                    run_once=run_once,\n                    printer=app.console.print,\n                    jitter_range=0.3,\n                    backoff=4,  # Up to ~1 minute interval during backoff\n                )\n            )\n            # schedule the sync loop\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=worker.sync_with_backend,\n                    interval=worker.heartbeat_interval_seconds,\n                    run_once=run_once,\n                    printer=app.console.print,\n                    jitter_range=0.3,\n                    backoff=4,\n                )\n            )\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=worker.check_for_cancelled_flow_runs,\n                    interval=PREFECT_WORKER_QUERY_SECONDS.value() * 2,\n                    run_once=run_once,\n                    printer=app.console.print,\n                    jitter_range=0.3,\n                    backoff=4,\n                )\n            )\n\n            started_event = await worker._emit_worker_started_event()\n\n            # if --with-healthcheck was passed, start the healthcheck server\n            if with_healthcheck:\n                # we'll start the ASGI server in a separate thread so that\n                # uvicorn does not block the main thread\n                server_thread = threading.Thread(\n                    name=\"healthcheck-server-thread\",\n                    target=partial(\n                        start_healthcheck_server,\n                        worker=worker,\n                        query_interval_seconds=PREFECT_WORKER_QUERY_SECONDS.value(),\n                    ),\n                    daemon=True,\n                )\n                server_thread.start()\n\n    await worker._emit_worker_stopped_event(started_event)\n    app.console.print(f\"Worker {worker.name!r} stopped!\")\n
    ","tags":["Python API","workers","CLI"]},{"location":"api-ref/prefect/client/base/","title":"base","text":"","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base","title":"prefect.client.base","text":"","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectResponse","title":"PrefectResponse","text":"

    Bases: Response

    A Prefect wrapper for the httpx.Response class.

    Provides more informative error messages.

    Source code in prefect/client/base.py
    class PrefectResponse(httpx.Response):\n    \"\"\"\n    A Prefect wrapper for the `httpx.Response` class.\n\n    Provides more informative error messages.\n    \"\"\"\n\n    def raise_for_status(self) -> None:\n        \"\"\"\n        Raise an exception if the response contains an HTTPStatusError.\n\n        The `PrefectHTTPStatusError` contains useful additional information that\n        is not contained in the `HTTPStatusError`.\n        \"\"\"\n        try:\n            return super().raise_for_status()\n        except HTTPStatusError as exc:\n            raise PrefectHTTPStatusError.from_httpx_error(exc) from exc.__cause__\n\n    @classmethod\n    def from_httpx_response(cls: Type[Self], response: httpx.Response) -> Self:\n        \"\"\"\n        Create a `PrefectReponse` from an `httpx.Response`.\n\n        By changing the `__class__` attribute of the Response, we change the method\n        resolution order to look for methods defined in PrefectResponse, while leaving\n        everything else about the original Response instance intact.\n        \"\"\"\n        new_response = copy.copy(response)\n        new_response.__class__ = cls\n        return new_response\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectResponse.raise_for_status","title":"raise_for_status","text":"

    Raise an exception if the response contains an HTTPStatusError.

    The PrefectHTTPStatusError contains useful additional information that is not contained in the HTTPStatusError.

    Source code in prefect/client/base.py
    def raise_for_status(self) -> None:\n    \"\"\"\n    Raise an exception if the response contains an HTTPStatusError.\n\n    The `PrefectHTTPStatusError` contains useful additional information that\n    is not contained in the `HTTPStatusError`.\n    \"\"\"\n    try:\n        return super().raise_for_status()\n    except HTTPStatusError as exc:\n        raise PrefectHTTPStatusError.from_httpx_error(exc) from exc.__cause__\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectResponse.from_httpx_response","title":"from_httpx_response classmethod","text":"

    Create a PrefectReponse from an httpx.Response.

    By changing the __class__ attribute of the Response, we change the method resolution order to look for methods defined in PrefectResponse, while leaving everything else about the original Response instance intact.

    Source code in prefect/client/base.py
    @classmethod\ndef from_httpx_response(cls: Type[Self], response: httpx.Response) -> Self:\n    \"\"\"\n    Create a `PrefectReponse` from an `httpx.Response`.\n\n    By changing the `__class__` attribute of the Response, we change the method\n    resolution order to look for methods defined in PrefectResponse, while leaving\n    everything else about the original Response instance intact.\n    \"\"\"\n    new_response = copy.copy(response)\n    new_response.__class__ = cls\n    return new_response\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectHttpxClient","title":"PrefectHttpxClient","text":"

    Bases: AsyncClient

    A Prefect wrapper for the async httpx client with support for retry-after headers for the provided status codes (typically 429, 502 and 503).

    Additionally, this client will always call raise_for_status on responses.

    For more details on rate limit headers, see: Configuring Cloudflare Rate Limiting

    Source code in prefect/client/base.py
    class PrefectHttpxClient(httpx.AsyncClient):\n    \"\"\"\n    A Prefect wrapper for the async httpx client with support for retry-after headers\n    for the provided status codes (typically 429, 502 and 503).\n\n    Additionally, this client will always call `raise_for_status` on responses.\n\n    For more details on rate limit headers, see:\n    [Configuring Cloudflare Rate Limiting](https://support.cloudflare.com/hc/en-us/articles/115001635128-Configuring-Rate-Limiting-from-UI)\n    \"\"\"\n\n    async def _send_with_retry(\n        self,\n        request: Callable,\n        retry_codes: Set[int] = set(),\n        retry_exceptions: Tuple[Exception, ...] = tuple(),\n    ):\n        \"\"\"\n        Send a request and retry it if it fails.\n\n        Sends the provided request and retries it up to PREFECT_CLIENT_MAX_RETRIES times\n        if the request either raises an exception listed in `retry_exceptions` or\n        receives a response with a status code listed in `retry_codes`.\n\n        Retries will be delayed based on either the retry header (preferred) or\n        exponential backoff if a retry header is not provided.\n        \"\"\"\n        try_count = 0\n        response = None\n\n        while try_count <= PREFECT_CLIENT_MAX_RETRIES.value():\n            try_count += 1\n            retry_seconds = None\n            exc_info = None\n\n            try:\n                response = await request()\n            except retry_exceptions:  # type: ignore\n                if try_count > PREFECT_CLIENT_MAX_RETRIES.value():\n                    raise\n                # Otherwise, we will ignore this error but capture the info for logging\n                exc_info = sys.exc_info()\n            else:\n                # We got a response; return immediately if it is not retryable\n                if response.status_code not in retry_codes:\n                    return response\n\n                if \"Retry-After\" in response.headers:\n                    retry_seconds = float(response.headers[\"Retry-After\"])\n\n            # Use an exponential back-off if not set in a header\n            if retry_seconds is None:\n                retry_seconds = 2**try_count\n\n            # Add jitter\n            jitter_factor = PREFECT_CLIENT_RETRY_JITTER_FACTOR.value()\n            if retry_seconds > 0 and jitter_factor > 0:\n                if response is not None and \"Retry-After\" in response.headers:\n                    # Always wait for _at least_ retry seconds if requested by the API\n                    retry_seconds = bounded_poisson_interval(\n                        retry_seconds, retry_seconds * (1 + jitter_factor)\n                    )\n                else:\n                    # Otherwise, use a symmetrical jitter\n                    retry_seconds = clamped_poisson_interval(\n                        retry_seconds, jitter_factor\n                    )\n\n            logger.debug(\n                (\n                    \"Encountered retryable exception during request. \"\n                    if exc_info\n                    else (\n                        \"Received response with retryable status code\"\n                        f\" {response.status_code}. \"\n                    )\n                )\n                + f\"Another attempt will be made in {retry_seconds}s. \"\n                \"This is attempt\"\n                f\" {try_count}/{PREFECT_CLIENT_MAX_RETRIES.value() + 1}.\",\n                exc_info=exc_info,\n            )\n            await anyio.sleep(retry_seconds)\n\n        assert (\n            response is not None\n        ), \"Retry handling ended without response or exception\"\n\n        # We ran out of retries, return the failed response\n        return response\n\n    async def send(self, *args, **kwargs) -> Response:\n        \"\"\"\n        Send a request with automatic retry behavior for the following status codes:\n\n        - 429 CloudFlare-style rate limiting\n        - 502 Bad Gateway\n        - 503 Service unavailable\n        \"\"\"\n\n        api_request = partial(super().send, *args, **kwargs)\n\n        response = await self._send_with_retry(\n            request=api_request,\n            retry_codes={\n                status.HTTP_429_TOO_MANY_REQUESTS,\n                status.HTTP_503_SERVICE_UNAVAILABLE,\n                status.HTTP_502_BAD_GATEWAY,\n                status.HTTP_408_REQUEST_TIMEOUT,\n                *PREFECT_CLIENT_RETRY_EXTRA_CODES.value(),\n            },\n            retry_exceptions=(\n                httpx.ReadTimeout,\n                httpx.PoolTimeout,\n                httpx.ConnectTimeout,\n                # `ConnectionResetError` when reading socket raises as a `ReadError`\n                httpx.ReadError,\n                # Sockets can be closed during writes resulting in a `WriteError`\n                httpx.WriteError,\n                # Uvicorn bug, see https://github.com/PrefectHQ/prefect/issues/7512\n                httpx.RemoteProtocolError,\n                # HTTP2 bug, see https://github.com/PrefectHQ/prefect/issues/7442\n                httpx.LocalProtocolError,\n            ),\n        )\n\n        # Convert to a Prefect response to add nicer errors messages\n        response = PrefectResponse.from_httpx_response(response)\n\n        # Always raise bad responses\n        # NOTE: We may want to remove this and handle responses per route in the\n        #       `PrefectClient`\n        response.raise_for_status()\n\n        return response\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.PrefectHttpxClient.send","title":"send async","text":"

    Send a request with automatic retry behavior for the following status codes:

    • 429 CloudFlare-style rate limiting
    • 502 Bad Gateway
    • 503 Service unavailable
    Source code in prefect/client/base.py
    async def send(self, *args, **kwargs) -> Response:\n    \"\"\"\n    Send a request with automatic retry behavior for the following status codes:\n\n    - 429 CloudFlare-style rate limiting\n    - 502 Bad Gateway\n    - 503 Service unavailable\n    \"\"\"\n\n    api_request = partial(super().send, *args, **kwargs)\n\n    response = await self._send_with_retry(\n        request=api_request,\n        retry_codes={\n            status.HTTP_429_TOO_MANY_REQUESTS,\n            status.HTTP_503_SERVICE_UNAVAILABLE,\n            status.HTTP_502_BAD_GATEWAY,\n            status.HTTP_408_REQUEST_TIMEOUT,\n            *PREFECT_CLIENT_RETRY_EXTRA_CODES.value(),\n        },\n        retry_exceptions=(\n            httpx.ReadTimeout,\n            httpx.PoolTimeout,\n            httpx.ConnectTimeout,\n            # `ConnectionResetError` when reading socket raises as a `ReadError`\n            httpx.ReadError,\n            # Sockets can be closed during writes resulting in a `WriteError`\n            httpx.WriteError,\n            # Uvicorn bug, see https://github.com/PrefectHQ/prefect/issues/7512\n            httpx.RemoteProtocolError,\n            # HTTP2 bug, see https://github.com/PrefectHQ/prefect/issues/7442\n            httpx.LocalProtocolError,\n        ),\n    )\n\n    # Convert to a Prefect response to add nicer errors messages\n    response = PrefectResponse.from_httpx_response(response)\n\n    # Always raise bad responses\n    # NOTE: We may want to remove this and handle responses per route in the\n    #       `PrefectClient`\n    response.raise_for_status()\n\n    return response\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/base/#prefect.client.base.app_lifespan_context","title":"app_lifespan_context async","text":"

    A context manager that calls startup/shutdown hooks for the given application.

    Lifespan contexts are cached per application to avoid calling the lifespan hooks more than once if the context is entered in nested code. A no-op context will be returned if the context for the given application is already being managed.

    This manager is robust to concurrent access within the event loop. For example, if you have concurrent contexts for the same application, it is guaranteed that startup hooks will be called before their context starts and shutdown hooks will only be called after their context exits.

    A reference count is used to support nested use of clients without running lifespan hooks excessively. The first client context entered will create and enter a lifespan context. Each subsequent client will increment a reference count but will not create a new lifespan context. When each client context exits, the reference count is decremented. When the last client context exits, the lifespan will be closed.

    In simple nested cases, the first client context will be the one to exit the lifespan. However, if client contexts are entered concurrently they may not exit in a consistent order. If the first client context was responsible for closing the lifespan, it would have to wait until all other client contexts to exit to avoid firing shutdown hooks while the application is in use. Waiting for the other clients to exit can introduce deadlocks, so, instead, the first client will exit without closing the lifespan context and reference counts will be used to ensure the lifespan is closed once all of the clients are done.

    Source code in prefect/client/base.py
    @asynccontextmanager\nasync def app_lifespan_context(app: ASGIApp) -> AsyncGenerator[None, None]:\n    \"\"\"\n    A context manager that calls startup/shutdown hooks for the given application.\n\n    Lifespan contexts are cached per application to avoid calling the lifespan hooks\n    more than once if the context is entered in nested code. A no-op context will be\n    returned if the context for the given application is already being managed.\n\n    This manager is robust to concurrent access within the event loop. For example,\n    if you have concurrent contexts for the same application, it is guaranteed that\n    startup hooks will be called before their context starts and shutdown hooks will\n    only be called after their context exits.\n\n    A reference count is used to support nested use of clients without running\n    lifespan hooks excessively. The first client context entered will create and enter\n    a lifespan context. Each subsequent client will increment a reference count but will\n    not create a new lifespan context. When each client context exits, the reference\n    count is decremented. When the last client context exits, the lifespan will be\n    closed.\n\n    In simple nested cases, the first client context will be the one to exit the\n    lifespan. However, if client contexts are entered concurrently they may not exit\n    in a consistent order. If the first client context was responsible for closing\n    the lifespan, it would have to wait until all other client contexts to exit to\n    avoid firing shutdown hooks while the application is in use. Waiting for the other\n    clients to exit can introduce deadlocks, so, instead, the first client will exit\n    without closing the lifespan context and reference counts will be used to ensure\n    the lifespan is closed once all of the clients are done.\n    \"\"\"\n    # TODO: A deadlock has been observed during multithreaded use of clients while this\n    #       lifespan context is being used. This has only been reproduced on Python 3.7\n    #       and while we hope to discourage using multiple event loops in threads, this\n    #       bug may emerge again.\n    #       See https://github.com/PrefectHQ/orion/pull/1696\n    thread_id = threading.get_ident()\n\n    # The id of the application is used instead of the hash so each application instance\n    # is managed independently even if they share the same settings. We include the\n    # thread id since applications are managed separately per thread.\n    key = (thread_id, id(app))\n\n    # On exception, this will be populated with exception details\n    exc_info = (None, None, None)\n\n    # Get a lock unique to this thread since anyio locks are not threadsafe\n    lock = APP_LIFESPANS_LOCKS[thread_id]\n\n    async with lock:\n        if key in APP_LIFESPANS:\n            # The lifespan is already being managed, just increment the reference count\n            APP_LIFESPANS_REF_COUNTS[key] += 1\n        else:\n            # Create a new lifespan manager\n            APP_LIFESPANS[key] = context = LifespanManager(\n                app, startup_timeout=30, shutdown_timeout=30\n            )\n            APP_LIFESPANS_REF_COUNTS[key] = 1\n\n            # Ensure we enter the context before releasing the lock so startup hooks\n            # are complete before another client can be used\n            await context.__aenter__()\n\n    try:\n        yield\n    except BaseException:\n        exc_info = sys.exc_info()\n        raise\n    finally:\n        # If we do not shield against anyio cancellation, the lock will return\n        # immediately and the code in its context will not run, leaving the lifespan\n        # open\n        with anyio.CancelScope(shield=True):\n            async with lock:\n                # After the consumer exits the context, decrement the reference count\n                APP_LIFESPANS_REF_COUNTS[key] -= 1\n\n                # If this the last context to exit, close the lifespan\n                if APP_LIFESPANS_REF_COUNTS[key] <= 0:\n                    APP_LIFESPANS_REF_COUNTS.pop(key)\n                    context = APP_LIFESPANS.pop(key)\n                    await context.__aexit__(*exc_info)\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/","title":"cloud","text":"","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud","title":"prefect.client.cloud","text":"","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud.CloudUnauthorizedError","title":"CloudUnauthorizedError","text":"

    Bases: PrefectException

    Raised when the CloudClient receives a 401 or 403 from the Cloud API.

    Source code in prefect/client/cloud.py
    class CloudUnauthorizedError(PrefectException):\n    \"\"\"\n    Raised when the CloudClient receives a 401 or 403 from the Cloud API.\n    \"\"\"\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud.CloudClient","title":"CloudClient","text":"Source code in prefect/client/cloud.py
    class CloudClient:\n    def __init__(\n        self,\n        host: str,\n        api_key: str,\n        httpx_settings: dict = None,\n    ) -> None:\n        httpx_settings = httpx_settings or dict()\n        httpx_settings.setdefault(\"headers\", dict())\n        httpx_settings[\"headers\"].setdefault(\"Authorization\", f\"Bearer {api_key}\")\n\n        httpx_settings.setdefault(\"base_url\", host)\n        if not PREFECT_UNIT_TEST_MODE.value():\n            httpx_settings.setdefault(\"follow_redirects\", True)\n        self._client = PrefectHttpxClient(**httpx_settings)\n\n    async def api_healthcheck(self):\n        \"\"\"\n        Attempts to connect to the Cloud API and raises the encountered exception if not\n        successful.\n\n        If successful, returns `None`.\n        \"\"\"\n        with anyio.fail_after(10):\n            await self.read_workspaces()\n\n    async def read_workspaces(self) -> List[Workspace]:\n        return pydantic.parse_obj_as(List[Workspace], await self.get(\"/me/workspaces\"))\n\n    async def read_worker_metadata(self) -> Dict[str, Any]:\n        configured_url = prefect.settings.PREFECT_API_URL.value()\n        account_id, workspace_id = re.findall(PARSE_API_URL_REGEX, configured_url)[0]\n        return await self.get(\n            f\"accounts/{account_id}/workspaces/{workspace_id}/collections/work_pool_types\"\n        )\n\n    async def __aenter__(self):\n        await self._client.__aenter__()\n        return self\n\n    async def __aexit__(self, *exc_info):\n        return await self._client.__aexit__(*exc_info)\n\n    def __enter__(self):\n        raise RuntimeError(\n            \"The `CloudClient` must be entered with an async context. Use 'async \"\n            \"with CloudClient(...)' not 'with CloudClient(...)'\"\n        )\n\n    def __exit__(self, *_):\n        assert False, \"This should never be called but must be defined for __enter__\"\n\n    async def get(self, route, **kwargs):\n        return await self.request(\"GET\", route, **kwargs)\n\n    async def request(self, method, route, **kwargs):\n        try:\n            res = await self._client.request(method, route, **kwargs)\n            res.raise_for_status()\n        except httpx.HTTPStatusError as exc:\n            if exc.response.status_code in (\n                status.HTTP_401_UNAUTHORIZED,\n                status.HTTP_403_FORBIDDEN,\n            ):\n                raise CloudUnauthorizedError\n            else:\n                raise exc\n\n        if res.status_code == status.HTTP_204_NO_CONTENT:\n            return\n\n        return res.json()\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud.CloudClient.api_healthcheck","title":"api_healthcheck async","text":"

    Attempts to connect to the Cloud API and raises the encountered exception if not successful.

    If successful, returns None.

    Source code in prefect/client/cloud.py
    async def api_healthcheck(self):\n    \"\"\"\n    Attempts to connect to the Cloud API and raises the encountered exception if not\n    successful.\n\n    If successful, returns `None`.\n    \"\"\"\n    with anyio.fail_after(10):\n        await self.read_workspaces()\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/cloud/#prefect.client.cloud.get_cloud_client","title":"get_cloud_client","text":"

    Needs a docstring.

    Source code in prefect/client/cloud.py
    def get_cloud_client(\n    host: Optional[str] = None,\n    api_key: Optional[str] = None,\n    httpx_settings: Optional[dict] = None,\n    infer_cloud_url: bool = False,\n) -> \"CloudClient\":\n    \"\"\"\n    Needs a docstring.\n    \"\"\"\n    if httpx_settings is not None:\n        httpx_settings = httpx_settings.copy()\n\n    if infer_cloud_url is False:\n        host = host or PREFECT_CLOUD_API_URL.value()\n    else:\n        configured_url = prefect.settings.PREFECT_API_URL.value()\n        host = re.sub(PARSE_API_URL_REGEX, \"\", configured_url)\n\n    return CloudClient(\n        host=host,\n        api_key=api_key or PREFECT_API_KEY.value(),\n        httpx_settings=httpx_settings,\n    )\n
    ","tags":["Python API"]},{"location":"api-ref/prefect/client/orchestration/","title":"orchestration","text":"

    Asynchronous client implementation for communicating with the Prefect REST API.

    Explore the client by communicating with an in-memory webserver \u2014 no setup required:

    $ # start python REPL with native await functionality\n$ python -m asyncio\n>>> from prefect import get_client\n>>> async with get_client() as client:\n...     response = await client.hello()\n...     print(response.json())\n\ud83d\udc4b\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration","title":"prefect.client.orchestration","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient","title":"PrefectClient","text":"

    An asynchronous client for interacting with the Prefect REST API.

    Parameters:

    Name Type Description Default api Union[str, ASGIApp]

    the REST API URL or FastAPI application to connect to

    required api_key str

    An optional API key for authentication.

    None api_version str

    The API version this client is compatible with.

    None httpx_settings dict

    An optional dictionary of settings to pass to the underlying httpx.AsyncClient

    None
    Say hello to a Prefect REST API\n\n<div class=\"terminal\">\n```\n>>> async with get_client() as client:\n>>>     response = await client.hello()\n>>>\n>>> print(response.json())\n\ud83d\udc4b\n```\n</div>\n
    Source code in prefect/client/orchestration.py
    class PrefectClient:\n    \"\"\"\n    An asynchronous client for interacting with the [Prefect REST API](/api-ref/rest-api/).\n\n    Args:\n        api: the REST API URL or FastAPI application to connect to\n        api_key: An optional API key for authentication.\n        api_version: The API version this client is compatible with.\n        httpx_settings: An optional dictionary of settings to pass to the underlying\n            `httpx.AsyncClient`\n\n    Examples:\n\n        Say hello to a Prefect REST API\n\n        <div class=\"terminal\">\n        ```\n        >>> async with get_client() as client:\n        >>>     response = await client.hello()\n        >>>\n        >>> print(response.json())\n        \ud83d\udc4b\n        ```\n        </div>\n    \"\"\"\n\n    def __init__(\n        self,\n        api: Union[str, ASGIApp],\n        *,\n        api_key: str = None,\n        api_version: str = None,\n        httpx_settings: dict = None,\n    ) -> None:\n        httpx_settings = httpx_settings.copy() if httpx_settings else {}\n        httpx_settings.setdefault(\"headers\", {})\n\n        if PREFECT_API_TLS_INSECURE_SKIP_VERIFY:\n            httpx_settings.setdefault(\"verify\", False)\n\n        if api_version is None:\n            api_version = SERVER_API_VERSION\n        httpx_settings[\"headers\"].setdefault(\"X-PREFECT-API-VERSION\", api_version)\n        if api_key:\n            httpx_settings[\"headers\"].setdefault(\"Authorization\", f\"Bearer {api_key}\")\n\n        # Context management\n        self._exit_stack = AsyncExitStack()\n        self._ephemeral_app: Optional[ASGIApp] = None\n        self.manage_lifespan = True\n        self.server_type: ServerType\n\n        # Only set if this client started the lifespan of the application\n        self._ephemeral_lifespan: Optional[LifespanManager] = None\n\n        self._closed = False\n        self._started = False\n\n        # Connect to an external application\n        if isinstance(api, str):\n            if httpx_settings.get(\"app\"):\n                raise ValueError(\n                    \"Invalid httpx settings: `app` cannot be set when providing an \"\n                    \"api url. `app` is only for use with ephemeral instances. Provide \"\n                    \"it as the `api` parameter instead.\"\n                )\n            httpx_settings.setdefault(\"base_url\", api)\n\n            # See https://www.python-httpx.org/advanced/#pool-limit-configuration\n            httpx_settings.setdefault(\n                \"limits\",\n                httpx.Limits(\n                    # We see instability when allowing the client to open many connections at once.\n                    # Limiting concurrency results in more stable performance.\n                    max_connections=16,\n                    max_keepalive_connections=8,\n                    # The Prefect Cloud LB will keep connections alive for 30s.\n                    # Only allow the client to keep connections alive for 25s.\n                    keepalive_expiry=25,\n                ),\n            )\n\n            # See https://www.python-httpx.org/http2/\n            # Enabling HTTP/2 support on the client does not necessarily mean that your requests\n            # and responses will be transported over HTTP/2, since both the client and the server\n            # need to support HTTP/2. If you connect to a server that only supports HTTP/1.1 the\n            # client will use a standard HTTP/1.1 connection instead.\n            httpx_settings.setdefault(\"http2\", PREFECT_API_ENABLE_HTTP2.value())\n\n            self.server_type = (\n                ServerType.CLOUD\n                if api.startswith(PREFECT_CLOUD_API_URL.value())\n                else ServerType.SERVER\n            )\n\n        # Connect to an in-process application\n        elif isinstance(api, ASGIApp):\n            self._ephemeral_app = api\n            self.server_type = ServerType.EPHEMERAL\n\n            # When using an ephemeral server, server-side exceptions can be raised\n            # client-side breaking all of our response error code handling. To work\n            # around this, we create an ASGI transport with application exceptions\n            # disabled instead of using the application directly.\n            # refs:\n            # - https://github.com/PrefectHQ/prefect/pull/9637\n            # - https://github.com/encode/starlette/blob/d3a11205ed35f8e5a58a711db0ff59c86fa7bb31/starlette/middleware/errors.py#L184\n            # - https://github.com/tiangolo/fastapi/blob/8cc967a7605d3883bd04ceb5d25cc94ae079612f/fastapi/applications.py#L163-L164\n            httpx_settings.setdefault(\n                \"transport\",\n                httpx.ASGITransport(\n                    app=self._ephemeral_app, raise_app_exceptions=False\n                ),\n            )\n            httpx_settings.setdefault(\"base_url\", \"http://ephemeral-prefect/api\")\n\n        else:\n            raise TypeError(\n                f\"Unexpected type {type(api).__name__!r} for argument `api`. Expected\"\n                \" 'str' or 'ASGIApp/FastAPI'\"\n            )\n\n        # See https://www.python-httpx.org/advanced/#timeout-configuration\n        httpx_settings.setdefault(\n            \"timeout\",\n            httpx.Timeout(\n                connect=PREFECT_API_REQUEST_TIMEOUT.value(),\n                read=PREFECT_API_REQUEST_TIMEOUT.value(),\n                write=PREFECT_API_REQUEST_TIMEOUT.value(),\n                pool=PREFECT_API_REQUEST_TIMEOUT.value(),\n            ),\n        )\n\n        if not PREFECT_UNIT_TEST_MODE:\n            httpx_settings.setdefault(\"follow_redirects\", True)\n        self._client = PrefectHttpxClient(**httpx_settings)\n        self._loop = None\n\n        # See https://www.python-httpx.org/advanced/#custom-transports\n        #\n        # If we're using an HTTP/S client (not the ephemeral client), adjust the\n        # transport to add retries _after_ it is instantiated. If we alter the transport\n        # before instantiation, the transport will not be aware of proxies unless we\n        # reproduce all of the logic to make it so.\n        #\n        # Only alter the transport to set our default of 3 retries, don't modify any\n        # transport a user may have provided via httpx_settings.\n        #\n        # Making liberal use of getattr and isinstance checks here to avoid any\n        # surprises if the internals of httpx or httpcore change on us\n        if isinstance(api, str) and not httpx_settings.get(\"transport\"):\n            transport_for_url = getattr(self._client, \"_transport_for_url\", None)\n            if callable(transport_for_url):\n                server_transport = transport_for_url(httpx.URL(api))\n                if isinstance(server_transport, httpx.AsyncHTTPTransport):\n                    pool = getattr(server_transport, \"_pool\", None)\n                    if isinstance(pool, httpcore.AsyncConnectionPool):\n                        pool._retries = 3\n\n        self.logger = get_logger(\"client\")\n\n    @property\n    def api_url(self) -> httpx.URL:\n        \"\"\"\n        Get the base URL for the API.\n        \"\"\"\n        return self._client.base_url\n\n    # API methods ----------------------------------------------------------------------\n\n    async def api_healthcheck(self) -> Optional[Exception]:\n        \"\"\"\n        Attempts to connect to the API and returns the encountered exception if not\n        successful.\n\n        If successful, returns `None`.\n        \"\"\"\n        try:\n            await self._client.get(\"/health\")\n            return None\n        except Exception as exc:\n            return exc\n\n    async def hello(self) -> httpx.Response:\n        \"\"\"\n        Send a GET request to /hello for testing purposes.\n        \"\"\"\n        return await self._client.get(\"/hello\")\n\n    async def create_flow(self, flow: \"FlowObject\") -> UUID:\n        \"\"\"\n        Create a flow in the Prefect API.\n\n        Args:\n            flow: a [Flow][prefect.flows.Flow] object\n\n        Raises:\n            httpx.RequestError: if a flow was not created for any reason\n\n        Returns:\n            the ID of the flow in the backend\n        \"\"\"\n        return await self.create_flow_from_name(flow.name)\n\n    async def create_flow_from_name(self, flow_name: str) -> UUID:\n        \"\"\"\n        Create a flow in the Prefect API.\n\n        Args:\n            flow_name: the name of the new flow\n\n        Raises:\n            httpx.RequestError: if a flow was not created for any reason\n\n        Returns:\n            the ID of the flow in the backend\n        \"\"\"\n        flow_data = FlowCreate(name=flow_name)\n        response = await self._client.post(\n            \"/flows/\", json=flow_data.dict(json_compatible=True)\n        )\n\n        flow_id = response.json().get(\"id\")\n        if not flow_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        # Return the id of the created flow\n        return UUID(flow_id)\n\n    async def read_flow(self, flow_id: UUID) -> Flow:\n        \"\"\"\n        Query the Prefect API for a flow by id.\n\n        Args:\n            flow_id: the flow ID of interest\n\n        Returns:\n            a [Flow model][prefect.client.schemas.objects.Flow] representation of the flow\n        \"\"\"\n        response = await self._client.get(f\"/flows/{flow_id}\")\n        return Flow.parse_obj(response.json())\n\n    async def read_flows(\n        self,\n        *,\n        flow_filter: FlowFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        deployment_filter: DeploymentFilter = None,\n        work_pool_filter: WorkPoolFilter = None,\n        work_queue_filter: WorkQueueFilter = None,\n        sort: FlowSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[Flow]:\n        \"\"\"\n        Query the Prefect API for flows. Only flows matching all criteria will\n        be returned.\n\n        Args:\n            flow_filter: filter criteria for flows\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            deployment_filter: filter criteria for deployments\n            work_pool_filter: filter criteria for work pools\n            work_queue_filter: filter criteria for work pool queues\n            sort: sort criteria for the flows\n            limit: limit for the flow query\n            offset: offset for the flow query\n\n        Returns:\n            a list of Flow model representations of the flows\n        \"\"\"\n        body = {\n            \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n                if flow_run_filter\n                else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"deployments\": (\n                deployment_filter.dict(json_compatible=True)\n                if deployment_filter\n                else None\n            ),\n            \"work_pools\": (\n                work_pool_filter.dict(json_compatible=True)\n                if work_pool_filter\n                else None\n            ),\n            \"work_queues\": (\n                work_queue_filter.dict(json_compatible=True)\n                if work_queue_filter\n                else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n\n        response = await self._client.post(\"/flows/filter\", json=body)\n        return pydantic.parse_obj_as(List[Flow], response.json())\n\n    async def read_flow_by_name(\n        self,\n        flow_name: str,\n    ) -> Flow:\n        \"\"\"\n        Query the Prefect API for a flow by name.\n\n        Args:\n            flow_name: the name of a flow\n\n        Returns:\n            a fully hydrated Flow model\n        \"\"\"\n        response = await self._client.get(f\"/flows/name/{flow_name}\")\n        return Flow.parse_obj(response.json())\n\n    async def create_flow_run_from_deployment(\n        self,\n        deployment_id: UUID,\n        *,\n        parameters: Dict[str, Any] = None,\n        context: dict = None,\n        state: prefect.states.State = None,\n        name: str = None,\n        tags: Iterable[str] = None,\n        idempotency_key: str = None,\n        parent_task_run_id: UUID = None,\n        work_queue_name: str = None,\n    ) -> FlowRun:\n        \"\"\"\n        Create a flow run for a deployment.\n\n        Args:\n            deployment_id: The deployment ID to create the flow run from\n            parameters: Parameter overrides for this flow run. Merged with the\n                deployment defaults\n            context: Optional run context data\n            state: The initial state for the run. If not provided, defaults to\n                `Scheduled` for now. Should always be a `Scheduled` type.\n            name: An optional name for the flow run. If not provided, the server will\n                generate a name.\n            tags: An optional iterable of tags to apply to the flow run; these tags\n                are merged with the deployment's tags.\n            idempotency_key: Optional idempotency key for creation of the flow run.\n                If the key matches the key of an existing flow run, the existing run will\n                be returned instead of creating a new one.\n            parent_task_run_id: if a subflow run is being created, the placeholder task\n                run identifier in the parent flow\n            work_queue_name: An optional work queue name to add this run to. If not provided,\n                will default to the deployment's set work queue.  If one is provided that does not\n                exist, a new work queue will be created within the deployment's work pool.\n\n        Raises:\n            httpx.RequestError: if the Prefect API does not successfully create a run for any reason\n\n        Returns:\n            The flow run model\n        \"\"\"\n        parameters = parameters or {}\n        context = context or {}\n        state = state or prefect.states.Scheduled()\n        tags = tags or []\n\n        flow_run_create = DeploymentFlowRunCreate(\n            parameters=parameters,\n            context=context,\n            state=state.to_state_create(),\n            tags=tags,\n            name=name,\n            idempotency_key=idempotency_key,\n            parent_task_run_id=parent_task_run_id,\n        )\n\n        # done separately to avoid including this field in payloads sent to older API versions\n        if work_queue_name:\n            flow_run_create.work_queue_name = work_queue_name\n\n        response = await self._client.post(\n            f\"/deployments/{deployment_id}/create_flow_run\",\n            json=flow_run_create.dict(json_compatible=True, exclude_unset=True),\n        )\n        return FlowRun.parse_obj(response.json())\n\n    async def create_flow_run(\n        self,\n        flow: \"FlowObject\",\n        name: str = None,\n        parameters: Dict[str, Any] = None,\n        context: dict = None,\n        tags: Iterable[str] = None,\n        parent_task_run_id: UUID = None,\n        state: \"prefect.states.State\" = None,\n    ) -> FlowRun:\n        \"\"\"\n        Create a flow run for a flow.\n\n        Args:\n            flow: The flow model to create the flow run for\n            name: An optional name for the flow run\n            parameters: Parameter overrides for this flow run.\n            context: Optional run context data\n            tags: a list of tags to apply to this flow run\n            parent_task_run_id: if a subflow run is being created, the placeholder task\n                run identifier in the parent flow\n            state: The initial state for the run. If not provided, defaults to\n                `Scheduled` for now. Should always be a `Scheduled` type.\n\n        Raises:\n            httpx.RequestError: if the Prefect API does not successfully create a run for any reason\n\n        Returns:\n            The flow run model\n        \"\"\"\n        parameters = parameters or {}\n        context = context or {}\n\n        if state is None:\n            state = prefect.states.Pending()\n\n        # Retrieve the flow id\n        flow_id = await self.create_flow(flow)\n\n        flow_run_create = FlowRunCreate(\n            flow_id=flow_id,\n            flow_version=flow.version,\n            name=name,\n            parameters=parameters,\n            context=context,\n            tags=list(tags or []),\n            parent_task_run_id=parent_task_run_id,\n            state=state.to_state_create(),\n            empirical_policy=FlowRunPolicy(\n                retries=flow.retries,\n                retry_delay=flow.retry_delay_seconds,\n            ),\n        )\n\n        flow_run_create_json = flow_run_create.dict(json_compatible=True)\n        response = await self._client.post(\"/flow_runs/\", json=flow_run_create_json)\n        flow_run = FlowRun.parse_obj(response.json())\n\n        # Restore the parameters to the local objects to retain expectations about\n        # Python objects\n        flow_run.parameters = parameters\n\n        return flow_run\n\n    async def update_flow_run(\n        self,\n        flow_run_id: UUID,\n        flow_version: Optional[str] = None,\n        parameters: Optional[dict] = None,\n        name: Optional[str] = None,\n        tags: Optional[Iterable[str]] = None,\n        empirical_policy: Optional[FlowRunPolicy] = None,\n        infrastructure_pid: Optional[str] = None,\n    ) -> httpx.Response:\n        \"\"\"\n        Update a flow run's details.\n\n        Args:\n            flow_run_id: The identifier for the flow run to update.\n            flow_version: A new version string for the flow run.\n            parameters: A dictionary of parameter values for the flow run. This will not\n                be merged with any existing parameters.\n            name: A new name for the flow run.\n            empirical_policy: A new flow run orchestration policy. This will not be\n                merged with any existing policy.\n            tags: An iterable of new tags for the flow run. These will not be merged with\n                any existing tags.\n            infrastructure_pid: The id of flow run as returned by an\n                infrastructure block.\n\n        Returns:\n            an `httpx.Response` object from the PATCH request\n        \"\"\"\n        params = {}\n        if flow_version is not None:\n            params[\"flow_version\"] = flow_version\n        if parameters is not None:\n            params[\"parameters\"] = parameters\n        if name is not None:\n            params[\"name\"] = name\n        if tags is not None:\n            params[\"tags\"] = tags\n        if empirical_policy is not None:\n            params[\"empirical_policy\"] = empirical_policy\n        if infrastructure_pid:\n            params[\"infrastructure_pid\"] = infrastructure_pid\n\n        flow_run_data = FlowRunUpdate(**params)\n\n        return await self._client.patch(\n            f\"/flow_runs/{flow_run_id}\",\n            json=flow_run_data.dict(json_compatible=True, exclude_unset=True),\n        )\n\n    async def delete_flow_run(\n        self,\n        flow_run_id: UUID,\n    ) -> None:\n        \"\"\"\n        Delete a flow run by UUID.\n\n        Args:\n            flow_run_id: The flow run UUID of interest.\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If requests fails\n        \"\"\"\n        try:\n            await self._client.delete(f\"/flow_runs/{flow_run_id}\"),\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def create_concurrency_limit(\n        self,\n        tag: str,\n        concurrency_limit: int,\n    ) -> UUID:\n        \"\"\"\n        Create a tag concurrency limit in the Prefect API. These limits govern concurrently\n        running tasks.\n\n        Args:\n            tag: a tag the concurrency limit is applied to\n            concurrency_limit: the maximum number of concurrent task runs for a given tag\n\n        Raises:\n            httpx.RequestError: if the concurrency limit was not created for any reason\n\n        Returns:\n            the ID of the concurrency limit in the backend\n        \"\"\"\n\n        concurrency_limit_create = ConcurrencyLimitCreate(\n            tag=tag,\n            concurrency_limit=concurrency_limit,\n        )\n        response = await self._client.post(\n            \"/concurrency_limits/\",\n            json=concurrency_limit_create.dict(json_compatible=True),\n        )\n\n        concurrency_limit_id = response.json().get(\"id\")\n\n        if not concurrency_limit_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        return UUID(concurrency_limit_id)\n\n    async def read_concurrency_limit_by_tag(\n        self,\n        tag: str,\n    ):\n        \"\"\"\n        Read the concurrency limit set on a specific tag.\n\n        Args:\n            tag: a tag the concurrency limit is applied to\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: if the concurrency limit was not created for any reason\n\n        Returns:\n            the concurrency limit set on a specific tag\n        \"\"\"\n        try:\n            response = await self._client.get(\n                f\"/concurrency_limits/tag/{tag}\",\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n        concurrency_limit_id = response.json().get(\"id\")\n\n        if not concurrency_limit_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        concurrency_limit = ConcurrencyLimit.parse_obj(response.json())\n        return concurrency_limit\n\n    async def read_concurrency_limits(\n        self,\n        limit: int,\n        offset: int,\n    ):\n        \"\"\"\n        Lists concurrency limits set on task run tags.\n\n        Args:\n            limit: the maximum number of concurrency limits returned\n            offset: the concurrency limit query offset\n\n        Returns:\n            a list of concurrency limits\n        \"\"\"\n\n        body = {\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n\n        response = await self._client.post(\"/concurrency_limits/filter\", json=body)\n        return pydantic.parse_obj_as(List[ConcurrencyLimit], response.json())\n\n    async def reset_concurrency_limit_by_tag(\n        self,\n        tag: str,\n        slot_override: Optional[List[Union[UUID, str]]] = None,\n    ):\n        \"\"\"\n        Resets the concurrency limit slots set on a specific tag.\n\n        Args:\n            tag: a tag the concurrency limit is applied to\n            slot_override: a list of task run IDs that are currently using a\n                concurrency slot, please check that any task run IDs included in\n                `slot_override` are currently running, otherwise those concurrency\n                slots will never be released.\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        \"\"\"\n        if slot_override is not None:\n            slot_override = [str(slot) for slot in slot_override]\n\n        try:\n            await self._client.post(\n                f\"/concurrency_limits/tag/{tag}/reset\",\n                json=dict(slot_override=slot_override),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_concurrency_limit_by_tag(\n        self,\n        tag: str,\n    ):\n        \"\"\"\n        Delete the concurrency limit set on a specific tag.\n\n        Args:\n            tag: a tag the concurrency limit is applied to\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        \"\"\"\n        try:\n            await self._client.delete(\n                f\"/concurrency_limits/tag/{tag}\",\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def create_work_queue(\n        self,\n        name: str,\n        tags: Optional[List[str]] = None,\n        description: Optional[str] = None,\n        is_paused: Optional[bool] = None,\n        concurrency_limit: Optional[int] = None,\n        priority: Optional[int] = None,\n        work_pool_name: Optional[str] = None,\n    ) -> WorkQueue:\n        \"\"\"\n        Create a work queue.\n\n        Args:\n            name: a unique name for the work queue\n            tags: DEPRECATED: an optional list of tags to filter on; only work scheduled with these tags\n                will be included in the queue. This option will be removed on 2023-02-23.\n            description: An optional description for the work queue.\n            is_paused: Whether or not the work queue is paused.\n            concurrency_limit: An optional concurrency limit for the work queue.\n            priority: The queue's priority. Lower values are higher priority (1 is the highest).\n            work_pool_name: The name of the work pool to use for this queue.\n\n        Raises:\n            prefect.exceptions.ObjectAlreadyExists: If request returns 409\n            httpx.RequestError: If request fails\n\n        Returns:\n            The created work queue\n        \"\"\"\n        if tags:\n            warnings.warn(\n                (\n                    \"The use of tags for creating work queue filters is deprecated.\"\n                    \" This option will be removed on 2023-02-23.\"\n                ),\n                DeprecationWarning,\n            )\n            filter = QueueFilter(tags=tags)\n        else:\n            filter = None\n        create_model = WorkQueueCreate(name=name, filter=filter)\n        if description is not None:\n            create_model.description = description\n        if is_paused is not None:\n            create_model.is_paused = is_paused\n        if concurrency_limit is not None:\n            create_model.concurrency_limit = concurrency_limit\n        if priority is not None:\n            create_model.priority = priority\n\n        data = create_model.dict(json_compatible=True)\n        try:\n            if work_pool_name is not None:\n                response = await self._client.post(\n                    f\"/work_pools/{work_pool_name}/queues\", json=data\n                )\n            else:\n                response = await self._client.post(\"/work_queues/\", json=data)\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            elif e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return WorkQueue.parse_obj(response.json())\n\n    async def read_work_queue_by_name(\n        self,\n        name: str,\n        work_pool_name: Optional[str] = None,\n    ) -> WorkQueue:\n        \"\"\"\n        Read a work queue by name.\n\n        Args:\n            name (str): a unique name for the work queue\n            work_pool_name (str, optional): the name of the work pool\n                the queue belongs to.\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: if no work queue is found\n            httpx.HTTPStatusError: other status errors\n\n        Returns:\n            WorkQueue: a work queue API object\n        \"\"\"\n        try:\n            if work_pool_name is not None:\n                response = await self._client.get(\n                    f\"/work_pools/{work_pool_name}/queues/{name}\"\n                )\n            else:\n                response = await self._client.get(f\"/work_queues/name/{name}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n        return WorkQueue.parse_obj(response.json())\n\n    async def update_work_queue(self, id: UUID, **kwargs):\n        \"\"\"\n        Update properties of a work queue.\n\n        Args:\n            id: the ID of the work queue to update\n            **kwargs: the fields to update\n\n        Raises:\n            ValueError: if no kwargs are provided\n            prefect.exceptions.ObjectNotFound: if request returns 404\n            httpx.RequestError: if the request fails\n\n        \"\"\"\n        if not kwargs:\n            raise ValueError(\"No fields provided to update.\")\n\n        data = WorkQueueUpdate(**kwargs).dict(json_compatible=True, exclude_unset=True)\n        try:\n            await self._client.patch(f\"/work_queues/{id}\", json=data)\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def get_runs_in_work_queue(\n        self,\n        id: UUID,\n        limit: int = 10,\n        scheduled_before: datetime.datetime = None,\n    ) -> List[FlowRun]:\n        \"\"\"\n        Read flow runs off a work queue.\n\n        Args:\n            id: the id of the work queue to read from\n            limit: a limit on the number of runs to return\n            scheduled_before: a timestamp; only runs scheduled before this time will be returned.\n                Defaults to now.\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        Returns:\n            List[FlowRun]: a list of FlowRun objects read from the queue\n        \"\"\"\n        if scheduled_before is None:\n            scheduled_before = pendulum.now(\"UTC\")\n\n        try:\n            response = await self._client.post(\n                f\"/work_queues/{id}/get_runs\",\n                json={\n                    \"limit\": limit,\n                    \"scheduled_before\": scheduled_before.isoformat(),\n                },\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return pydantic.parse_obj_as(List[FlowRun], response.json())\n\n    async def read_work_queue(\n        self,\n        id: UUID,\n    ) -> WorkQueue:\n        \"\"\"\n        Read a work queue.\n\n        Args:\n            id: the id of the work queue to load\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        Returns:\n            WorkQueue: an instantiated WorkQueue object\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/work_queues/{id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return WorkQueue.parse_obj(response.json())\n\n    async def read_work_queue_status(\n        self,\n        id: UUID,\n    ) -> WorkQueueStatusDetail:\n        \"\"\"\n        Read a work queue status.\n\n        Args:\n            id: the id of the work queue to load\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        Returns:\n            WorkQueueStatus: an instantiated WorkQueueStatus object\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/work_queues/{id}/status\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return WorkQueueStatusDetail.parse_obj(response.json())\n\n    async def match_work_queues(\n        self,\n        prefixes: List[str],\n        work_pool_name: Optional[str] = None,\n    ) -> List[WorkQueue]:\n        \"\"\"\n        Query the Prefect API for work queues with names with a specific prefix.\n\n        Args:\n            prefixes: a list of strings used to match work queue name prefixes\n            work_pool_name: an optional work pool name to scope the query to\n\n        Returns:\n            a list of WorkQueue model representations\n                of the work queues\n        \"\"\"\n        page_length = 100\n        current_page = 0\n        work_queues = []\n\n        while True:\n            new_queues = await self.read_work_queues(\n                work_pool_name=work_pool_name,\n                offset=current_page * page_length,\n                limit=page_length,\n                work_queue_filter=WorkQueueFilter(\n                    name=WorkQueueFilterName(startswith_=prefixes)\n                ),\n            )\n            if not new_queues:\n                break\n            work_queues += new_queues\n            current_page += 1\n\n        return work_queues\n\n    async def delete_work_queue_by_id(\n        self,\n        id: UUID,\n    ):\n        \"\"\"\n        Delete a work queue by its ID.\n\n        Args:\n            id: the id of the work queue to delete\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If requests fails\n        \"\"\"\n        try:\n            await self._client.delete(\n                f\"/work_queues/{id}\",\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def create_block_type(self, block_type: BlockTypeCreate) -> BlockType:\n        \"\"\"\n        Create a block type in the Prefect API.\n        \"\"\"\n        try:\n            response = await self._client.post(\n                \"/block_types/\",\n                json=block_type.dict(\n                    json_compatible=True, exclude_unset=True, exclude={\"id\"}\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            else:\n                raise\n        return BlockType.parse_obj(response.json())\n\n    async def create_block_schema(self, block_schema: BlockSchemaCreate) -> BlockSchema:\n        \"\"\"\n        Create a block schema in the Prefect API.\n        \"\"\"\n        try:\n            response = await self._client.post(\n                \"/block_schemas/\",\n                json=block_schema.dict(\n                    json_compatible=True,\n                    exclude_unset=True,\n                    exclude={\"id\", \"block_type\", \"checksum\"},\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            else:\n                raise\n        return BlockSchema.parse_obj(response.json())\n\n    async def create_block_document(\n        self,\n        block_document: Union[BlockDocument, BlockDocumentCreate],\n        include_secrets: bool = True,\n    ) -> BlockDocument:\n        \"\"\"\n        Create a block document in the Prefect API. This data is used to configure a\n        corresponding Block.\n\n        Args:\n            include_secrets (bool): whether to include secret values\n                on the stored Block, corresponding to Pydantic's `SecretStr` and\n                `SecretBytes` fields. Note Blocks may not work as expected if\n                this is set to `False`.\n        \"\"\"\n        if isinstance(block_document, BlockDocument):\n            block_document = BlockDocumentCreate.parse_obj(\n                block_document.dict(\n                    json_compatible=True,\n                    include_secrets=include_secrets,\n                    exclude_unset=True,\n                    exclude={\"id\", \"block_schema\", \"block_type\"},\n                ),\n            )\n\n        try:\n            response = await self._client.post(\n                \"/block_documents/\",\n                json=block_document.dict(\n                    json_compatible=True,\n                    include_secrets=include_secrets,\n                    exclude_unset=True,\n                    exclude={\"id\", \"block_schema\", \"block_type\"},\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            else:\n                raise\n        return BlockDocument.parse_obj(response.json())\n\n    async def update_block_document(\n        self,\n        block_document_id: UUID,\n        block_document: BlockDocumentUpdate,\n    ):\n        \"\"\"\n        Update a block document in the Prefect API.\n        \"\"\"\n        try:\n            await self._client.patch(\n                f\"/block_documents/{block_document_id}\",\n                json=block_document.dict(\n                    json_compatible=True,\n                    exclude_unset=True,\n                    include={\"data\", \"merge_existing_data\", \"block_schema_id\"},\n                    include_secrets=True,\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_block_document(self, block_document_id: UUID):\n        \"\"\"\n        Delete a block document.\n        \"\"\"\n        try:\n            await self._client.delete(f\"/block_documents/{block_document_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_block_type_by_slug(self, slug: str) -> BlockType:\n        \"\"\"\n        Read a block type by its slug.\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/block_types/slug/{slug}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return BlockType.parse_obj(response.json())\n\n    async def read_block_schema_by_checksum(\n        self, checksum: str, version: Optional[str] = None\n    ) -> BlockSchema:\n        \"\"\"\n        Look up a block schema checksum\n        \"\"\"\n        try:\n            url = f\"/block_schemas/checksum/{checksum}\"\n            if version is not None:\n                url = f\"{url}?version={version}\"\n            response = await self._client.get(url)\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return BlockSchema.parse_obj(response.json())\n\n    async def update_block_type(self, block_type_id: UUID, block_type: BlockTypeUpdate):\n        \"\"\"\n        Update a block document in the Prefect API.\n        \"\"\"\n        try:\n            await self._client.patch(\n                f\"/block_types/{block_type_id}\",\n                json=block_type.dict(\n                    json_compatible=True,\n                    exclude_unset=True,\n                    include=BlockTypeUpdate.updatable_fields(),\n                    include_secrets=True,\n                ),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_block_type(self, block_type_id: UUID):\n        \"\"\"\n        Delete a block type.\n        \"\"\"\n        try:\n            await self._client.delete(f\"/block_types/{block_type_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            elif (\n                e.response.status_code == status.HTTP_403_FORBIDDEN\n                and e.response.json()[\"detail\"]\n                == \"protected block types cannot be deleted.\"\n            ):\n                raise prefect.exceptions.ProtectedBlockError(\n                    \"Protected block types cannot be deleted.\"\n                ) from e\n            else:\n                raise\n\n    async def read_block_types(self) -> List[BlockType]:\n        \"\"\"\n        Read all block types\n        Raises:\n            httpx.RequestError: if the block types were not found\n\n        Returns:\n            List of BlockTypes.\n        \"\"\"\n        response = await self._client.post(\"/block_types/filter\", json={})\n        return pydantic.parse_obj_as(List[BlockType], response.json())\n\n    async def read_block_schemas(self) -> List[BlockSchema]:\n        \"\"\"\n        Read all block schemas\n        Raises:\n            httpx.RequestError: if a valid block schema was not found\n\n        Returns:\n            A BlockSchema.\n        \"\"\"\n        response = await self._client.post(\"/block_schemas/filter\", json={})\n        return pydantic.parse_obj_as(List[BlockSchema], response.json())\n\n    async def get_most_recent_block_schema_for_block_type(\n        self,\n        block_type_id: UUID,\n    ) -> Optional[BlockSchema]:\n        \"\"\"\n        Fetches the most recent block schema for a specified block type ID.\n\n        Args:\n            block_type_id: The ID of the block type.\n\n        Raises:\n            httpx.RequestError: If the request fails for any reason.\n\n        Returns:\n            The most recent block schema or None.\n        \"\"\"\n        try:\n            response = await self._client.post(\n                \"/block_schemas/filter\",\n                json={\n                    \"block_schemas\": {\"block_type_id\": {\"any_\": [str(block_type_id)]}},\n                    \"limit\": 1,\n                },\n            )\n        except httpx.HTTPStatusError:\n            raise\n        return BlockSchema.parse_obj(response.json()[0]) if response.json() else None\n\n    async def read_block_document(\n        self,\n        block_document_id: UUID,\n        include_secrets: bool = True,\n    ):\n        \"\"\"\n        Read the block document with the specified ID.\n\n        Args:\n            block_document_id: the block document id\n            include_secrets (bool): whether to include secret values\n                on the Block, corresponding to Pydantic's `SecretStr` and\n                `SecretBytes` fields. These fields are automatically obfuscated\n                by Pydantic, but users can additionally choose not to receive\n                their values from the API. Note that any business logic on the\n                Block may not work if this is `False`.\n\n        Raises:\n            httpx.RequestError: if the block document was not found for any reason\n\n        Returns:\n            A block document or None.\n        \"\"\"\n        assert (\n            block_document_id is not None\n        ), \"Unexpected ID on block document. Was it persisted?\"\n        try:\n            response = await self._client.get(\n                f\"/block_documents/{block_document_id}\",\n                params=dict(include_secrets=include_secrets),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return BlockDocument.parse_obj(response.json())\n\n    async def read_block_document_by_name(\n        self,\n        name: str,\n        block_type_slug: str,\n        include_secrets: bool = True,\n    ) -> BlockDocument:\n        \"\"\"\n        Read the block document with the specified name that corresponds to a\n        specific block type name.\n\n        Args:\n            name: The block document name.\n            block_type_slug: The block type slug.\n            include_secrets (bool): whether to include secret values\n                on the Block, corresponding to Pydantic's `SecretStr` and\n                `SecretBytes` fields. These fields are automatically obfuscated\n                by Pydantic, but users can additionally choose not to receive\n                their values from the API. Note that any business logic on the\n                Block may not work if this is `False`.\n\n        Raises:\n            httpx.RequestError: if the block document was not found for any reason\n\n        Returns:\n            A block document or None.\n        \"\"\"\n        try:\n            response = await self._client.get(\n                f\"/block_types/slug/{block_type_slug}/block_documents/name/{name}\",\n                params=dict(include_secrets=include_secrets),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return BlockDocument.parse_obj(response.json())\n\n    async def read_block_documents(\n        self,\n        block_schema_type: Optional[str] = None,\n        offset: Optional[int] = None,\n        limit: Optional[int] = None,\n        include_secrets: bool = True,\n    ):\n        \"\"\"\n        Read block documents\n\n        Args:\n            block_schema_type: an optional block schema type\n            offset: an offset\n            limit: the number of blocks to return\n            include_secrets (bool): whether to include secret values\n                on the Block, corresponding to Pydantic's `SecretStr` and\n                `SecretBytes` fields. These fields are automatically obfuscated\n                by Pydantic, but users can additionally choose not to receive\n                their values from the API. Note that any business logic on the\n                Block may not work if this is `False`.\n\n        Returns:\n            A list of block documents\n        \"\"\"\n        response = await self._client.post(\n            \"/block_documents/filter\",\n            json=dict(\n                block_schema_type=block_schema_type,\n                offset=offset,\n                limit=limit,\n                include_secrets=include_secrets,\n            ),\n        )\n        return pydantic.parse_obj_as(List[BlockDocument], response.json())\n\n    async def read_block_documents_by_type(\n        self,\n        block_type_slug: str,\n        offset: Optional[int] = None,\n        limit: Optional[int] = None,\n        include_secrets: bool = True,\n    ) -> List[BlockDocument]:\n        \"\"\"Retrieve block documents by block type slug.\n\n        Args:\n            block_type_slug: The block type slug.\n            offset: an offset\n            limit: the number of blocks to return\n            include_secrets: whether to include secret values\n\n        Returns:\n            A list of block documents\n        \"\"\"\n        response = await self._client.get(\n            f\"/block_types/slug/{block_type_slug}/block_documents\",\n            params=dict(\n                offset=offset,\n                limit=limit,\n                include_secrets=include_secrets,\n            ),\n        )\n\n        return pydantic.parse_obj_as(List[BlockDocument], response.json())\n\n    async def create_deployment(\n        self,\n        flow_id: UUID,\n        name: str,\n        version: str = None,\n        schedule: SCHEDULE_TYPES = None,\n        parameters: Dict[str, Any] = None,\n        description: str = None,\n        work_queue_name: str = None,\n        work_pool_name: str = None,\n        tags: List[str] = None,\n        storage_document_id: UUID = None,\n        manifest_path: str = None,\n        path: str = None,\n        entrypoint: str = None,\n        infrastructure_document_id: UUID = None,\n        infra_overrides: Dict[str, Any] = None,\n        parameter_openapi_schema: dict = None,\n        is_schedule_active: Optional[bool] = None,\n        pull_steps: Optional[List[dict]] = None,\n        enforce_parameter_schema: Optional[bool] = None,\n    ) -> UUID:\n        \"\"\"\n        Create a deployment.\n\n        Args:\n            flow_id: the flow ID to create a deployment for\n            name: the name of the deployment\n            version: an optional version string for the deployment\n            schedule: an optional schedule to apply to the deployment\n            tags: an optional list of tags to apply to the deployment\n            storage_document_id: an reference to the storage block document\n                used for the deployed flow\n            infrastructure_document_id: an reference to the infrastructure block document\n                to use for this deployment\n\n        Raises:\n            httpx.RequestError: if the deployment was not created for any reason\n\n        Returns:\n            the ID of the deployment in the backend\n        \"\"\"\n        deployment_create = DeploymentCreate(\n            flow_id=flow_id,\n            name=name,\n            version=version,\n            schedule=schedule,\n            parameters=dict(parameters or {}),\n            tags=list(tags or []),\n            work_queue_name=work_queue_name,\n            description=description,\n            storage_document_id=storage_document_id,\n            path=path,\n            entrypoint=entrypoint,\n            manifest_path=manifest_path,  # for backwards compat\n            infrastructure_document_id=infrastructure_document_id,\n            infra_overrides=infra_overrides or {},\n            parameter_openapi_schema=parameter_openapi_schema,\n            is_schedule_active=is_schedule_active,\n            pull_steps=pull_steps,\n            enforce_parameter_schema=enforce_parameter_schema,\n        )\n\n        if work_pool_name is not None:\n            deployment_create.work_pool_name = work_pool_name\n\n        # Exclude newer fields that are not set to avoid compatibility issues\n        exclude = {\n            field\n            for field in [\"work_pool_name\", \"work_queue_name\"]\n            if field not in deployment_create.__fields_set__\n        }\n\n        if deployment_create.is_schedule_active is None:\n            exclude.add(\"is_schedule_active\")\n\n        if deployment_create.pull_steps is None:\n            exclude.add(\"pull_steps\")\n\n        if deployment_create.enforce_parameter_schema is None:\n            exclude.add(\"enforce_parameter_schema\")\n\n        json = deployment_create.dict(json_compatible=True, exclude=exclude)\n        response = await self._client.post(\n            \"/deployments/\",\n            json=json,\n        )\n        deployment_id = response.json().get(\"id\")\n        if not deployment_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        return UUID(deployment_id)\n\n    async def update_schedule(self, deployment_id: UUID, active: bool = True):\n        path = \"set_schedule_active\" if active else \"set_schedule_inactive\"\n        await self._client.post(\n            f\"/deployments/{deployment_id}/{path}\",\n        )\n\n    async def update_deployment(\n        self,\n        deployment: Deployment,\n        schedule: SCHEDULE_TYPES = None,\n        is_schedule_active: bool = None,\n    ):\n        deployment_update = DeploymentUpdate(\n            version=deployment.version,\n            schedule=schedule if schedule is not None else deployment.schedule,\n            is_schedule_active=(\n                is_schedule_active\n                if is_schedule_active is not None\n                else deployment.is_schedule_active\n            ),\n            description=deployment.description,\n            work_queue_name=deployment.work_queue_name,\n            tags=deployment.tags,\n            manifest_path=deployment.manifest_path,\n            path=deployment.path,\n            entrypoint=deployment.entrypoint,\n            parameters=deployment.parameters,\n            storage_document_id=deployment.storage_document_id,\n            infrastructure_document_id=deployment.infrastructure_document_id,\n            infra_overrides=deployment.infra_overrides,\n            enforce_parameter_schema=deployment.enforce_parameter_schema,\n        )\n\n        if getattr(deployment, \"work_pool_name\", None) is not None:\n            deployment_update.work_pool_name = deployment.work_pool_name\n\n        exclude = set()\n        if deployment.enforce_parameter_schema is None:\n            exclude.add(\"enforce_parameter_schema\")\n\n        await self._client.patch(\n            f\"/deployments/{deployment.id}\",\n            json=deployment_update.dict(json_compatible=True, exclude=exclude),\n        )\n\n    async def _create_deployment_from_schema(self, schema: DeploymentCreate) -> UUID:\n        \"\"\"\n        Create a deployment from a prepared `DeploymentCreate` schema.\n        \"\"\"\n        # TODO: We are likely to remove this method once we have considered the\n        #       packaging interface for deployments further.\n        response = await self._client.post(\n            \"/deployments/\", json=schema.dict(json_compatible=True)\n        )\n        deployment_id = response.json().get(\"id\")\n        if not deployment_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        return UUID(deployment_id)\n\n    async def read_deployment(\n        self,\n        deployment_id: UUID,\n    ) -> DeploymentResponse:\n        \"\"\"\n        Query the Prefect API for a deployment by id.\n\n        Args:\n            deployment_id: the deployment ID of interest\n\n        Returns:\n            a [Deployment model][prefect.client.schemas.objects.Deployment] representation of the deployment\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/deployments/{deployment_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return DeploymentResponse.parse_obj(response.json())\n\n    async def read_deployment_by_name(\n        self,\n        name: str,\n    ) -> DeploymentResponse:\n        \"\"\"\n        Query the Prefect API for a deployment by name.\n\n        Args:\n            name: A deployed flow's name: <FLOW_NAME>/<DEPLOYMENT_NAME>\n\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If request fails\n\n        Returns:\n            a Deployment model representation of the deployment\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/deployments/name/{name}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n        return DeploymentResponse.parse_obj(response.json())\n\n    async def read_deployments(\n        self,\n        *,\n        flow_filter: FlowFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        deployment_filter: DeploymentFilter = None,\n        work_pool_filter: WorkPoolFilter = None,\n        work_queue_filter: WorkQueueFilter = None,\n        limit: int = None,\n        sort: DeploymentSort = None,\n        offset: int = 0,\n    ) -> List[DeploymentResponse]:\n        \"\"\"\n        Query the Prefect API for deployments. Only deployments matching all\n        the provided criteria will be returned.\n\n        Args:\n            flow_filter: filter criteria for flows\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            deployment_filter: filter criteria for deployments\n            work_pool_filter: filter criteria for work pools\n            work_queue_filter: filter criteria for work pool queues\n            limit: a limit for the deployment query\n            offset: an offset for the deployment query\n\n        Returns:\n            a list of Deployment model representations\n                of the deployments\n        \"\"\"\n        body = {\n            \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n                if flow_run_filter\n                else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"deployments\": (\n                deployment_filter.dict(json_compatible=True)\n                if deployment_filter\n                else None\n            ),\n            \"work_pools\": (\n                work_pool_filter.dict(json_compatible=True)\n                if work_pool_filter\n                else None\n            ),\n            \"work_pool_queues\": (\n                work_queue_filter.dict(json_compatible=True)\n                if work_queue_filter\n                else None\n            ),\n            \"limit\": limit,\n            \"offset\": offset,\n            \"sort\": sort,\n        }\n\n        response = await self._client.post(\"/deployments/filter\", json=body)\n        return pydantic.parse_obj_as(List[DeploymentResponse], response.json())\n\n    async def delete_deployment(\n        self,\n        deployment_id: UUID,\n    ):\n        \"\"\"\n        Delete deployment by id.\n\n        Args:\n            deployment_id: The deployment id of interest.\n        Raises:\n            prefect.exceptions.ObjectNotFound: If request returns 404\n            httpx.RequestError: If requests fails\n        \"\"\"\n        try:\n            await self._client.delete(f\"/deployments/{deployment_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_flow_run(self, flow_run_id: UUID) -> FlowRun:\n        \"\"\"\n        Query the Prefect API for a flow run by id.\n\n        Args:\n            flow_run_id: the flow run ID of interest\n\n        Returns:\n            a Flow Run model representation of the flow run\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/flow_runs/{flow_run_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n        return FlowRun.parse_obj(response.json())\n\n    async def resume_flow_run(\n        self, flow_run_id: UUID, run_input: Optional[Dict] = None\n    ) -> OrchestrationResult:\n        \"\"\"\n        Resumes a paused flow run.\n\n        Args:\n            flow_run_id: the flow run ID of interest\n            run_input: the input to resume the flow run with\n\n        Returns:\n            an OrchestrationResult model representation of state orchestration output\n        \"\"\"\n        try:\n            response = await self._client.post(\n                f\"/flow_runs/{flow_run_id}/resume\", json={\"run_input\": run_input}\n            )\n        except httpx.HTTPStatusError:\n            raise\n\n        return OrchestrationResult.parse_obj(response.json())\n\n    async def read_flow_runs(\n        self,\n        *,\n        flow_filter: FlowFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        deployment_filter: DeploymentFilter = None,\n        work_pool_filter: WorkPoolFilter = None,\n        work_queue_filter: WorkQueueFilter = None,\n        sort: FlowRunSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[FlowRun]:\n        \"\"\"\n        Query the Prefect API for flow runs. Only flow runs matching all criteria will\n        be returned.\n\n        Args:\n            flow_filter: filter criteria for flows\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            deployment_filter: filter criteria for deployments\n            work_pool_filter: filter criteria for work pools\n            work_queue_filter: filter criteria for work pool queues\n            sort: sort criteria for the flow runs\n            limit: limit for the flow run query\n            offset: offset for the flow run query\n\n        Returns:\n            a list of Flow Run model representations\n                of the flow runs\n        \"\"\"\n        body = {\n            \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n                if flow_run_filter\n                else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"deployments\": (\n                deployment_filter.dict(json_compatible=True)\n                if deployment_filter\n                else None\n            ),\n            \"work_pools\": (\n                work_pool_filter.dict(json_compatible=True)\n                if work_pool_filter\n                else None\n            ),\n            \"work_pool_queues\": (\n                work_queue_filter.dict(json_compatible=True)\n                if work_queue_filter\n                else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n\n        response = await self._client.post(\"/flow_runs/filter\", json=body)\n        return pydantic.parse_obj_as(List[FlowRun], response.json())\n\n    async def set_flow_run_state(\n        self,\n        flow_run_id: UUID,\n        state: \"prefect.states.State\",\n        force: bool = False,\n    ) -> OrchestrationResult:\n        \"\"\"\n        Set the state of a flow run.\n\n        Args:\n            flow_run_id: the id of the flow run\n            state: the state to set\n            force: if True, disregard orchestration logic when setting the state,\n                forcing the Prefect API to accept the state\n\n        Returns:\n            an OrchestrationResult model representation of state orchestration output\n        \"\"\"\n        state_create = state.to_state_create()\n        state_create.state_details.flow_run_id = flow_run_id\n        try:\n            response = await self._client.post(\n                f\"/flow_runs/{flow_run_id}/set_state\",\n                json=dict(state=state_create.dict(json_compatible=True), force=force),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n        return OrchestrationResult.parse_obj(response.json())\n\n    async def read_flow_run_states(\n        self, flow_run_id: UUID\n    ) -> List[prefect.states.State]:\n        \"\"\"\n        Query for the states of a flow run\n\n        Args:\n            flow_run_id: the id of the flow run\n\n        Returns:\n            a list of State model representations\n                of the flow run states\n        \"\"\"\n        response = await self._client.get(\n            \"/flow_run_states/\", params=dict(flow_run_id=str(flow_run_id))\n        )\n        return pydantic.parse_obj_as(List[prefect.states.State], response.json())\n\n    async def set_task_run_name(self, task_run_id: UUID, name: str):\n        task_run_data = TaskRunUpdate(name=name)\n        return await self._client.patch(\n            f\"/task_runs/{task_run_id}\",\n            json=task_run_data.dict(json_compatible=True, exclude_unset=True),\n        )\n\n    async def create_task_run(\n        self,\n        task: \"TaskObject\",\n        flow_run_id: UUID,\n        dynamic_key: str,\n        name: str = None,\n        extra_tags: Iterable[str] = None,\n        state: prefect.states.State = None,\n        task_inputs: Dict[\n            str,\n            List[\n                Union[\n                    TaskRunResult,\n                    Parameter,\n                    Constant,\n                ]\n            ],\n        ] = None,\n    ) -> TaskRun:\n        \"\"\"\n        Create a task run\n\n        Args:\n            task: The Task to run\n            flow_run_id: The flow run id with which to associate the task run\n            dynamic_key: A key unique to this particular run of a Task within the flow\n            name: An optional name for the task run\n            extra_tags: an optional list of extra tags to apply to the task run in\n                addition to `task.tags`\n            state: The initial state for the run. If not provided, defaults to\n                `Pending` for now. Should always be a `Scheduled` type.\n            task_inputs: the set of inputs passed to the task\n\n        Returns:\n            The created task run.\n        \"\"\"\n        tags = set(task.tags).union(extra_tags or [])\n\n        if state is None:\n            state = prefect.states.Pending()\n\n        task_run_data = TaskRunCreate(\n            name=name,\n            flow_run_id=flow_run_id,\n            task_key=task.task_key,\n            dynamic_key=dynamic_key,\n            tags=list(tags),\n            task_version=task.version,\n            empirical_policy=TaskRunPolicy(\n                retries=task.retries,\n                retry_delay=task.retry_delay_seconds,\n                retry_jitter_factor=task.retry_jitter_factor,\n            ),\n            state=state.to_state_create(),\n            task_inputs=task_inputs or {},\n        )\n\n        response = await self._client.post(\n            \"/task_runs/\", json=task_run_data.dict(json_compatible=True)\n        )\n        return TaskRun.parse_obj(response.json())\n\n    async def read_task_run(self, task_run_id: UUID) -> TaskRun:\n        \"\"\"\n        Query the Prefect API for a task run by id.\n\n        Args:\n            task_run_id: the task run ID of interest\n\n        Returns:\n            a Task Run model representation of the task run\n        \"\"\"\n        response = await self._client.get(f\"/task_runs/{task_run_id}\")\n        return TaskRun.parse_obj(response.json())\n\n    async def read_task_runs(\n        self,\n        *,\n        flow_filter: FlowFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        deployment_filter: DeploymentFilter = None,\n        sort: TaskRunSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[TaskRun]:\n        \"\"\"\n        Query the Prefect API for task runs. Only task runs matching all criteria will\n        be returned.\n\n        Args:\n            flow_filter: filter criteria for flows\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            deployment_filter: filter criteria for deployments\n            sort: sort criteria for the task runs\n            limit: a limit for the task run query\n            offset: an offset for the task run query\n\n        Returns:\n            a list of Task Run model representations\n                of the task runs\n        \"\"\"\n        body = {\n            \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n                if flow_run_filter\n                else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"deployments\": (\n                deployment_filter.dict(json_compatible=True)\n                if deployment_filter\n                else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n        response = await self._client.post(\"/task_runs/filter\", json=body)\n        return pydantic.parse_obj_as(List[TaskRun], response.json())\n\n    async def set_task_run_state(\n        self,\n        task_run_id: UUID,\n        state: prefect.states.State,\n        force: bool = False,\n    ) -> OrchestrationResult:\n        \"\"\"\n        Set the state of a task run.\n\n        Args:\n            task_run_id: the id of the task run\n            state: the state to set\n            force: if True, disregard orchestration logic when setting the state,\n                forcing the Prefect API to accept the state\n\n        Returns:\n            an OrchestrationResult model representation of state orchestration output\n        \"\"\"\n        state_create = state.to_state_create()\n        state_create.state_details.task_run_id = task_run_id\n        response = await self._client.post(\n            f\"/task_runs/{task_run_id}/set_state\",\n            json=dict(state=state_create.dict(json_compatible=True), force=force),\n        )\n        return OrchestrationResult.parse_obj(response.json())\n\n    async def read_task_run_states(\n        self, task_run_id: UUID\n    ) -> List[prefect.states.State]:\n        \"\"\"\n        Query for the states of a task run\n\n        Args:\n            task_run_id: the id of the task run\n\n        Returns:\n            a list of State model representations of the task run states\n        \"\"\"\n        response = await self._client.get(\n            \"/task_run_states/\", params=dict(task_run_id=str(task_run_id))\n        )\n        return pydantic.parse_obj_as(List[prefect.states.State], response.json())\n\n    async def create_logs(self, logs: Iterable[Union[LogCreate, dict]]) -> None:\n        \"\"\"\n        Create logs for a flow or task run\n\n        Args:\n            logs: An iterable of `LogCreate` objects or already json-compatible dicts\n        \"\"\"\n        serialized_logs = [\n            log.dict(json_compatible=True) if isinstance(log, LogCreate) else log\n            for log in logs\n        ]\n        await self._client.post(\"/logs/\", json=serialized_logs)\n\n    async def create_flow_run_notification_policy(\n        self,\n        block_document_id: UUID,\n        is_active: bool = True,\n        tags: List[str] = None,\n        state_names: List[str] = None,\n        message_template: Optional[str] = None,\n    ) -> UUID:\n        \"\"\"\n        Create a notification policy for flow runs\n\n        Args:\n            block_document_id: The block document UUID\n            is_active: Whether the notification policy is active\n            tags: List of flow tags\n            state_names: List of state names\n            message_template: Notification message template\n        \"\"\"\n        if tags is None:\n            tags = []\n        if state_names is None:\n            state_names = []\n\n        policy = FlowRunNotificationPolicyCreate(\n            block_document_id=block_document_id,\n            is_active=is_active,\n            tags=tags,\n            state_names=state_names,\n            message_template=message_template,\n        )\n        response = await self._client.post(\n            \"/flow_run_notification_policies/\",\n            json=policy.dict(json_compatible=True),\n        )\n\n        policy_id = response.json().get(\"id\")\n        if not policy_id:\n            raise httpx.RequestError(f\"Malformed response: {response}\")\n\n        return UUID(policy_id)\n\n    async def read_flow_run_notification_policies(\n        self,\n        flow_run_notification_policy_filter: FlowRunNotificationPolicyFilter,\n        limit: Optional[int] = None,\n        offset: int = 0,\n    ) -> List[FlowRunNotificationPolicy]:\n        \"\"\"\n        Query the Prefect API for flow run notification policies. Only policies matching all criteria will\n        be returned.\n\n        Args:\n            flow_run_notification_policy_filter: filter criteria for notification policies\n            limit: a limit for the notification policies query\n            offset: an offset for the notification policies query\n\n        Returns:\n            a list of FlowRunNotificationPolicy model representations\n                of the notification policies\n        \"\"\"\n        body = {\n            \"flow_run_notification_policy_filter\": (\n                flow_run_notification_policy_filter.dict(json_compatible=True)\n                if flow_run_notification_policy_filter\n                else None\n            ),\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n        response = await self._client.post(\n            \"/flow_run_notification_policies/filter\", json=body\n        )\n        return pydantic.parse_obj_as(List[FlowRunNotificationPolicy], response.json())\n\n    async def read_logs(\n        self,\n        log_filter: LogFilter = None,\n        limit: int = None,\n        offset: int = None,\n        sort: LogSort = LogSort.TIMESTAMP_ASC,\n    ) -> List[Log]:\n        \"\"\"\n        Read flow and task run logs.\n        \"\"\"\n        body = {\n            \"logs\": log_filter.dict(json_compatible=True) if log_filter else None,\n            \"limit\": limit,\n            \"offset\": offset,\n            \"sort\": sort,\n        }\n\n        response = await self._client.post(\"/logs/filter\", json=body)\n        return pydantic.parse_obj_as(List[Log], response.json())\n\n    async def resolve_datadoc(self, datadoc: DataDocument) -> Any:\n        \"\"\"\n        Recursively decode possibly nested data documents.\n\n        \"server\" encoded documents will be retrieved from the server.\n\n        Args:\n            datadoc: The data document to resolve\n\n        Returns:\n            a decoded object, the innermost data\n        \"\"\"\n        if not isinstance(datadoc, DataDocument):\n            raise TypeError(\n                f\"`resolve_datadoc` received invalid type {type(datadoc).__name__}\"\n            )\n\n        async def resolve_inner(data):\n            if isinstance(data, bytes):\n                try:\n                    data = DataDocument.parse_raw(data)\n                except pydantic.ValidationError:\n                    return data\n\n            if isinstance(data, DataDocument):\n                return await resolve_inner(data.decode())\n\n            return data\n\n        return await resolve_inner(datadoc)\n\n    async def send_worker_heartbeat(\n        self,\n        work_pool_name: str,\n        worker_name: str,\n        heartbeat_interval_seconds: Optional[float] = None,\n    ):\n        \"\"\"\n        Sends a worker heartbeat for a given work pool.\n\n        Args:\n            work_pool_name: The name of the work pool to heartbeat against.\n            worker_name: The name of the worker sending the heartbeat.\n        \"\"\"\n        await self._client.post(\n            f\"/work_pools/{work_pool_name}/workers/heartbeat\",\n            json={\n                \"name\": worker_name,\n                \"heartbeat_interval_seconds\": heartbeat_interval_seconds,\n            },\n        )\n\n    async def read_workers_for_work_pool(\n        self,\n        work_pool_name: str,\n        worker_filter: Optional[WorkerFilter] = None,\n        offset: Optional[int] = None,\n        limit: Optional[int] = None,\n    ) -> List[Worker]:\n        \"\"\"\n        Reads workers for a given work pool.\n\n        Args:\n            work_pool_name: The name of the work pool for which to get\n                member workers.\n            worker_filter: Criteria by which to filter workers.\n            limit: Limit for the worker query.\n            offset: Limit for the worker query.\n        \"\"\"\n        response = await self._client.post(\n            f\"/work_pools/{work_pool_name}/workers/filter\",\n            json={\n                \"worker_filter\": (\n                    worker_filter.dict(json_compatible=True, exclude_unset=True)\n                    if worker_filter\n                    else None\n                ),\n                \"offset\": offset,\n                \"limit\": limit,\n            },\n        )\n\n        return pydantic.parse_obj_as(List[Worker], response.json())\n\n    async def read_work_pool(self, work_pool_name: str) -> WorkPool:\n        \"\"\"\n        Reads information for a given work pool\n\n        Args:\n            work_pool_name: The name of the work pool to for which to get\n                information.\n\n        Returns:\n            Information about the requested work pool.\n        \"\"\"\n        try:\n            response = await self._client.get(f\"/work_pools/{work_pool_name}\")\n            return pydantic.parse_obj_as(WorkPool, response.json())\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_work_pools(\n        self,\n        limit: Optional[int] = None,\n        offset: int = 0,\n        work_pool_filter: Optional[WorkPoolFilter] = None,\n    ) -> List[WorkPool]:\n        \"\"\"\n        Reads work pools.\n\n        Args:\n            limit: Limit for the work pool query.\n            offset: Offset for the work pool query.\n            work_pool_filter: Criteria by which to filter work pools.\n\n        Returns:\n            A list of work pools.\n        \"\"\"\n\n        body = {\n            \"limit\": limit,\n            \"offset\": offset,\n            \"work_pools\": (\n                work_pool_filter.dict(json_compatible=True)\n                if work_pool_filter\n                else None\n            ),\n        }\n        response = await self._client.post(\"/work_pools/filter\", json=body)\n        return pydantic.parse_obj_as(List[WorkPool], response.json())\n\n    async def create_work_pool(\n        self,\n        work_pool: WorkPoolCreate,\n    ) -> WorkPool:\n        \"\"\"\n        Creates a work pool with the provided configuration.\n\n        Args:\n            work_pool: Desired configuration for the new work pool.\n\n        Returns:\n            Information about the newly created work pool.\n        \"\"\"\n        try:\n            response = await self._client.post(\n                \"/work_pools/\",\n                json=work_pool.dict(json_compatible=True, exclude_unset=True),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_409_CONFLICT:\n                raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n            else:\n                raise\n\n        return pydantic.parse_obj_as(WorkPool, response.json())\n\n    async def update_work_pool(\n        self,\n        work_pool_name: str,\n        work_pool: WorkPoolUpdate,\n    ):\n        \"\"\"\n        Updates a work pool.\n\n        Args:\n            work_pool_name: Name of the work pool to update.\n            work_pool: Fields to update in the work pool.\n        \"\"\"\n        try:\n            await self._client.patch(\n                f\"/work_pools/{work_pool_name}\",\n                json=work_pool.dict(json_compatible=True, exclude_unset=True),\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_work_pool(\n        self,\n        work_pool_name: str,\n    ):\n        \"\"\"\n        Deletes a work pool.\n\n        Args:\n            work_pool_name: Name of the work pool to delete.\n        \"\"\"\n        try:\n            await self._client.delete(f\"/work_pools/{work_pool_name}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_work_queues(\n        self,\n        work_pool_name: Optional[str] = None,\n        work_queue_filter: Optional[WorkQueueFilter] = None,\n        limit: Optional[int] = None,\n        offset: Optional[int] = None,\n    ) -> List[WorkQueue]:\n        \"\"\"\n        Retrieves queues for a work pool.\n\n        Args:\n            work_pool_name: Name of the work pool for which to get queues.\n            work_queue_filter: Criteria by which to filter queues.\n            limit: Limit for the queue query.\n            offset: Limit for the queue query.\n\n        Returns:\n            List of queues for the specified work pool.\n        \"\"\"\n        json = {\n            \"work_queues\": (\n                work_queue_filter.dict(json_compatible=True, exclude_unset=True)\n                if work_queue_filter\n                else None\n            ),\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n\n        if work_pool_name:\n            try:\n                response = await self._client.post(\n                    f\"/work_pools/{work_pool_name}/queues/filter\",\n                    json=json,\n                )\n            except httpx.HTTPStatusError as e:\n                if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                    raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n                else:\n                    raise\n        else:\n            response = await self._client.post(\"/work_queues/filter\", json=json)\n\n        return pydantic.parse_obj_as(List[WorkQueue], response.json())\n\n    async def get_scheduled_flow_runs_for_deployments(\n        self,\n        deployment_ids: List[UUID],\n        scheduled_before: Optional[datetime.datetime] = None,\n        limit: Optional[int] = None,\n    ):\n        body: Dict[str, Any] = dict(deployment_ids=[str(id) for id in deployment_ids])\n        if scheduled_before:\n            body[\"scheduled_before\"] = str(scheduled_before)\n        if limit:\n            body[\"limit\"] = limit\n\n        response = await self._client.post(\n            \"/deployments/get_scheduled_flow_runs\",\n            json=body,\n        )\n\n        return pydantic.parse_obj_as(List[FlowRunResponse], response.json())\n\n    async def get_scheduled_flow_runs_for_work_pool(\n        self,\n        work_pool_name: str,\n        work_queue_names: Optional[List[str]] = None,\n        scheduled_before: Optional[datetime.datetime] = None,\n    ) -> List[WorkerFlowRunResponse]:\n        \"\"\"\n        Retrieves scheduled flow runs for the provided set of work pool queues.\n\n        Args:\n            work_pool_name: The name of the work pool that the work pool\n                queues are associated with.\n            work_queue_names: The names of the work pool queues from which\n                to get scheduled flow runs.\n            scheduled_before: Datetime used to filter returned flow runs. Flow runs\n                scheduled for after the given datetime string will not be returned.\n\n        Returns:\n            A list of worker flow run responses containing information about the\n            retrieved flow runs.\n        \"\"\"\n        body: Dict[str, Any] = {}\n        if work_queue_names is not None:\n            body[\"work_queue_names\"] = list(work_queue_names)\n        if scheduled_before:\n            body[\"scheduled_before\"] = str(scheduled_before)\n\n        response = await self._client.post(\n            f\"/work_pools/{work_pool_name}/get_scheduled_flow_runs\",\n            json=body,\n        )\n        return pydantic.parse_obj_as(List[WorkerFlowRunResponse], response.json())\n\n    async def create_artifact(\n        self,\n        artifact: ArtifactCreate,\n    ) -> Artifact:\n        \"\"\"\n        Creates an artifact with the provided configuration.\n\n        Args:\n            artifact: Desired configuration for the new artifact.\n        Returns:\n            Information about the newly created artifact.\n        \"\"\"\n\n        response = await self._client.post(\n            \"/artifacts/\",\n            json=artifact.dict(json_compatible=True, exclude_unset=True),\n        )\n\n        return pydantic.parse_obj_as(Artifact, response.json())\n\n    async def read_artifacts(\n        self,\n        *,\n        artifact_filter: ArtifactFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        sort: ArtifactSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[Artifact]:\n        \"\"\"\n        Query the Prefect API for artifacts. Only artifacts matching all criteria will\n        be returned.\n        Args:\n            artifact_filter: filter criteria for artifacts\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            sort: sort criteria for the artifacts\n            limit: limit for the artifact query\n            offset: offset for the artifact query\n        Returns:\n            a list of Artifact model representations of the artifacts\n        \"\"\"\n        body = {\n            \"artifacts\": (\n                artifact_filter.dict(json_compatible=True) if artifact_filter else None\n            ),\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True) if flow_run_filter else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n        response = await self._client.post(\"/artifacts/filter\", json=body)\n        return pydantic.parse_obj_as(List[Artifact], response.json())\n\n    async def read_latest_artifacts(\n        self,\n        *,\n        artifact_filter: ArtifactCollectionFilter = None,\n        flow_run_filter: FlowRunFilter = None,\n        task_run_filter: TaskRunFilter = None,\n        sort: ArtifactCollectionSort = None,\n        limit: int = None,\n        offset: int = 0,\n    ) -> List[ArtifactCollection]:\n        \"\"\"\n        Query the Prefect API for artifacts. Only artifacts matching all criteria will\n        be returned.\n        Args:\n            artifact_filter: filter criteria for artifacts\n            flow_run_filter: filter criteria for flow runs\n            task_run_filter: filter criteria for task runs\n            sort: sort criteria for the artifacts\n            limit: limit for the artifact query\n            offset: offset for the artifact query\n        Returns:\n            a list of Artifact model representations of the artifacts\n        \"\"\"\n        body = {\n            \"artifacts\": (\n                artifact_filter.dict(json_compatible=True) if artifact_filter else None\n            ),\n            \"flow_runs\": (\n                flow_run_filter.dict(json_compatible=True) if flow_run_filter else None\n            ),\n            \"task_runs\": (\n                task_run_filter.dict(json_compatible=True) if task_run_filter else None\n            ),\n            \"sort\": sort,\n            \"limit\": limit,\n            \"offset\": offset,\n        }\n        response = await self._client.post(\"/artifacts/latest/filter\", json=body)\n        return pydantic.parse_obj_as(List[ArtifactCollection], response.json())\n\n    async def delete_artifact(self, artifact_id: UUID) -> None:\n        \"\"\"\n        Deletes an artifact with the provided id.\n\n        Args:\n            artifact_id: The id of the artifact to delete.\n        \"\"\"\n        try:\n            await self._client.delete(f\"/artifacts/{artifact_id}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_variable_by_name(self, name: str) -> Optional[Variable]:\n        \"\"\"Reads a variable by name. Returns None if no variable is found.\"\"\"\n        try:\n            response = await self._client.get(f\"/variables/name/{name}\")\n            return pydantic.parse_obj_as(Variable, response.json())\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                return None\n            else:\n                raise\n\n    async def delete_variable_by_name(self, name: str):\n        \"\"\"Deletes a variable by name.\"\"\"\n        try:\n            await self._client.delete(f\"/variables/name/{name}\")\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == 404:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_variables(self, limit: int = None) -> List[Variable]:\n        \"\"\"Reads all variables.\"\"\"\n        response = await self._client.post(\"/variables/filter\", json={\"limit\": limit})\n        return pydantic.parse_obj_as(List[Variable], response.json())\n\n    async def read_worker_metadata(self) -> Dict[str, Any]:\n        \"\"\"Reads worker metadata stored in Prefect collection registry.\"\"\"\n        response = await self._client.get(\"collections/views/aggregate-worker-metadata\")\n        response.raise_for_status()\n        return response.json()\n\n    async def create_automation(self, automation: Automation) -> UUID:\n        \"\"\"Creates an automation in Prefect Cloud.\"\"\"\n        if self.server_type != ServerType.CLOUD:\n            raise RuntimeError(\"Automations are only supported for Prefect Cloud.\")\n\n        response = await self._client.post(\n            \"/automations/\",\n            json=automation.dict(json_compatible=True),\n        )\n\n        return UUID(response.json()[\"id\"])\n\n    async def read_resource_related_automations(\n        self, resource_id: str\n    ) -> List[ExistingAutomation]:\n        if self.server_type != ServerType.CLOUD:\n            raise RuntimeError(\"Automations are only supported for Prefect Cloud.\")\n\n        response = await self._client.get(f\"/automations/related-to/{resource_id}\")\n        response.raise_for_status()\n        return pydantic.parse_obj_as(List[ExistingAutomation], response.json())\n\n    async def delete_resource_owned_automations(self, resource_id: str):\n        if self.server_type != ServerType.CLOUD:\n            raise RuntimeError(\"Automations are only supported for Prefect Cloud.\")\n\n        await self._client.delete(f\"/automations/owned-by/{resource_id}\")\n\n    async def increment_concurrency_slots(\n        self, names: List[str], slots: int, mode: str\n    ) -> httpx.Response:\n        return await self._client.post(\n            \"/v2/concurrency_limits/increment\",\n            json={\"names\": names, \"slots\": slots, \"mode\": mode},\n        )\n\n    async def release_concurrency_slots(\n        self, names: List[str], slots: int, occupancy_seconds: float\n    ) -> httpx.Response:\n        return await self._client.post(\n            \"/v2/concurrency_limits/decrement\",\n            json={\n                \"names\": names,\n                \"slots\": slots,\n                \"occupancy_seconds\": occupancy_seconds,\n            },\n        )\n\n    async def create_global_concurrency_limit(\n        self, concurrency_limit: GlobalConcurrencyLimitCreate\n    ) -> UUID:\n        response = await self._client.post(\n            \"/v2/concurrency_limits/\",\n            json=concurrency_limit.dict(json_compatible=True, exclude_unset=True),\n        )\n        return UUID(response.json()[\"id\"])\n\n    async def update_global_concurrency_limit(\n        self, name: str, concurrency_limit: GlobalConcurrencyLimitUpdate\n    ) -> httpx.Response:\n        try:\n            response = await self._client.patch(\n                f\"/v2/concurrency_limits/{name}\",\n                json=concurrency_limit.dict(json_compatible=True, exclude_unset=True),\n            )\n            return response\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def delete_global_concurrency_limit_by_name(\n        self, name: str\n    ) -> httpx.Response:\n        try:\n            response = await self._client.delete(f\"/v2/concurrency_limits/{name}\")\n            return response\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_global_concurrency_limit_by_name(\n        self, name: str\n    ) -> Dict[str, object]:\n        try:\n            response = await self._client.get(f\"/v2/concurrency_limits/{name}\")\n            return response.json()\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n\n    async def read_global_concurrency_limits(\n        self, limit: int = 10, offset: int = 0\n    ) -> List[Dict[str, object]]:\n        response = await self._client.post(\n            \"/v2/concurrency_limits/filter\",\n            json={\n                \"limit\": limit,\n                \"offset\": offset,\n            },\n        )\n        return response.json()\n\n    async def create_flow_run_input(\n        self, flow_run_id: UUID, key: str, value: str, sender: Optional[str] = None\n    ):\n        \"\"\"\n        Creates a flow run input.\n\n        Args:\n            flow_run_id: The flow run id.\n            key: The input key.\n            value: The input value.\n            sender: The sender of the input.\n        \"\"\"\n\n        # Initialize the input to ensure that the key is valid.\n        FlowRunInput(flow_run_id=flow_run_id, key=key, value=value)\n\n        response = await self._client.post(\n            f\"/flow_runs/{flow_run_id}/input\",\n            json={\"key\": key, \"value\": value, \"sender\": sender},\n        )\n        response.raise_for_status()\n\n    async def filter_flow_run_input(\n        self, flow_run_id: UUID, key_prefix: str, limit: int, exclude_keys: Set[str]\n    ) -> List[FlowRunInput]:\n        response = await self._client.post(\n            f\"/flow_runs/{flow_run_id}/input/filter\",\n            json={\n                \"prefix\": key_prefix,\n                \"limit\": limit,\n                \"exclude_keys\": list(exclude_keys),\n            },\n        )\n        response.raise_for_status()\n        return pydantic.parse_obj_as(List[FlowRunInput], response.json())\n\n    async def read_flow_run_input(self, flow_run_id: UUID, key: str) -> str:\n        \"\"\"\n        Reads a flow run input.\n\n        Args:\n            flow_run_id: The flow run id.\n            key: The input key.\n        \"\"\"\n        response = await self._client.get(f\"/flow_runs/{flow_run_id}/input/{key}\")\n        response.raise_for_status()\n        return response.content.decode()\n\n    async def delete_flow_run_input(self, flow_run_id: UUID, key: str):\n        \"\"\"\n        Deletes a flow run input.\n\n        Args:\n            flow_run_id: The flow run id.\n            key: The input key.\n        \"\"\"\n        response = await self._client.delete(f\"/flow_runs/{flow_run_id}/input/{key}\")\n        response.raise_for_status()\n\n    async def __aenter__(self):\n        \"\"\"\n        Start the client.\n\n        If the client is already started, this will raise an exception.\n\n        If the client is already closed, this will raise an exception. Use a new client\n        instance instead.\n        \"\"\"\n        if self._closed:\n            # httpx.AsyncClient does not allow reuse so we will not either.\n            raise RuntimeError(\n                \"The client cannot be started again after closing. \"\n                \"Retrieve a new client with `get_client()` instead.\"\n            )\n\n        if self._started:\n            # httpx.AsyncClient does not allow reentrancy so we will not either.\n            raise RuntimeError(\"The client cannot be started more than once.\")\n\n        self._loop = asyncio.get_running_loop()\n        await self._exit_stack.__aenter__()\n\n        # Enter a lifespan context if using an ephemeral application.\n        # See https://github.com/encode/httpx/issues/350\n        if self._ephemeral_app and self.manage_lifespan:\n            self._ephemeral_lifespan = await self._exit_stack.enter_async_context(\n                app_lifespan_context(self._ephemeral_app)\n            )\n\n        if self._ephemeral_app:\n            self.logger.debug(\n                \"Using ephemeral application with database at \"\n                f\"{PREFECT_API_DATABASE_CONNECTION_URL.value()}\"\n            )\n        else:\n            self.logger.debug(f\"Connecting to API at {self.api_url}\")\n\n        # Enter the httpx client's context\n        await self._exit_stack.enter_async_context(self._client)\n\n        self._started = True\n\n        return self\n\n    async def __aexit__(self, *exc_info):\n        \"\"\"\n        Shutdown the client.\n        \"\"\"\n        self._closed = True\n        return await self._exit_stack.__aexit__(*exc_info)\n\n    def __enter__(self):\n        raise RuntimeError(\n            \"The `PrefectClient` must be entered with an async context. Use 'async \"\n            \"with PrefectClient(...)' not 'with PrefectClient(...)'\"\n        )\n\n    def __exit__(self, *_):\n        assert False, \"This should never be called but must be defined for __enter__\"\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.api_url","title":"api_url: httpx.URL property","text":"

    Get the base URL for the API.

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.api_healthcheck","title":"api_healthcheck async","text":"

    Attempts to connect to the API and returns the encountered exception if not successful.

    If successful, returns None.

    Source code in prefect/client/orchestration.py
    async def api_healthcheck(self) -> Optional[Exception]:\n    \"\"\"\n    Attempts to connect to the API and returns the encountered exception if not\n    successful.\n\n    If successful, returns `None`.\n    \"\"\"\n    try:\n        await self._client.get(\"/health\")\n        return None\n    except Exception as exc:\n        return exc\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.hello","title":"hello async","text":"

    Send a GET request to /hello for testing purposes.

    Source code in prefect/client/orchestration.py
    async def hello(self) -> httpx.Response:\n    \"\"\"\n    Send a GET request to /hello for testing purposes.\n    \"\"\"\n    return await self._client.get(\"/hello\")\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow","title":"create_flow async","text":"

    Create a flow in the Prefect API.

    Parameters:

    Name Type Description Default flow Flow

    a Flow object

    required

    Raises:

    Type Description RequestError

    if a flow was not created for any reason

    Returns:

    Type Description UUID

    the ID of the flow in the backend

    Source code in prefect/client/orchestration.py
    async def create_flow(self, flow: \"FlowObject\") -> UUID:\n    \"\"\"\n    Create a flow in the Prefect API.\n\n    Args:\n        flow: a [Flow][prefect.flows.Flow] object\n\n    Raises:\n        httpx.RequestError: if a flow was not created for any reason\n\n    Returns:\n        the ID of the flow in the backend\n    \"\"\"\n    return await self.create_flow_from_name(flow.name)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_from_name","title":"create_flow_from_name async","text":"

    Create a flow in the Prefect API.

    Parameters:

    Name Type Description Default flow_name str

    the name of the new flow

    required

    Raises:

    Type Description RequestError

    if a flow was not created for any reason

    Returns:

    Type Description UUID

    the ID of the flow in the backend

    Source code in prefect/client/orchestration.py
    async def create_flow_from_name(self, flow_name: str) -> UUID:\n    \"\"\"\n    Create a flow in the Prefect API.\n\n    Args:\n        flow_name: the name of the new flow\n\n    Raises:\n        httpx.RequestError: if a flow was not created for any reason\n\n    Returns:\n        the ID of the flow in the backend\n    \"\"\"\n    flow_data = FlowCreate(name=flow_name)\n    response = await self._client.post(\n        \"/flows/\", json=flow_data.dict(json_compatible=True)\n    )\n\n    flow_id = response.json().get(\"id\")\n    if not flow_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    # Return the id of the created flow\n    return UUID(flow_id)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow","title":"read_flow async","text":"

    Query the Prefect API for a flow by id.

    Parameters:

    Name Type Description Default flow_id UUID

    the flow ID of interest

    required

    Returns:

    Type Description Flow

    a Flow model representation of the flow

    Source code in prefect/client/orchestration.py
    async def read_flow(self, flow_id: UUID) -> Flow:\n    \"\"\"\n    Query the Prefect API for a flow by id.\n\n    Args:\n        flow_id: the flow ID of interest\n\n    Returns:\n        a [Flow model][prefect.client.schemas.objects.Flow] representation of the flow\n    \"\"\"\n    response = await self._client.get(f\"/flows/{flow_id}\")\n    return Flow.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flows","title":"read_flows async","text":"

    Query the Prefect API for flows. Only flows matching all criteria will be returned.

    Parameters:

    Name Type Description Default flow_filter FlowFilter

    filter criteria for flows

    None flow_run_filter FlowRunFilter

    filter criteria for flow runs

    None task_run_filter TaskRunFilter

    filter criteria for task runs

    None deployment_filter DeploymentFilter

    filter criteria for deployments

    None work_pool_filter WorkPoolFilter

    filter criteria for work pools

    None work_queue_filter WorkQueueFilter

    filter criteria for work pool queues

    None sort FlowSort

    sort criteria for the flows

    None limit int

    limit for the flow query

    None offset int

    offset for the flow query

    0

    Returns:

    Type Description List[Flow]

    a list of Flow model representations of the flows

    Source code in prefect/client/orchestration.py
    async def read_flows(\n    self,\n    *,\n    flow_filter: FlowFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    deployment_filter: DeploymentFilter = None,\n    work_pool_filter: WorkPoolFilter = None,\n    work_queue_filter: WorkQueueFilter = None,\n    sort: FlowSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[Flow]:\n    \"\"\"\n    Query the Prefect API for flows. Only flows matching all criteria will\n    be returned.\n\n    Args:\n        flow_filter: filter criteria for flows\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        deployment_filter: filter criteria for deployments\n        work_pool_filter: filter criteria for work pools\n        work_queue_filter: filter criteria for work pool queues\n        sort: sort criteria for the flows\n        limit: limit for the flow query\n        offset: offset for the flow query\n\n    Returns:\n        a list of Flow model representations of the flows\n    \"\"\"\n    body = {\n        \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n            if flow_run_filter\n            else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"deployments\": (\n            deployment_filter.dict(json_compatible=True)\n            if deployment_filter\n            else None\n        ),\n        \"work_pools\": (\n            work_pool_filter.dict(json_compatible=True)\n            if work_pool_filter\n            else None\n        ),\n        \"work_queues\": (\n            work_queue_filter.dict(json_compatible=True)\n            if work_queue_filter\n            else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n\n    response = await self._client.post(\"/flows/filter\", json=body)\n    return pydantic.parse_obj_as(List[Flow], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_by_name","title":"read_flow_by_name async","text":"

    Query the Prefect API for a flow by name.

    Parameters:

    Name Type Description Default flow_name str

    the name of a flow

    required

    Returns:

    Type Description Flow

    a fully hydrated Flow model

    Source code in prefect/client/orchestration.py
    async def read_flow_by_name(\n    self,\n    flow_name: str,\n) -> Flow:\n    \"\"\"\n    Query the Prefect API for a flow by name.\n\n    Args:\n        flow_name: the name of a flow\n\n    Returns:\n        a fully hydrated Flow model\n    \"\"\"\n    response = await self._client.get(f\"/flows/name/{flow_name}\")\n    return Flow.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_run_from_deployment","title":"create_flow_run_from_deployment async","text":"

    Create a flow run for a deployment.

    Parameters:

    Name Type Description Default deployment_id UUID

    The deployment ID to create the flow run from

    required parameters Dict[str, Any]

    Parameter overrides for this flow run. Merged with the deployment defaults

    None context dict

    Optional run context data

    None state State

    The initial state for the run. If not provided, defaults to Scheduled for now. Should always be a Scheduled type.

    None name str

    An optional name for the flow run. If not provided, the server will generate a name.

    None tags Iterable[str]

    An optional iterable of tags to apply to the flow run; these tags are merged with the deployment's tags.

    None idempotency_key str

    Optional idempotency key for creation of the flow run. If the key matches the key of an existing flow run, the existing run will be returned instead of creating a new one.

    None parent_task_run_id UUID

    if a subflow run is being created, the placeholder task run identifier in the parent flow

    None work_queue_name str

    An optional work queue name to add this run to. If not provided, will default to the deployment's set work queue. If one is provided that does not exist, a new work queue will be created within the deployment's work pool.

    None

    Raises:

    Type Description RequestError

    if the Prefect API does not successfully create a run for any reason

    Returns:

    Type Description FlowRun

    The flow run model

    Source code in prefect/client/orchestration.py
    async def create_flow_run_from_deployment(\n    self,\n    deployment_id: UUID,\n    *,\n    parameters: Dict[str, Any] = None,\n    context: dict = None,\n    state: prefect.states.State = None,\n    name: str = None,\n    tags: Iterable[str] = None,\n    idempotency_key: str = None,\n    parent_task_run_id: UUID = None,\n    work_queue_name: str = None,\n) -> FlowRun:\n    \"\"\"\n    Create a flow run for a deployment.\n\n    Args:\n        deployment_id: The deployment ID to create the flow run from\n        parameters: Parameter overrides for this flow run. Merged with the\n            deployment defaults\n        context: Optional run context data\n        state: The initial state for the run. If not provided, defaults to\n            `Scheduled` for now. Should always be a `Scheduled` type.\n        name: An optional name for the flow run. If not provided, the server will\n            generate a name.\n        tags: An optional iterable of tags to apply to the flow run; these tags\n            are merged with the deployment's tags.\n        idempotency_key: Optional idempotency key for creation of the flow run.\n            If the key matches the key of an existing flow run, the existing run will\n            be returned instead of creating a new one.\n        parent_task_run_id: if a subflow run is being created, the placeholder task\n            run identifier in the parent flow\n        work_queue_name: An optional work queue name to add this run to. If not provided,\n            will default to the deployment's set work queue.  If one is provided that does not\n            exist, a new work queue will be created within the deployment's work pool.\n\n    Raises:\n        httpx.RequestError: if the Prefect API does not successfully create a run for any reason\n\n    Returns:\n        The flow run model\n    \"\"\"\n    parameters = parameters or {}\n    context = context or {}\n    state = state or prefect.states.Scheduled()\n    tags = tags or []\n\n    flow_run_create = DeploymentFlowRunCreate(\n        parameters=parameters,\n        context=context,\n        state=state.to_state_create(),\n        tags=tags,\n        name=name,\n        idempotency_key=idempotency_key,\n        parent_task_run_id=parent_task_run_id,\n    )\n\n    # done separately to avoid including this field in payloads sent to older API versions\n    if work_queue_name:\n        flow_run_create.work_queue_name = work_queue_name\n\n    response = await self._client.post(\n        f\"/deployments/{deployment_id}/create_flow_run\",\n        json=flow_run_create.dict(json_compatible=True, exclude_unset=True),\n    )\n    return FlowRun.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_run","title":"create_flow_run async","text":"

    Create a flow run for a flow.

    Parameters:

    Name Type Description Default flow Flow

    The flow model to create the flow run for

    required name str

    An optional name for the flow run

    None parameters Dict[str, Any]

    Parameter overrides for this flow run.

    None context dict

    Optional run context data

    None tags Iterable[str]

    a list of tags to apply to this flow run

    None parent_task_run_id UUID

    if a subflow run is being created, the placeholder task run identifier in the parent flow

    None state State

    The initial state for the run. If not provided, defaults to Scheduled for now. Should always be a Scheduled type.

    None

    Raises:

    Type Description RequestError

    if the Prefect API does not successfully create a run for any reason

    Returns:

    Type Description FlowRun

    The flow run model

    Source code in prefect/client/orchestration.py
    async def create_flow_run(\n    self,\n    flow: \"FlowObject\",\n    name: str = None,\n    parameters: Dict[str, Any] = None,\n    context: dict = None,\n    tags: Iterable[str] = None,\n    parent_task_run_id: UUID = None,\n    state: \"prefect.states.State\" = None,\n) -> FlowRun:\n    \"\"\"\n    Create a flow run for a flow.\n\n    Args:\n        flow: The flow model to create the flow run for\n        name: An optional name for the flow run\n        parameters: Parameter overrides for this flow run.\n        context: Optional run context data\n        tags: a list of tags to apply to this flow run\n        parent_task_run_id: if a subflow run is being created, the placeholder task\n            run identifier in the parent flow\n        state: The initial state for the run. If not provided, defaults to\n            `Scheduled` for now. Should always be a `Scheduled` type.\n\n    Raises:\n        httpx.RequestError: if the Prefect API does not successfully create a run for any reason\n\n    Returns:\n        The flow run model\n    \"\"\"\n    parameters = parameters or {}\n    context = context or {}\n\n    if state is None:\n        state = prefect.states.Pending()\n\n    # Retrieve the flow id\n    flow_id = await self.create_flow(flow)\n\n    flow_run_create = FlowRunCreate(\n        flow_id=flow_id,\n        flow_version=flow.version,\n        name=name,\n        parameters=parameters,\n        context=context,\n        tags=list(tags or []),\n        parent_task_run_id=parent_task_run_id,\n        state=state.to_state_create(),\n        empirical_policy=FlowRunPolicy(\n            retries=flow.retries,\n            retry_delay=flow.retry_delay_seconds,\n        ),\n    )\n\n    flow_run_create_json = flow_run_create.dict(json_compatible=True)\n    response = await self._client.post(\"/flow_runs/\", json=flow_run_create_json)\n    flow_run = FlowRun.parse_obj(response.json())\n\n    # Restore the parameters to the local objects to retain expectations about\n    # Python objects\n    flow_run.parameters = parameters\n\n    return flow_run\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_flow_run","title":"update_flow_run async","text":"

    Update a flow run's details.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The identifier for the flow run to update.

    required flow_version Optional[str]

    A new version string for the flow run.

    None parameters Optional[dict]

    A dictionary of parameter values for the flow run. This will not be merged with any existing parameters.

    None name Optional[str]

    A new name for the flow run.

    None empirical_policy Optional[FlowRunPolicy]

    A new flow run orchestration policy. This will not be merged with any existing policy.

    None tags Optional[Iterable[str]]

    An iterable of new tags for the flow run. These will not be merged with any existing tags.

    None infrastructure_pid Optional[str]

    The id of flow run as returned by an infrastructure block.

    None

    Returns:

    Type Description Response

    an httpx.Response object from the PATCH request

    Source code in prefect/client/orchestration.py
    async def update_flow_run(\n    self,\n    flow_run_id: UUID,\n    flow_version: Optional[str] = None,\n    parameters: Optional[dict] = None,\n    name: Optional[str] = None,\n    tags: Optional[Iterable[str]] = None,\n    empirical_policy: Optional[FlowRunPolicy] = None,\n    infrastructure_pid: Optional[str] = None,\n) -> httpx.Response:\n    \"\"\"\n    Update a flow run's details.\n\n    Args:\n        flow_run_id: The identifier for the flow run to update.\n        flow_version: A new version string for the flow run.\n        parameters: A dictionary of parameter values for the flow run. This will not\n            be merged with any existing parameters.\n        name: A new name for the flow run.\n        empirical_policy: A new flow run orchestration policy. This will not be\n            merged with any existing policy.\n        tags: An iterable of new tags for the flow run. These will not be merged with\n            any existing tags.\n        infrastructure_pid: The id of flow run as returned by an\n            infrastructure block.\n\n    Returns:\n        an `httpx.Response` object from the PATCH request\n    \"\"\"\n    params = {}\n    if flow_version is not None:\n        params[\"flow_version\"] = flow_version\n    if parameters is not None:\n        params[\"parameters\"] = parameters\n    if name is not None:\n        params[\"name\"] = name\n    if tags is not None:\n        params[\"tags\"] = tags\n    if empirical_policy is not None:\n        params[\"empirical_policy\"] = empirical_policy\n    if infrastructure_pid:\n        params[\"infrastructure_pid\"] = infrastructure_pid\n\n    flow_run_data = FlowRunUpdate(**params)\n\n    return await self._client.patch(\n        f\"/flow_runs/{flow_run_id}\",\n        json=flow_run_data.dict(json_compatible=True, exclude_unset=True),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_flow_run","title":"delete_flow_run async","text":"

    Delete a flow run by UUID.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The flow run UUID of interest.

    required Source code in prefect/client/orchestration.py
    async def delete_flow_run(\n    self,\n    flow_run_id: UUID,\n) -> None:\n    \"\"\"\n    Delete a flow run by UUID.\n\n    Args:\n        flow_run_id: The flow run UUID of interest.\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If requests fails\n    \"\"\"\n    try:\n        await self._client.delete(f\"/flow_runs/{flow_run_id}\"),\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_concurrency_limit","title":"create_concurrency_limit async","text":"

    Create a tag concurrency limit in the Prefect API. These limits govern concurrently running tasks.

    Parameters:

    Name Type Description Default tag str

    a tag the concurrency limit is applied to

    required concurrency_limit int

    the maximum number of concurrent task runs for a given tag

    required

    Raises:

    Type Description RequestError

    if the concurrency limit was not created for any reason

    Returns:

    Type Description UUID

    the ID of the concurrency limit in the backend

    Source code in prefect/client/orchestration.py
    async def create_concurrency_limit(\n    self,\n    tag: str,\n    concurrency_limit: int,\n) -> UUID:\n    \"\"\"\n    Create a tag concurrency limit in the Prefect API. These limits govern concurrently\n    running tasks.\n\n    Args:\n        tag: a tag the concurrency limit is applied to\n        concurrency_limit: the maximum number of concurrent task runs for a given tag\n\n    Raises:\n        httpx.RequestError: if the concurrency limit was not created for any reason\n\n    Returns:\n        the ID of the concurrency limit in the backend\n    \"\"\"\n\n    concurrency_limit_create = ConcurrencyLimitCreate(\n        tag=tag,\n        concurrency_limit=concurrency_limit,\n    )\n    response = await self._client.post(\n        \"/concurrency_limits/\",\n        json=concurrency_limit_create.dict(json_compatible=True),\n    )\n\n    concurrency_limit_id = response.json().get(\"id\")\n\n    if not concurrency_limit_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    return UUID(concurrency_limit_id)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_concurrency_limit_by_tag","title":"read_concurrency_limit_by_tag async","text":"

    Read the concurrency limit set on a specific tag.

    Parameters:

    Name Type Description Default tag str

    a tag the concurrency limit is applied to

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    if the concurrency limit was not created for any reason

    Returns:

    Type Description

    the concurrency limit set on a specific tag

    Source code in prefect/client/orchestration.py
    async def read_concurrency_limit_by_tag(\n    self,\n    tag: str,\n):\n    \"\"\"\n    Read the concurrency limit set on a specific tag.\n\n    Args:\n        tag: a tag the concurrency limit is applied to\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: if the concurrency limit was not created for any reason\n\n    Returns:\n        the concurrency limit set on a specific tag\n    \"\"\"\n    try:\n        response = await self._client.get(\n            f\"/concurrency_limits/tag/{tag}\",\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n\n    concurrency_limit_id = response.json().get(\"id\")\n\n    if not concurrency_limit_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    concurrency_limit = ConcurrencyLimit.parse_obj(response.json())\n    return concurrency_limit\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_concurrency_limits","title":"read_concurrency_limits async","text":"

    Lists concurrency limits set on task run tags.

    Parameters:

    Name Type Description Default limit int

    the maximum number of concurrency limits returned

    required offset int

    the concurrency limit query offset

    required

    Returns:

    Type Description

    a list of concurrency limits

    Source code in prefect/client/orchestration.py
    async def read_concurrency_limits(\n    self,\n    limit: int,\n    offset: int,\n):\n    \"\"\"\n    Lists concurrency limits set on task run tags.\n\n    Args:\n        limit: the maximum number of concurrency limits returned\n        offset: the concurrency limit query offset\n\n    Returns:\n        a list of concurrency limits\n    \"\"\"\n\n    body = {\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n\n    response = await self._client.post(\"/concurrency_limits/filter\", json=body)\n    return pydantic.parse_obj_as(List[ConcurrencyLimit], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.reset_concurrency_limit_by_tag","title":"reset_concurrency_limit_by_tag async","text":"

    Resets the concurrency limit slots set on a specific tag.

    Parameters:

    Name Type Description Default tag str

    a tag the concurrency limit is applied to

    required slot_override Optional[List[Union[UUID, str]]]

    a list of task run IDs that are currently using a concurrency slot, please check that any task run IDs included in slot_override are currently running, otherwise those concurrency slots will never be released.

    None

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Source code in prefect/client/orchestration.py
    async def reset_concurrency_limit_by_tag(\n    self,\n    tag: str,\n    slot_override: Optional[List[Union[UUID, str]]] = None,\n):\n    \"\"\"\n    Resets the concurrency limit slots set on a specific tag.\n\n    Args:\n        tag: a tag the concurrency limit is applied to\n        slot_override: a list of task run IDs that are currently using a\n            concurrency slot, please check that any task run IDs included in\n            `slot_override` are currently running, otherwise those concurrency\n            slots will never be released.\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    \"\"\"\n    if slot_override is not None:\n        slot_override = [str(slot) for slot in slot_override]\n\n    try:\n        await self._client.post(\n            f\"/concurrency_limits/tag/{tag}/reset\",\n            json=dict(slot_override=slot_override),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_concurrency_limit_by_tag","title":"delete_concurrency_limit_by_tag async","text":"

    Delete the concurrency limit set on a specific tag.

    Parameters:

    Name Type Description Default tag str

    a tag the concurrency limit is applied to

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Source code in prefect/client/orchestration.py
    async def delete_concurrency_limit_by_tag(\n    self,\n    tag: str,\n):\n    \"\"\"\n    Delete the concurrency limit set on a specific tag.\n\n    Args:\n        tag: a tag the concurrency limit is applied to\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    \"\"\"\n    try:\n        await self._client.delete(\n            f\"/concurrency_limits/tag/{tag}\",\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_work_queue","title":"create_work_queue async","text":"

    Create a work queue.

    Parameters:

    Name Type Description Default name str

    a unique name for the work queue

    required tags Optional[List[str]]

    will be included in the queue. This option will be removed on 2023-02-23.

    None description Optional[str]

    An optional description for the work queue.

    None is_paused Optional[bool]

    Whether or not the work queue is paused.

    None concurrency_limit Optional[int]

    An optional concurrency limit for the work queue.

    None priority Optional[int]

    The queue's priority. Lower values are higher priority (1 is the highest).

    None work_pool_name Optional[str]

    The name of the work pool to use for this queue.

    None

    Raises:

    Type Description ObjectAlreadyExists

    If request returns 409

    RequestError

    If request fails

    Returns:

    Type Description WorkQueue

    The created work queue

    Source code in prefect/client/orchestration.py
    async def create_work_queue(\n    self,\n    name: str,\n    tags: Optional[List[str]] = None,\n    description: Optional[str] = None,\n    is_paused: Optional[bool] = None,\n    concurrency_limit: Optional[int] = None,\n    priority: Optional[int] = None,\n    work_pool_name: Optional[str] = None,\n) -> WorkQueue:\n    \"\"\"\n    Create a work queue.\n\n    Args:\n        name: a unique name for the work queue\n        tags: DEPRECATED: an optional list of tags to filter on; only work scheduled with these tags\n            will be included in the queue. This option will be removed on 2023-02-23.\n        description: An optional description for the work queue.\n        is_paused: Whether or not the work queue is paused.\n        concurrency_limit: An optional concurrency limit for the work queue.\n        priority: The queue's priority. Lower values are higher priority (1 is the highest).\n        work_pool_name: The name of the work pool to use for this queue.\n\n    Raises:\n        prefect.exceptions.ObjectAlreadyExists: If request returns 409\n        httpx.RequestError: If request fails\n\n    Returns:\n        The created work queue\n    \"\"\"\n    if tags:\n        warnings.warn(\n            (\n                \"The use of tags for creating work queue filters is deprecated.\"\n                \" This option will be removed on 2023-02-23.\"\n            ),\n            DeprecationWarning,\n        )\n        filter = QueueFilter(tags=tags)\n    else:\n        filter = None\n    create_model = WorkQueueCreate(name=name, filter=filter)\n    if description is not None:\n        create_model.description = description\n    if is_paused is not None:\n        create_model.is_paused = is_paused\n    if concurrency_limit is not None:\n        create_model.concurrency_limit = concurrency_limit\n    if priority is not None:\n        create_model.priority = priority\n\n    data = create_model.dict(json_compatible=True)\n    try:\n        if work_pool_name is not None:\n            response = await self._client.post(\n                f\"/work_pools/{work_pool_name}/queues\", json=data\n            )\n        else:\n            response = await self._client.post(\"/work_queues/\", json=data)\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        elif e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return WorkQueue.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_queue_by_name","title":"read_work_queue_by_name async","text":"

    Read a work queue by name.

    Parameters:

    Name Type Description Default name str

    a unique name for the work queue

    required work_pool_name str

    the name of the work pool the queue belongs to.

    None

    Raises:

    Type Description ObjectNotFound

    if no work queue is found

    HTTPStatusError

    other status errors

    Returns:

    Name Type Description WorkQueue WorkQueue

    a work queue API object

    Source code in prefect/client/orchestration.py
    async def read_work_queue_by_name(\n    self,\n    name: str,\n    work_pool_name: Optional[str] = None,\n) -> WorkQueue:\n    \"\"\"\n    Read a work queue by name.\n\n    Args:\n        name (str): a unique name for the work queue\n        work_pool_name (str, optional): the name of the work pool\n            the queue belongs to.\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: if no work queue is found\n        httpx.HTTPStatusError: other status errors\n\n    Returns:\n        WorkQueue: a work queue API object\n    \"\"\"\n    try:\n        if work_pool_name is not None:\n            response = await self._client.get(\n                f\"/work_pools/{work_pool_name}/queues/{name}\"\n            )\n        else:\n            response = await self._client.get(f\"/work_queues/name/{name}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n\n    return WorkQueue.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_work_queue","title":"update_work_queue async","text":"

    Update properties of a work queue.

    Parameters:

    Name Type Description Default id UUID

    the ID of the work queue to update

    required **kwargs

    the fields to update

    {}

    Raises:

    Type Description ValueError

    if no kwargs are provided

    ObjectNotFound

    if request returns 404

    RequestError

    if the request fails

    Source code in prefect/client/orchestration.py
    async def update_work_queue(self, id: UUID, **kwargs):\n    \"\"\"\n    Update properties of a work queue.\n\n    Args:\n        id: the ID of the work queue to update\n        **kwargs: the fields to update\n\n    Raises:\n        ValueError: if no kwargs are provided\n        prefect.exceptions.ObjectNotFound: if request returns 404\n        httpx.RequestError: if the request fails\n\n    \"\"\"\n    if not kwargs:\n        raise ValueError(\"No fields provided to update.\")\n\n    data = WorkQueueUpdate(**kwargs).dict(json_compatible=True, exclude_unset=True)\n    try:\n        await self._client.patch(f\"/work_queues/{id}\", json=data)\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.get_runs_in_work_queue","title":"get_runs_in_work_queue async","text":"

    Read flow runs off a work queue.

    Parameters:

    Name Type Description Default id UUID

    the id of the work queue to read from

    required limit int

    a limit on the number of runs to return

    10 scheduled_before datetime

    a timestamp; only runs scheduled before this time will be returned. Defaults to now.

    None

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Returns:

    Type Description List[FlowRun]

    List[FlowRun]: a list of FlowRun objects read from the queue

    Source code in prefect/client/orchestration.py
    async def get_runs_in_work_queue(\n    self,\n    id: UUID,\n    limit: int = 10,\n    scheduled_before: datetime.datetime = None,\n) -> List[FlowRun]:\n    \"\"\"\n    Read flow runs off a work queue.\n\n    Args:\n        id: the id of the work queue to read from\n        limit: a limit on the number of runs to return\n        scheduled_before: a timestamp; only runs scheduled before this time will be returned.\n            Defaults to now.\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    Returns:\n        List[FlowRun]: a list of FlowRun objects read from the queue\n    \"\"\"\n    if scheduled_before is None:\n        scheduled_before = pendulum.now(\"UTC\")\n\n    try:\n        response = await self._client.post(\n            f\"/work_queues/{id}/get_runs\",\n            json={\n                \"limit\": limit,\n                \"scheduled_before\": scheduled_before.isoformat(),\n            },\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return pydantic.parse_obj_as(List[FlowRun], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_queue","title":"read_work_queue async","text":"

    Read a work queue.

    Parameters:

    Name Type Description Default id UUID

    the id of the work queue to load

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Returns:

    Name Type Description WorkQueue WorkQueue

    an instantiated WorkQueue object

    Source code in prefect/client/orchestration.py
    async def read_work_queue(\n    self,\n    id: UUID,\n) -> WorkQueue:\n    \"\"\"\n    Read a work queue.\n\n    Args:\n        id: the id of the work queue to load\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    Returns:\n        WorkQueue: an instantiated WorkQueue object\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/work_queues/{id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return WorkQueue.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_queue_status","title":"read_work_queue_status async","text":"

    Read a work queue status.

    Parameters:

    Name Type Description Default id UUID

    the id of the work queue to load

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Returns:

    Name Type Description WorkQueueStatus WorkQueueStatusDetail

    an instantiated WorkQueueStatus object

    Source code in prefect/client/orchestration.py
    async def read_work_queue_status(\n    self,\n    id: UUID,\n) -> WorkQueueStatusDetail:\n    \"\"\"\n    Read a work queue status.\n\n    Args:\n        id: the id of the work queue to load\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    Returns:\n        WorkQueueStatus: an instantiated WorkQueueStatus object\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/work_queues/{id}/status\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return WorkQueueStatusDetail.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.match_work_queues","title":"match_work_queues async","text":"

    Query the Prefect API for work queues with names with a specific prefix.

    Parameters:

    Name Type Description Default prefixes List[str]

    a list of strings used to match work queue name prefixes

    required work_pool_name Optional[str]

    an optional work pool name to scope the query to

    None

    Returns:

    Type Description List[WorkQueue]

    a list of WorkQueue model representations of the work queues

    Source code in prefect/client/orchestration.py
    async def match_work_queues(\n    self,\n    prefixes: List[str],\n    work_pool_name: Optional[str] = None,\n) -> List[WorkQueue]:\n    \"\"\"\n    Query the Prefect API for work queues with names with a specific prefix.\n\n    Args:\n        prefixes: a list of strings used to match work queue name prefixes\n        work_pool_name: an optional work pool name to scope the query to\n\n    Returns:\n        a list of WorkQueue model representations\n            of the work queues\n    \"\"\"\n    page_length = 100\n    current_page = 0\n    work_queues = []\n\n    while True:\n        new_queues = await self.read_work_queues(\n            work_pool_name=work_pool_name,\n            offset=current_page * page_length,\n            limit=page_length,\n            work_queue_filter=WorkQueueFilter(\n                name=WorkQueueFilterName(startswith_=prefixes)\n            ),\n        )\n        if not new_queues:\n            break\n        work_queues += new_queues\n        current_page += 1\n\n    return work_queues\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_work_queue_by_id","title":"delete_work_queue_by_id async","text":"

    Delete a work queue by its ID.

    Parameters:

    Name Type Description Default id UUID

    the id of the work queue to delete

    required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If requests fails

    Source code in prefect/client/orchestration.py
    async def delete_work_queue_by_id(\n    self,\n    id: UUID,\n):\n    \"\"\"\n    Delete a work queue by its ID.\n\n    Args:\n        id: the id of the work queue to delete\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If requests fails\n    \"\"\"\n    try:\n        await self._client.delete(\n            f\"/work_queues/{id}\",\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_block_type","title":"create_block_type async","text":"

    Create a block type in the Prefect API.

    Source code in prefect/client/orchestration.py
    async def create_block_type(self, block_type: BlockTypeCreate) -> BlockType:\n    \"\"\"\n    Create a block type in the Prefect API.\n    \"\"\"\n    try:\n        response = await self._client.post(\n            \"/block_types/\",\n            json=block_type.dict(\n                json_compatible=True, exclude_unset=True, exclude={\"id\"}\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        else:\n            raise\n    return BlockType.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_block_schema","title":"create_block_schema async","text":"

    Create a block schema in the Prefect API.

    Source code in prefect/client/orchestration.py
    async def create_block_schema(self, block_schema: BlockSchemaCreate) -> BlockSchema:\n    \"\"\"\n    Create a block schema in the Prefect API.\n    \"\"\"\n    try:\n        response = await self._client.post(\n            \"/block_schemas/\",\n            json=block_schema.dict(\n                json_compatible=True,\n                exclude_unset=True,\n                exclude={\"id\", \"block_type\", \"checksum\"},\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        else:\n            raise\n    return BlockSchema.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_block_document","title":"create_block_document async","text":"

    Create a block document in the Prefect API. This data is used to configure a corresponding Block.

    Parameters:

    Name Type Description Default include_secrets bool

    whether to include secret values on the stored Block, corresponding to Pydantic's SecretStr and SecretBytes fields. Note Blocks may not work as expected if this is set to False.

    True Source code in prefect/client/orchestration.py
    async def create_block_document(\n    self,\n    block_document: Union[BlockDocument, BlockDocumentCreate],\n    include_secrets: bool = True,\n) -> BlockDocument:\n    \"\"\"\n    Create a block document in the Prefect API. This data is used to configure a\n    corresponding Block.\n\n    Args:\n        include_secrets (bool): whether to include secret values\n            on the stored Block, corresponding to Pydantic's `SecretStr` and\n            `SecretBytes` fields. Note Blocks may not work as expected if\n            this is set to `False`.\n    \"\"\"\n    if isinstance(block_document, BlockDocument):\n        block_document = BlockDocumentCreate.parse_obj(\n            block_document.dict(\n                json_compatible=True,\n                include_secrets=include_secrets,\n                exclude_unset=True,\n                exclude={\"id\", \"block_schema\", \"block_type\"},\n            ),\n        )\n\n    try:\n        response = await self._client.post(\n            \"/block_documents/\",\n            json=block_document.dict(\n                json_compatible=True,\n                include_secrets=include_secrets,\n                exclude_unset=True,\n                exclude={\"id\", \"block_schema\", \"block_type\"},\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        else:\n            raise\n    return BlockDocument.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_block_document","title":"update_block_document async","text":"

    Update a block document in the Prefect API.

    Source code in prefect/client/orchestration.py
    async def update_block_document(\n    self,\n    block_document_id: UUID,\n    block_document: BlockDocumentUpdate,\n):\n    \"\"\"\n    Update a block document in the Prefect API.\n    \"\"\"\n    try:\n        await self._client.patch(\n            f\"/block_documents/{block_document_id}\",\n            json=block_document.dict(\n                json_compatible=True,\n                exclude_unset=True,\n                include={\"data\", \"merge_existing_data\", \"block_schema_id\"},\n                include_secrets=True,\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_block_document","title":"delete_block_document async","text":"

    Delete a block document.

    Source code in prefect/client/orchestration.py
    async def delete_block_document(self, block_document_id: UUID):\n    \"\"\"\n    Delete a block document.\n    \"\"\"\n    try:\n        await self._client.delete(f\"/block_documents/{block_document_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_type_by_slug","title":"read_block_type_by_slug async","text":"

    Read a block type by its slug.

    Source code in prefect/client/orchestration.py
    async def read_block_type_by_slug(self, slug: str) -> BlockType:\n    \"\"\"\n    Read a block type by its slug.\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/block_types/slug/{slug}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return BlockType.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_schema_by_checksum","title":"read_block_schema_by_checksum async","text":"

    Look up a block schema checksum

    Source code in prefect/client/orchestration.py
    async def read_block_schema_by_checksum(\n    self, checksum: str, version: Optional[str] = None\n) -> BlockSchema:\n    \"\"\"\n    Look up a block schema checksum\n    \"\"\"\n    try:\n        url = f\"/block_schemas/checksum/{checksum}\"\n        if version is not None:\n            url = f\"{url}?version={version}\"\n        response = await self._client.get(url)\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return BlockSchema.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_block_type","title":"update_block_type async","text":"

    Update a block document in the Prefect API.

    Source code in prefect/client/orchestration.py
    async def update_block_type(self, block_type_id: UUID, block_type: BlockTypeUpdate):\n    \"\"\"\n    Update a block document in the Prefect API.\n    \"\"\"\n    try:\n        await self._client.patch(\n            f\"/block_types/{block_type_id}\",\n            json=block_type.dict(\n                json_compatible=True,\n                exclude_unset=True,\n                include=BlockTypeUpdate.updatable_fields(),\n                include_secrets=True,\n            ),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_block_type","title":"delete_block_type async","text":"

    Delete a block type.

    Source code in prefect/client/orchestration.py
    async def delete_block_type(self, block_type_id: UUID):\n    \"\"\"\n    Delete a block type.\n    \"\"\"\n    try:\n        await self._client.delete(f\"/block_types/{block_type_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        elif (\n            e.response.status_code == status.HTTP_403_FORBIDDEN\n            and e.response.json()[\"detail\"]\n            == \"protected block types cannot be deleted.\"\n        ):\n            raise prefect.exceptions.ProtectedBlockError(\n                \"Protected block types cannot be deleted.\"\n            ) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_types","title":"read_block_types async","text":"

    Read all block types Raises: httpx.RequestError: if the block types were not found

    Returns:

    Type Description List[BlockType]

    List of BlockTypes.

    Source code in prefect/client/orchestration.py
    async def read_block_types(self) -> List[BlockType]:\n    \"\"\"\n    Read all block types\n    Raises:\n        httpx.RequestError: if the block types were not found\n\n    Returns:\n        List of BlockTypes.\n    \"\"\"\n    response = await self._client.post(\"/block_types/filter\", json={})\n    return pydantic.parse_obj_as(List[BlockType], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_schemas","title":"read_block_schemas async","text":"

    Read all block schemas Raises: httpx.RequestError: if a valid block schema was not found

    Returns:

    Type Description List[BlockSchema]

    A BlockSchema.

    Source code in prefect/client/orchestration.py
    async def read_block_schemas(self) -> List[BlockSchema]:\n    \"\"\"\n    Read all block schemas\n    Raises:\n        httpx.RequestError: if a valid block schema was not found\n\n    Returns:\n        A BlockSchema.\n    \"\"\"\n    response = await self._client.post(\"/block_schemas/filter\", json={})\n    return pydantic.parse_obj_as(List[BlockSchema], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.get_most_recent_block_schema_for_block_type","title":"get_most_recent_block_schema_for_block_type async","text":"

    Fetches the most recent block schema for a specified block type ID.

    Parameters:

    Name Type Description Default block_type_id UUID

    The ID of the block type.

    required

    Raises:

    Type Description RequestError

    If the request fails for any reason.

    Returns:

    Type Description Optional[BlockSchema]

    The most recent block schema or None.

    Source code in prefect/client/orchestration.py
    async def get_most_recent_block_schema_for_block_type(\n    self,\n    block_type_id: UUID,\n) -> Optional[BlockSchema]:\n    \"\"\"\n    Fetches the most recent block schema for a specified block type ID.\n\n    Args:\n        block_type_id: The ID of the block type.\n\n    Raises:\n        httpx.RequestError: If the request fails for any reason.\n\n    Returns:\n        The most recent block schema or None.\n    \"\"\"\n    try:\n        response = await self._client.post(\n            \"/block_schemas/filter\",\n            json={\n                \"block_schemas\": {\"block_type_id\": {\"any_\": [str(block_type_id)]}},\n                \"limit\": 1,\n            },\n        )\n    except httpx.HTTPStatusError:\n        raise\n    return BlockSchema.parse_obj(response.json()[0]) if response.json() else None\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_document","title":"read_block_document async","text":"

    Read the block document with the specified ID.

    Parameters:

    Name Type Description Default block_document_id UUID

    the block document id

    required include_secrets bool

    whether to include secret values on the Block, corresponding to Pydantic's SecretStr and SecretBytes fields. These fields are automatically obfuscated by Pydantic, but users can additionally choose not to receive their values from the API. Note that any business logic on the Block may not work if this is False.

    True

    Raises:

    Type Description RequestError

    if the block document was not found for any reason

    Returns:

    Type Description

    A block document or None.

    Source code in prefect/client/orchestration.py
    async def read_block_document(\n    self,\n    block_document_id: UUID,\n    include_secrets: bool = True,\n):\n    \"\"\"\n    Read the block document with the specified ID.\n\n    Args:\n        block_document_id: the block document id\n        include_secrets (bool): whether to include secret values\n            on the Block, corresponding to Pydantic's `SecretStr` and\n            `SecretBytes` fields. These fields are automatically obfuscated\n            by Pydantic, but users can additionally choose not to receive\n            their values from the API. Note that any business logic on the\n            Block may not work if this is `False`.\n\n    Raises:\n        httpx.RequestError: if the block document was not found for any reason\n\n    Returns:\n        A block document or None.\n    \"\"\"\n    assert (\n        block_document_id is not None\n    ), \"Unexpected ID on block document. Was it persisted?\"\n    try:\n        response = await self._client.get(\n            f\"/block_documents/{block_document_id}\",\n            params=dict(include_secrets=include_secrets),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return BlockDocument.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_document_by_name","title":"read_block_document_by_name async","text":"

    Read the block document with the specified name that corresponds to a specific block type name.

    Parameters:

    Name Type Description Default name str

    The block document name.

    required block_type_slug str

    The block type slug.

    required include_secrets bool

    whether to include secret values on the Block, corresponding to Pydantic's SecretStr and SecretBytes fields. These fields are automatically obfuscated by Pydantic, but users can additionally choose not to receive their values from the API. Note that any business logic on the Block may not work if this is False.

    True

    Raises:

    Type Description RequestError

    if the block document was not found for any reason

    Returns:

    Type Description BlockDocument

    A block document or None.

    Source code in prefect/client/orchestration.py
    async def read_block_document_by_name(\n    self,\n    name: str,\n    block_type_slug: str,\n    include_secrets: bool = True,\n) -> BlockDocument:\n    \"\"\"\n    Read the block document with the specified name that corresponds to a\n    specific block type name.\n\n    Args:\n        name: The block document name.\n        block_type_slug: The block type slug.\n        include_secrets (bool): whether to include secret values\n            on the Block, corresponding to Pydantic's `SecretStr` and\n            `SecretBytes` fields. These fields are automatically obfuscated\n            by Pydantic, but users can additionally choose not to receive\n            their values from the API. Note that any business logic on the\n            Block may not work if this is `False`.\n\n    Raises:\n        httpx.RequestError: if the block document was not found for any reason\n\n    Returns:\n        A block document or None.\n    \"\"\"\n    try:\n        response = await self._client.get(\n            f\"/block_types/slug/{block_type_slug}/block_documents/name/{name}\",\n            params=dict(include_secrets=include_secrets),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return BlockDocument.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_documents","title":"read_block_documents async","text":"

    Read block documents

    Parameters:

    Name Type Description Default block_schema_type Optional[str]

    an optional block schema type

    None offset Optional[int]

    an offset

    None limit Optional[int]

    the number of blocks to return

    None include_secrets bool

    whether to include secret values on the Block, corresponding to Pydantic's SecretStr and SecretBytes fields. These fields are automatically obfuscated by Pydantic, but users can additionally choose not to receive their values from the API. Note that any business logic on the Block may not work if this is False.

    True

    Returns:

    Type Description

    A list of block documents

    Source code in prefect/client/orchestration.py
    async def read_block_documents(\n    self,\n    block_schema_type: Optional[str] = None,\n    offset: Optional[int] = None,\n    limit: Optional[int] = None,\n    include_secrets: bool = True,\n):\n    \"\"\"\n    Read block documents\n\n    Args:\n        block_schema_type: an optional block schema type\n        offset: an offset\n        limit: the number of blocks to return\n        include_secrets (bool): whether to include secret values\n            on the Block, corresponding to Pydantic's `SecretStr` and\n            `SecretBytes` fields. These fields are automatically obfuscated\n            by Pydantic, but users can additionally choose not to receive\n            their values from the API. Note that any business logic on the\n            Block may not work if this is `False`.\n\n    Returns:\n        A list of block documents\n    \"\"\"\n    response = await self._client.post(\n        \"/block_documents/filter\",\n        json=dict(\n            block_schema_type=block_schema_type,\n            offset=offset,\n            limit=limit,\n            include_secrets=include_secrets,\n        ),\n    )\n    return pydantic.parse_obj_as(List[BlockDocument], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_block_documents_by_type","title":"read_block_documents_by_type async","text":"

    Retrieve block documents by block type slug.

    Parameters:

    Name Type Description Default block_type_slug str

    The block type slug.

    required offset Optional[int]

    an offset

    None limit Optional[int]

    the number of blocks to return

    None include_secrets bool

    whether to include secret values

    True

    Returns:

    Type Description List[BlockDocument]

    A list of block documents

    Source code in prefect/client/orchestration.py
    async def read_block_documents_by_type(\n    self,\n    block_type_slug: str,\n    offset: Optional[int] = None,\n    limit: Optional[int] = None,\n    include_secrets: bool = True,\n) -> List[BlockDocument]:\n    \"\"\"Retrieve block documents by block type slug.\n\n    Args:\n        block_type_slug: The block type slug.\n        offset: an offset\n        limit: the number of blocks to return\n        include_secrets: whether to include secret values\n\n    Returns:\n        A list of block documents\n    \"\"\"\n    response = await self._client.get(\n        f\"/block_types/slug/{block_type_slug}/block_documents\",\n        params=dict(\n            offset=offset,\n            limit=limit,\n            include_secrets=include_secrets,\n        ),\n    )\n\n    return pydantic.parse_obj_as(List[BlockDocument], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_deployment","title":"create_deployment async","text":"

    Create a deployment.

    Parameters:

    Name Type Description Default flow_id UUID

    the flow ID to create a deployment for

    required name str

    the name of the deployment

    required version str

    an optional version string for the deployment

    None schedule SCHEDULE_TYPES

    an optional schedule to apply to the deployment

    None tags List[str]

    an optional list of tags to apply to the deployment

    None storage_document_id UUID

    an reference to the storage block document used for the deployed flow

    None infrastructure_document_id UUID

    an reference to the infrastructure block document to use for this deployment

    None

    Raises:

    Type Description RequestError

    if the deployment was not created for any reason

    Returns:

    Type Description UUID

    the ID of the deployment in the backend

    Source code in prefect/client/orchestration.py
    async def create_deployment(\n    self,\n    flow_id: UUID,\n    name: str,\n    version: str = None,\n    schedule: SCHEDULE_TYPES = None,\n    parameters: Dict[str, Any] = None,\n    description: str = None,\n    work_queue_name: str = None,\n    work_pool_name: str = None,\n    tags: List[str] = None,\n    storage_document_id: UUID = None,\n    manifest_path: str = None,\n    path: str = None,\n    entrypoint: str = None,\n    infrastructure_document_id: UUID = None,\n    infra_overrides: Dict[str, Any] = None,\n    parameter_openapi_schema: dict = None,\n    is_schedule_active: Optional[bool] = None,\n    pull_steps: Optional[List[dict]] = None,\n    enforce_parameter_schema: Optional[bool] = None,\n) -> UUID:\n    \"\"\"\n    Create a deployment.\n\n    Args:\n        flow_id: the flow ID to create a deployment for\n        name: the name of the deployment\n        version: an optional version string for the deployment\n        schedule: an optional schedule to apply to the deployment\n        tags: an optional list of tags to apply to the deployment\n        storage_document_id: an reference to the storage block document\n            used for the deployed flow\n        infrastructure_document_id: an reference to the infrastructure block document\n            to use for this deployment\n\n    Raises:\n        httpx.RequestError: if the deployment was not created for any reason\n\n    Returns:\n        the ID of the deployment in the backend\n    \"\"\"\n    deployment_create = DeploymentCreate(\n        flow_id=flow_id,\n        name=name,\n        version=version,\n        schedule=schedule,\n        parameters=dict(parameters or {}),\n        tags=list(tags or []),\n        work_queue_name=work_queue_name,\n        description=description,\n        storage_document_id=storage_document_id,\n        path=path,\n        entrypoint=entrypoint,\n        manifest_path=manifest_path,  # for backwards compat\n        infrastructure_document_id=infrastructure_document_id,\n        infra_overrides=infra_overrides or {},\n        parameter_openapi_schema=parameter_openapi_schema,\n        is_schedule_active=is_schedule_active,\n        pull_steps=pull_steps,\n        enforce_parameter_schema=enforce_parameter_schema,\n    )\n\n    if work_pool_name is not None:\n        deployment_create.work_pool_name = work_pool_name\n\n    # Exclude newer fields that are not set to avoid compatibility issues\n    exclude = {\n        field\n        for field in [\"work_pool_name\", \"work_queue_name\"]\n        if field not in deployment_create.__fields_set__\n    }\n\n    if deployment_create.is_schedule_active is None:\n        exclude.add(\"is_schedule_active\")\n\n    if deployment_create.pull_steps is None:\n        exclude.add(\"pull_steps\")\n\n    if deployment_create.enforce_parameter_schema is None:\n        exclude.add(\"enforce_parameter_schema\")\n\n    json = deployment_create.dict(json_compatible=True, exclude=exclude)\n    response = await self._client.post(\n        \"/deployments/\",\n        json=json,\n    )\n    deployment_id = response.json().get(\"id\")\n    if not deployment_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    return UUID(deployment_id)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_deployment","title":"read_deployment async","text":"

    Query the Prefect API for a deployment by id.

    Parameters:

    Name Type Description Default deployment_id UUID

    the deployment ID of interest

    required

    Returns:

    Type Description DeploymentResponse

    a Deployment model representation of the deployment

    Source code in prefect/client/orchestration.py
    async def read_deployment(\n    self,\n    deployment_id: UUID,\n) -> DeploymentResponse:\n    \"\"\"\n    Query the Prefect API for a deployment by id.\n\n    Args:\n        deployment_id: the deployment ID of interest\n\n    Returns:\n        a [Deployment model][prefect.client.schemas.objects.Deployment] representation of the deployment\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/deployments/{deployment_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return DeploymentResponse.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_deployment_by_name","title":"read_deployment_by_name async","text":"

    Query the Prefect API for a deployment by name.

    Parameters:

    Name Type Description Default name str

    A deployed flow's name: / required

    Raises:

    Type Description ObjectNotFound

    If request returns 404

    RequestError

    If request fails

    Returns:

    Type Description DeploymentResponse

    a Deployment model representation of the deployment

    Source code in prefect/client/orchestration.py
    async def read_deployment_by_name(\n    self,\n    name: str,\n) -> DeploymentResponse:\n    \"\"\"\n    Query the Prefect API for a deployment by name.\n\n    Args:\n        name: A deployed flow's name: <FLOW_NAME>/<DEPLOYMENT_NAME>\n\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If request fails\n\n    Returns:\n        a Deployment model representation of the deployment\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/deployments/name/{name}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n\n    return DeploymentResponse.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_deployments","title":"read_deployments async","text":"

    Query the Prefect API for deployments. Only deployments matching all the provided criteria will be returned.

    Parameters:

    Name Type Description Default flow_filter FlowFilter

    filter criteria for flows

    None flow_run_filter FlowRunFilter

    filter criteria for flow runs

    None task_run_filter TaskRunFilter

    filter criteria for task runs

    None deployment_filter DeploymentFilter

    filter criteria for deployments

    None work_pool_filter WorkPoolFilter

    filter criteria for work pools

    None work_queue_filter WorkQueueFilter

    filter criteria for work pool queues

    None limit int

    a limit for the deployment query

    None offset int

    an offset for the deployment query

    0

    Returns:

    Type Description List[DeploymentResponse]

    a list of Deployment model representations of the deployments

    Source code in prefect/client/orchestration.py
    async def read_deployments(\n    self,\n    *,\n    flow_filter: FlowFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    deployment_filter: DeploymentFilter = None,\n    work_pool_filter: WorkPoolFilter = None,\n    work_queue_filter: WorkQueueFilter = None,\n    limit: int = None,\n    sort: DeploymentSort = None,\n    offset: int = 0,\n) -> List[DeploymentResponse]:\n    \"\"\"\n    Query the Prefect API for deployments. Only deployments matching all\n    the provided criteria will be returned.\n\n    Args:\n        flow_filter: filter criteria for flows\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        deployment_filter: filter criteria for deployments\n        work_pool_filter: filter criteria for work pools\n        work_queue_filter: filter criteria for work pool queues\n        limit: a limit for the deployment query\n        offset: an offset for the deployment query\n\n    Returns:\n        a list of Deployment model representations\n            of the deployments\n    \"\"\"\n    body = {\n        \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n            if flow_run_filter\n            else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"deployments\": (\n            deployment_filter.dict(json_compatible=True)\n            if deployment_filter\n            else None\n        ),\n        \"work_pools\": (\n            work_pool_filter.dict(json_compatible=True)\n            if work_pool_filter\n            else None\n        ),\n        \"work_pool_queues\": (\n            work_queue_filter.dict(json_compatible=True)\n            if work_queue_filter\n            else None\n        ),\n        \"limit\": limit,\n        \"offset\": offset,\n        \"sort\": sort,\n    }\n\n    response = await self._client.post(\"/deployments/filter\", json=body)\n    return pydantic.parse_obj_as(List[DeploymentResponse], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_deployment","title":"delete_deployment async","text":"

    Delete deployment by id.

    Parameters:

    Name Type Description Default deployment_id UUID

    The deployment id of interest.

    required Source code in prefect/client/orchestration.py
    async def delete_deployment(\n    self,\n    deployment_id: UUID,\n):\n    \"\"\"\n    Delete deployment by id.\n\n    Args:\n        deployment_id: The deployment id of interest.\n    Raises:\n        prefect.exceptions.ObjectNotFound: If request returns 404\n        httpx.RequestError: If requests fails\n    \"\"\"\n    try:\n        await self._client.delete(f\"/deployments/{deployment_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_run","title":"read_flow_run async","text":"

    Query the Prefect API for a flow run by id.

    Parameters:

    Name Type Description Default flow_run_id UUID

    the flow run ID of interest

    required

    Returns:

    Type Description FlowRun

    a Flow Run model representation of the flow run

    Source code in prefect/client/orchestration.py
    async def read_flow_run(self, flow_run_id: UUID) -> FlowRun:\n    \"\"\"\n    Query the Prefect API for a flow run by id.\n\n    Args:\n        flow_run_id: the flow run ID of interest\n\n    Returns:\n        a Flow Run model representation of the flow run\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/flow_runs/{flow_run_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n    return FlowRun.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.resume_flow_run","title":"resume_flow_run async","text":"

    Resumes a paused flow run.

    Parameters:

    Name Type Description Default flow_run_id UUID

    the flow run ID of interest

    required run_input Optional[Dict]

    the input to resume the flow run with

    None

    Returns:

    Type Description OrchestrationResult

    an OrchestrationResult model representation of state orchestration output

    Source code in prefect/client/orchestration.py
    async def resume_flow_run(\n    self, flow_run_id: UUID, run_input: Optional[Dict] = None\n) -> OrchestrationResult:\n    \"\"\"\n    Resumes a paused flow run.\n\n    Args:\n        flow_run_id: the flow run ID of interest\n        run_input: the input to resume the flow run with\n\n    Returns:\n        an OrchestrationResult model representation of state orchestration output\n    \"\"\"\n    try:\n        response = await self._client.post(\n            f\"/flow_runs/{flow_run_id}/resume\", json={\"run_input\": run_input}\n        )\n    except httpx.HTTPStatusError:\n        raise\n\n    return OrchestrationResult.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_runs","title":"read_flow_runs async","text":"

    Query the Prefect API for flow runs. Only flow runs matching all criteria will be returned.

    Parameters:

    Name Type Description Default flow_filter FlowFilter

    filter criteria for flows

    None flow_run_filter FlowRunFilter

    filter criteria for flow runs

    None task_run_filter TaskRunFilter

    filter criteria for task runs

    None deployment_filter DeploymentFilter

    filter criteria for deployments

    None work_pool_filter WorkPoolFilter

    filter criteria for work pools

    None work_queue_filter WorkQueueFilter

    filter criteria for work pool queues

    None sort FlowRunSort

    sort criteria for the flow runs

    None limit int

    limit for the flow run query

    None offset int

    offset for the flow run query

    0

    Returns:

    Type Description List[FlowRun]

    a list of Flow Run model representations of the flow runs

    Source code in prefect/client/orchestration.py
    async def read_flow_runs(\n    self,\n    *,\n    flow_filter: FlowFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    deployment_filter: DeploymentFilter = None,\n    work_pool_filter: WorkPoolFilter = None,\n    work_queue_filter: WorkQueueFilter = None,\n    sort: FlowRunSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[FlowRun]:\n    \"\"\"\n    Query the Prefect API for flow runs. Only flow runs matching all criteria will\n    be returned.\n\n    Args:\n        flow_filter: filter criteria for flows\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        deployment_filter: filter criteria for deployments\n        work_pool_filter: filter criteria for work pools\n        work_queue_filter: filter criteria for work pool queues\n        sort: sort criteria for the flow runs\n        limit: limit for the flow run query\n        offset: offset for the flow run query\n\n    Returns:\n        a list of Flow Run model representations\n            of the flow runs\n    \"\"\"\n    body = {\n        \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n            if flow_run_filter\n            else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"deployments\": (\n            deployment_filter.dict(json_compatible=True)\n            if deployment_filter\n            else None\n        ),\n        \"work_pools\": (\n            work_pool_filter.dict(json_compatible=True)\n            if work_pool_filter\n            else None\n        ),\n        \"work_pool_queues\": (\n            work_queue_filter.dict(json_compatible=True)\n            if work_queue_filter\n            else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n\n    response = await self._client.post(\"/flow_runs/filter\", json=body)\n    return pydantic.parse_obj_as(List[FlowRun], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.set_flow_run_state","title":"set_flow_run_state async","text":"

    Set the state of a flow run.

    Parameters:

    Name Type Description Default flow_run_id UUID

    the id of the flow run

    required state State

    the state to set

    required force bool

    if True, disregard orchestration logic when setting the state, forcing the Prefect API to accept the state

    False

    Returns:

    Type Description OrchestrationResult

    an OrchestrationResult model representation of state orchestration output

    Source code in prefect/client/orchestration.py
    async def set_flow_run_state(\n    self,\n    flow_run_id: UUID,\n    state: \"prefect.states.State\",\n    force: bool = False,\n) -> OrchestrationResult:\n    \"\"\"\n    Set the state of a flow run.\n\n    Args:\n        flow_run_id: the id of the flow run\n        state: the state to set\n        force: if True, disregard orchestration logic when setting the state,\n            forcing the Prefect API to accept the state\n\n    Returns:\n        an OrchestrationResult model representation of state orchestration output\n    \"\"\"\n    state_create = state.to_state_create()\n    state_create.state_details.flow_run_id = flow_run_id\n    try:\n        response = await self._client.post(\n            f\"/flow_runs/{flow_run_id}/set_state\",\n            json=dict(state=state_create.dict(json_compatible=True), force=force),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n\n    return OrchestrationResult.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_run_states","title":"read_flow_run_states async","text":"

    Query for the states of a flow run

    Parameters:

    Name Type Description Default flow_run_id UUID

    the id of the flow run

    required

    Returns:

    Type Description List[State]

    a list of State model representations of the flow run states

    Source code in prefect/client/orchestration.py
    async def read_flow_run_states(\n    self, flow_run_id: UUID\n) -> List[prefect.states.State]:\n    \"\"\"\n    Query for the states of a flow run\n\n    Args:\n        flow_run_id: the id of the flow run\n\n    Returns:\n        a list of State model representations\n            of the flow run states\n    \"\"\"\n    response = await self._client.get(\n        \"/flow_run_states/\", params=dict(flow_run_id=str(flow_run_id))\n    )\n    return pydantic.parse_obj_as(List[prefect.states.State], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_task_run","title":"create_task_run async","text":"

    Create a task run

    Parameters:

    Name Type Description Default task Task

    The Task to run

    required flow_run_id UUID

    The flow run id with which to associate the task run

    required dynamic_key str

    A key unique to this particular run of a Task within the flow

    required name str

    An optional name for the task run

    None extra_tags Iterable[str]

    an optional list of extra tags to apply to the task run in addition to task.tags

    None state State

    The initial state for the run. If not provided, defaults to Pending for now. Should always be a Scheduled type.

    None task_inputs Dict[str, List[Union[TaskRunResult, Parameter, Constant]]]

    the set of inputs passed to the task

    None

    Returns:

    Type Description TaskRun

    The created task run.

    Source code in prefect/client/orchestration.py
    async def create_task_run(\n    self,\n    task: \"TaskObject\",\n    flow_run_id: UUID,\n    dynamic_key: str,\n    name: str = None,\n    extra_tags: Iterable[str] = None,\n    state: prefect.states.State = None,\n    task_inputs: Dict[\n        str,\n        List[\n            Union[\n                TaskRunResult,\n                Parameter,\n                Constant,\n            ]\n        ],\n    ] = None,\n) -> TaskRun:\n    \"\"\"\n    Create a task run\n\n    Args:\n        task: The Task to run\n        flow_run_id: The flow run id with which to associate the task run\n        dynamic_key: A key unique to this particular run of a Task within the flow\n        name: An optional name for the task run\n        extra_tags: an optional list of extra tags to apply to the task run in\n            addition to `task.tags`\n        state: The initial state for the run. If not provided, defaults to\n            `Pending` for now. Should always be a `Scheduled` type.\n        task_inputs: the set of inputs passed to the task\n\n    Returns:\n        The created task run.\n    \"\"\"\n    tags = set(task.tags).union(extra_tags or [])\n\n    if state is None:\n        state = prefect.states.Pending()\n\n    task_run_data = TaskRunCreate(\n        name=name,\n        flow_run_id=flow_run_id,\n        task_key=task.task_key,\n        dynamic_key=dynamic_key,\n        tags=list(tags),\n        task_version=task.version,\n        empirical_policy=TaskRunPolicy(\n            retries=task.retries,\n            retry_delay=task.retry_delay_seconds,\n            retry_jitter_factor=task.retry_jitter_factor,\n        ),\n        state=state.to_state_create(),\n        task_inputs=task_inputs or {},\n    )\n\n    response = await self._client.post(\n        \"/task_runs/\", json=task_run_data.dict(json_compatible=True)\n    )\n    return TaskRun.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_task_run","title":"read_task_run async","text":"

    Query the Prefect API for a task run by id.

    Parameters:

    Name Type Description Default task_run_id UUID

    the task run ID of interest

    required

    Returns:

    Type Description TaskRun

    a Task Run model representation of the task run

    Source code in prefect/client/orchestration.py
    async def read_task_run(self, task_run_id: UUID) -> TaskRun:\n    \"\"\"\n    Query the Prefect API for a task run by id.\n\n    Args:\n        task_run_id: the task run ID of interest\n\n    Returns:\n        a Task Run model representation of the task run\n    \"\"\"\n    response = await self._client.get(f\"/task_runs/{task_run_id}\")\n    return TaskRun.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_task_runs","title":"read_task_runs async","text":"

    Query the Prefect API for task runs. Only task runs matching all criteria will be returned.

    Parameters:

    Name Type Description Default flow_filter FlowFilter

    filter criteria for flows

    None flow_run_filter FlowRunFilter

    filter criteria for flow runs

    None task_run_filter TaskRunFilter

    filter criteria for task runs

    None deployment_filter DeploymentFilter

    filter criteria for deployments

    None sort TaskRunSort

    sort criteria for the task runs

    None limit int

    a limit for the task run query

    None offset int

    an offset for the task run query

    0

    Returns:

    Type Description List[TaskRun]

    a list of Task Run model representations of the task runs

    Source code in prefect/client/orchestration.py
    async def read_task_runs(\n    self,\n    *,\n    flow_filter: FlowFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    deployment_filter: DeploymentFilter = None,\n    sort: TaskRunSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[TaskRun]:\n    \"\"\"\n    Query the Prefect API for task runs. Only task runs matching all criteria will\n    be returned.\n\n    Args:\n        flow_filter: filter criteria for flows\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        deployment_filter: filter criteria for deployments\n        sort: sort criteria for the task runs\n        limit: a limit for the task run query\n        offset: an offset for the task run query\n\n    Returns:\n        a list of Task Run model representations\n            of the task runs\n    \"\"\"\n    body = {\n        \"flows\": flow_filter.dict(json_compatible=True) if flow_filter else None,\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True, exclude_unset=True)\n            if flow_run_filter\n            else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"deployments\": (\n            deployment_filter.dict(json_compatible=True)\n            if deployment_filter\n            else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n    response = await self._client.post(\"/task_runs/filter\", json=body)\n    return pydantic.parse_obj_as(List[TaskRun], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.set_task_run_state","title":"set_task_run_state async","text":"

    Set the state of a task run.

    Parameters:

    Name Type Description Default task_run_id UUID

    the id of the task run

    required state State

    the state to set

    required force bool

    if True, disregard orchestration logic when setting the state, forcing the Prefect API to accept the state

    False

    Returns:

    Type Description OrchestrationResult

    an OrchestrationResult model representation of state orchestration output

    Source code in prefect/client/orchestration.py
    async def set_task_run_state(\n    self,\n    task_run_id: UUID,\n    state: prefect.states.State,\n    force: bool = False,\n) -> OrchestrationResult:\n    \"\"\"\n    Set the state of a task run.\n\n    Args:\n        task_run_id: the id of the task run\n        state: the state to set\n        force: if True, disregard orchestration logic when setting the state,\n            forcing the Prefect API to accept the state\n\n    Returns:\n        an OrchestrationResult model representation of state orchestration output\n    \"\"\"\n    state_create = state.to_state_create()\n    state_create.state_details.task_run_id = task_run_id\n    response = await self._client.post(\n        f\"/task_runs/{task_run_id}/set_state\",\n        json=dict(state=state_create.dict(json_compatible=True), force=force),\n    )\n    return OrchestrationResult.parse_obj(response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_task_run_states","title":"read_task_run_states async","text":"

    Query for the states of a task run

    Parameters:

    Name Type Description Default task_run_id UUID

    the id of the task run

    required

    Returns:

    Type Description List[State]

    a list of State model representations of the task run states

    Source code in prefect/client/orchestration.py
    async def read_task_run_states(\n    self, task_run_id: UUID\n) -> List[prefect.states.State]:\n    \"\"\"\n    Query for the states of a task run\n\n    Args:\n        task_run_id: the id of the task run\n\n    Returns:\n        a list of State model representations of the task run states\n    \"\"\"\n    response = await self._client.get(\n        \"/task_run_states/\", params=dict(task_run_id=str(task_run_id))\n    )\n    return pydantic.parse_obj_as(List[prefect.states.State], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_logs","title":"create_logs async","text":"

    Create logs for a flow or task run

    Parameters:

    Name Type Description Default logs Iterable[Union[LogCreate, dict]]

    An iterable of LogCreate objects or already json-compatible dicts

    required Source code in prefect/client/orchestration.py
    async def create_logs(self, logs: Iterable[Union[LogCreate, dict]]) -> None:\n    \"\"\"\n    Create logs for a flow or task run\n\n    Args:\n        logs: An iterable of `LogCreate` objects or already json-compatible dicts\n    \"\"\"\n    serialized_logs = [\n        log.dict(json_compatible=True) if isinstance(log, LogCreate) else log\n        for log in logs\n    ]\n    await self._client.post(\"/logs/\", json=serialized_logs)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_run_notification_policy","title":"create_flow_run_notification_policy async","text":"

    Create a notification policy for flow runs

    Parameters:

    Name Type Description Default block_document_id UUID

    The block document UUID

    required is_active bool

    Whether the notification policy is active

    True tags List[str]

    List of flow tags

    None state_names List[str]

    List of state names

    None message_template Optional[str]

    Notification message template

    None Source code in prefect/client/orchestration.py
    async def create_flow_run_notification_policy(\n    self,\n    block_document_id: UUID,\n    is_active: bool = True,\n    tags: List[str] = None,\n    state_names: List[str] = None,\n    message_template: Optional[str] = None,\n) -> UUID:\n    \"\"\"\n    Create a notification policy for flow runs\n\n    Args:\n        block_document_id: The block document UUID\n        is_active: Whether the notification policy is active\n        tags: List of flow tags\n        state_names: List of state names\n        message_template: Notification message template\n    \"\"\"\n    if tags is None:\n        tags = []\n    if state_names is None:\n        state_names = []\n\n    policy = FlowRunNotificationPolicyCreate(\n        block_document_id=block_document_id,\n        is_active=is_active,\n        tags=tags,\n        state_names=state_names,\n        message_template=message_template,\n    )\n    response = await self._client.post(\n        \"/flow_run_notification_policies/\",\n        json=policy.dict(json_compatible=True),\n    )\n\n    policy_id = response.json().get(\"id\")\n    if not policy_id:\n        raise httpx.RequestError(f\"Malformed response: {response}\")\n\n    return UUID(policy_id)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_run_notification_policies","title":"read_flow_run_notification_policies async","text":"

    Query the Prefect API for flow run notification policies. Only policies matching all criteria will be returned.

    Parameters:

    Name Type Description Default flow_run_notification_policy_filter FlowRunNotificationPolicyFilter

    filter criteria for notification policies

    required limit Optional[int]

    a limit for the notification policies query

    None offset int

    an offset for the notification policies query

    0

    Returns:

    Type Description List[FlowRunNotificationPolicy]

    a list of FlowRunNotificationPolicy model representations of the notification policies

    Source code in prefect/client/orchestration.py
    async def read_flow_run_notification_policies(\n    self,\n    flow_run_notification_policy_filter: FlowRunNotificationPolicyFilter,\n    limit: Optional[int] = None,\n    offset: int = 0,\n) -> List[FlowRunNotificationPolicy]:\n    \"\"\"\n    Query the Prefect API for flow run notification policies. Only policies matching all criteria will\n    be returned.\n\n    Args:\n        flow_run_notification_policy_filter: filter criteria for notification policies\n        limit: a limit for the notification policies query\n        offset: an offset for the notification policies query\n\n    Returns:\n        a list of FlowRunNotificationPolicy model representations\n            of the notification policies\n    \"\"\"\n    body = {\n        \"flow_run_notification_policy_filter\": (\n            flow_run_notification_policy_filter.dict(json_compatible=True)\n            if flow_run_notification_policy_filter\n            else None\n        ),\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n    response = await self._client.post(\n        \"/flow_run_notification_policies/filter\", json=body\n    )\n    return pydantic.parse_obj_as(List[FlowRunNotificationPolicy], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_logs","title":"read_logs async","text":"

    Read flow and task run logs.

    Source code in prefect/client/orchestration.py
    async def read_logs(\n    self,\n    log_filter: LogFilter = None,\n    limit: int = None,\n    offset: int = None,\n    sort: LogSort = LogSort.TIMESTAMP_ASC,\n) -> List[Log]:\n    \"\"\"\n    Read flow and task run logs.\n    \"\"\"\n    body = {\n        \"logs\": log_filter.dict(json_compatible=True) if log_filter else None,\n        \"limit\": limit,\n        \"offset\": offset,\n        \"sort\": sort,\n    }\n\n    response = await self._client.post(\"/logs/filter\", json=body)\n    return pydantic.parse_obj_as(List[Log], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.resolve_datadoc","title":"resolve_datadoc async","text":"

    Recursively decode possibly nested data documents.

    \"server\" encoded documents will be retrieved from the server.

    Parameters:

    Name Type Description Default datadoc DataDocument

    The data document to resolve

    required

    Returns:

    Type Description Any

    a decoded object, the innermost data

    Source code in prefect/client/orchestration.py
    async def resolve_datadoc(self, datadoc: DataDocument) -> Any:\n    \"\"\"\n    Recursively decode possibly nested data documents.\n\n    \"server\" encoded documents will be retrieved from the server.\n\n    Args:\n        datadoc: The data document to resolve\n\n    Returns:\n        a decoded object, the innermost data\n    \"\"\"\n    if not isinstance(datadoc, DataDocument):\n        raise TypeError(\n            f\"`resolve_datadoc` received invalid type {type(datadoc).__name__}\"\n        )\n\n    async def resolve_inner(data):\n        if isinstance(data, bytes):\n            try:\n                data = DataDocument.parse_raw(data)\n            except pydantic.ValidationError:\n                return data\n\n        if isinstance(data, DataDocument):\n            return await resolve_inner(data.decode())\n\n        return data\n\n    return await resolve_inner(datadoc)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.send_worker_heartbeat","title":"send_worker_heartbeat async","text":"

    Sends a worker heartbeat for a given work pool.

    Parameters:

    Name Type Description Default work_pool_name str

    The name of the work pool to heartbeat against.

    required worker_name str

    The name of the worker sending the heartbeat.

    required Source code in prefect/client/orchestration.py
    async def send_worker_heartbeat(\n    self,\n    work_pool_name: str,\n    worker_name: str,\n    heartbeat_interval_seconds: Optional[float] = None,\n):\n    \"\"\"\n    Sends a worker heartbeat for a given work pool.\n\n    Args:\n        work_pool_name: The name of the work pool to heartbeat against.\n        worker_name: The name of the worker sending the heartbeat.\n    \"\"\"\n    await self._client.post(\n        f\"/work_pools/{work_pool_name}/workers/heartbeat\",\n        json={\n            \"name\": worker_name,\n            \"heartbeat_interval_seconds\": heartbeat_interval_seconds,\n        },\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_workers_for_work_pool","title":"read_workers_for_work_pool async","text":"

    Reads workers for a given work pool.

    Parameters:

    Name Type Description Default work_pool_name str

    The name of the work pool for which to get member workers.

    required worker_filter Optional[WorkerFilter]

    Criteria by which to filter workers.

    None limit Optional[int]

    Limit for the worker query.

    None offset Optional[int]

    Limit for the worker query.

    None Source code in prefect/client/orchestration.py
    async def read_workers_for_work_pool(\n    self,\n    work_pool_name: str,\n    worker_filter: Optional[WorkerFilter] = None,\n    offset: Optional[int] = None,\n    limit: Optional[int] = None,\n) -> List[Worker]:\n    \"\"\"\n    Reads workers for a given work pool.\n\n    Args:\n        work_pool_name: The name of the work pool for which to get\n            member workers.\n        worker_filter: Criteria by which to filter workers.\n        limit: Limit for the worker query.\n        offset: Limit for the worker query.\n    \"\"\"\n    response = await self._client.post(\n        f\"/work_pools/{work_pool_name}/workers/filter\",\n        json={\n            \"worker_filter\": (\n                worker_filter.dict(json_compatible=True, exclude_unset=True)\n                if worker_filter\n                else None\n            ),\n            \"offset\": offset,\n            \"limit\": limit,\n        },\n    )\n\n    return pydantic.parse_obj_as(List[Worker], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_pool","title":"read_work_pool async","text":"

    Reads information for a given work pool

    Parameters:

    Name Type Description Default work_pool_name str

    The name of the work pool to for which to get information.

    required

    Returns:

    Type Description WorkPool

    Information about the requested work pool.

    Source code in prefect/client/orchestration.py
    async def read_work_pool(self, work_pool_name: str) -> WorkPool:\n    \"\"\"\n    Reads information for a given work pool\n\n    Args:\n        work_pool_name: The name of the work pool to for which to get\n            information.\n\n    Returns:\n        Information about the requested work pool.\n    \"\"\"\n    try:\n        response = await self._client.get(f\"/work_pools/{work_pool_name}\")\n        return pydantic.parse_obj_as(WorkPool, response.json())\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_pools","title":"read_work_pools async","text":"

    Reads work pools.

    Parameters:

    Name Type Description Default limit Optional[int]

    Limit for the work pool query.

    None offset int

    Offset for the work pool query.

    0 work_pool_filter Optional[WorkPoolFilter]

    Criteria by which to filter work pools.

    None

    Returns:

    Type Description List[WorkPool]

    A list of work pools.

    Source code in prefect/client/orchestration.py
    async def read_work_pools(\n    self,\n    limit: Optional[int] = None,\n    offset: int = 0,\n    work_pool_filter: Optional[WorkPoolFilter] = None,\n) -> List[WorkPool]:\n    \"\"\"\n    Reads work pools.\n\n    Args:\n        limit: Limit for the work pool query.\n        offset: Offset for the work pool query.\n        work_pool_filter: Criteria by which to filter work pools.\n\n    Returns:\n        A list of work pools.\n    \"\"\"\n\n    body = {\n        \"limit\": limit,\n        \"offset\": offset,\n        \"work_pools\": (\n            work_pool_filter.dict(json_compatible=True)\n            if work_pool_filter\n            else None\n        ),\n    }\n    response = await self._client.post(\"/work_pools/filter\", json=body)\n    return pydantic.parse_obj_as(List[WorkPool], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_work_pool","title":"create_work_pool async","text":"

    Creates a work pool with the provided configuration.

    Parameters:

    Name Type Description Default work_pool WorkPoolCreate

    Desired configuration for the new work pool.

    required

    Returns:

    Type Description WorkPool

    Information about the newly created work pool.

    Source code in prefect/client/orchestration.py
    async def create_work_pool(\n    self,\n    work_pool: WorkPoolCreate,\n) -> WorkPool:\n    \"\"\"\n    Creates a work pool with the provided configuration.\n\n    Args:\n        work_pool: Desired configuration for the new work pool.\n\n    Returns:\n        Information about the newly created work pool.\n    \"\"\"\n    try:\n        response = await self._client.post(\n            \"/work_pools/\",\n            json=work_pool.dict(json_compatible=True, exclude_unset=True),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_409_CONFLICT:\n            raise prefect.exceptions.ObjectAlreadyExists(http_exc=e) from e\n        else:\n            raise\n\n    return pydantic.parse_obj_as(WorkPool, response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.update_work_pool","title":"update_work_pool async","text":"

    Updates a work pool.

    Parameters:

    Name Type Description Default work_pool_name str

    Name of the work pool to update.

    required work_pool WorkPoolUpdate

    Fields to update in the work pool.

    required Source code in prefect/client/orchestration.py
    async def update_work_pool(\n    self,\n    work_pool_name: str,\n    work_pool: WorkPoolUpdate,\n):\n    \"\"\"\n    Updates a work pool.\n\n    Args:\n        work_pool_name: Name of the work pool to update.\n        work_pool: Fields to update in the work pool.\n    \"\"\"\n    try:\n        await self._client.patch(\n            f\"/work_pools/{work_pool_name}\",\n            json=work_pool.dict(json_compatible=True, exclude_unset=True),\n        )\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_work_pool","title":"delete_work_pool async","text":"

    Deletes a work pool.

    Parameters:

    Name Type Description Default work_pool_name str

    Name of the work pool to delete.

    required Source code in prefect/client/orchestration.py
    async def delete_work_pool(\n    self,\n    work_pool_name: str,\n):\n    \"\"\"\n    Deletes a work pool.\n\n    Args:\n        work_pool_name: Name of the work pool to delete.\n    \"\"\"\n    try:\n        await self._client.delete(f\"/work_pools/{work_pool_name}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_work_queues","title":"read_work_queues async","text":"

    Retrieves queues for a work pool.

    Parameters:

    Name Type Description Default work_pool_name Optional[str]

    Name of the work pool for which to get queues.

    None work_queue_filter Optional[WorkQueueFilter]

    Criteria by which to filter queues.

    None limit Optional[int]

    Limit for the queue query.

    None offset Optional[int]

    Limit for the queue query.

    None

    Returns:

    Type Description List[WorkQueue]

    List of queues for the specified work pool.

    Source code in prefect/client/orchestration.py
    async def read_work_queues(\n    self,\n    work_pool_name: Optional[str] = None,\n    work_queue_filter: Optional[WorkQueueFilter] = None,\n    limit: Optional[int] = None,\n    offset: Optional[int] = None,\n) -> List[WorkQueue]:\n    \"\"\"\n    Retrieves queues for a work pool.\n\n    Args:\n        work_pool_name: Name of the work pool for which to get queues.\n        work_queue_filter: Criteria by which to filter queues.\n        limit: Limit for the queue query.\n        offset: Limit for the queue query.\n\n    Returns:\n        List of queues for the specified work pool.\n    \"\"\"\n    json = {\n        \"work_queues\": (\n            work_queue_filter.dict(json_compatible=True, exclude_unset=True)\n            if work_queue_filter\n            else None\n        ),\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n\n    if work_pool_name:\n        try:\n            response = await self._client.post(\n                f\"/work_pools/{work_pool_name}/queues/filter\",\n                json=json,\n            )\n        except httpx.HTTPStatusError as e:\n            if e.response.status_code == status.HTTP_404_NOT_FOUND:\n                raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n            else:\n                raise\n    else:\n        response = await self._client.post(\"/work_queues/filter\", json=json)\n\n    return pydantic.parse_obj_as(List[WorkQueue], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.get_scheduled_flow_runs_for_work_pool","title":"get_scheduled_flow_runs_for_work_pool async","text":"

    Retrieves scheduled flow runs for the provided set of work pool queues.

    Parameters:

    Name Type Description Default work_pool_name str

    The name of the work pool that the work pool queues are associated with.

    required work_queue_names Optional[List[str]]

    The names of the work pool queues from which to get scheduled flow runs.

    None scheduled_before Optional[datetime]

    Datetime used to filter returned flow runs. Flow runs scheduled for after the given datetime string will not be returned.

    None

    Returns:

    Type Description List[WorkerFlowRunResponse]

    A list of worker flow run responses containing information about the

    List[WorkerFlowRunResponse]

    retrieved flow runs.

    Source code in prefect/client/orchestration.py
    async def get_scheduled_flow_runs_for_work_pool(\n    self,\n    work_pool_name: str,\n    work_queue_names: Optional[List[str]] = None,\n    scheduled_before: Optional[datetime.datetime] = None,\n) -> List[WorkerFlowRunResponse]:\n    \"\"\"\n    Retrieves scheduled flow runs for the provided set of work pool queues.\n\n    Args:\n        work_pool_name: The name of the work pool that the work pool\n            queues are associated with.\n        work_queue_names: The names of the work pool queues from which\n            to get scheduled flow runs.\n        scheduled_before: Datetime used to filter returned flow runs. Flow runs\n            scheduled for after the given datetime string will not be returned.\n\n    Returns:\n        A list of worker flow run responses containing information about the\n        retrieved flow runs.\n    \"\"\"\n    body: Dict[str, Any] = {}\n    if work_queue_names is not None:\n        body[\"work_queue_names\"] = list(work_queue_names)\n    if scheduled_before:\n        body[\"scheduled_before\"] = str(scheduled_before)\n\n    response = await self._client.post(\n        f\"/work_pools/{work_pool_name}/get_scheduled_flow_runs\",\n        json=body,\n    )\n    return pydantic.parse_obj_as(List[WorkerFlowRunResponse], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_artifact","title":"create_artifact async","text":"

    Creates an artifact with the provided configuration.

    Parameters:

    Name Type Description Default artifact ArtifactCreate

    Desired configuration for the new artifact.

    required Source code in prefect/client/orchestration.py
    async def create_artifact(\n    self,\n    artifact: ArtifactCreate,\n) -> Artifact:\n    \"\"\"\n    Creates an artifact with the provided configuration.\n\n    Args:\n        artifact: Desired configuration for the new artifact.\n    Returns:\n        Information about the newly created artifact.\n    \"\"\"\n\n    response = await self._client.post(\n        \"/artifacts/\",\n        json=artifact.dict(json_compatible=True, exclude_unset=True),\n    )\n\n    return pydantic.parse_obj_as(Artifact, response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_artifacts","title":"read_artifacts async","text":"

    Query the Prefect API for artifacts. Only artifacts matching all criteria will be returned. Args: artifact_filter: filter criteria for artifacts flow_run_filter: filter criteria for flow runs task_run_filter: filter criteria for task runs sort: sort criteria for the artifacts limit: limit for the artifact query offset: offset for the artifact query Returns: a list of Artifact model representations of the artifacts

    Source code in prefect/client/orchestration.py
    async def read_artifacts(\n    self,\n    *,\n    artifact_filter: ArtifactFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    sort: ArtifactSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[Artifact]:\n    \"\"\"\n    Query the Prefect API for artifacts. Only artifacts matching all criteria will\n    be returned.\n    Args:\n        artifact_filter: filter criteria for artifacts\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        sort: sort criteria for the artifacts\n        limit: limit for the artifact query\n        offset: offset for the artifact query\n    Returns:\n        a list of Artifact model representations of the artifacts\n    \"\"\"\n    body = {\n        \"artifacts\": (\n            artifact_filter.dict(json_compatible=True) if artifact_filter else None\n        ),\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True) if flow_run_filter else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n    response = await self._client.post(\"/artifacts/filter\", json=body)\n    return pydantic.parse_obj_as(List[Artifact], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_latest_artifacts","title":"read_latest_artifacts async","text":"

    Query the Prefect API for artifacts. Only artifacts matching all criteria will be returned. Args: artifact_filter: filter criteria for artifacts flow_run_filter: filter criteria for flow runs task_run_filter: filter criteria for task runs sort: sort criteria for the artifacts limit: limit for the artifact query offset: offset for the artifact query Returns: a list of Artifact model representations of the artifacts

    Source code in prefect/client/orchestration.py
    async def read_latest_artifacts(\n    self,\n    *,\n    artifact_filter: ArtifactCollectionFilter = None,\n    flow_run_filter: FlowRunFilter = None,\n    task_run_filter: TaskRunFilter = None,\n    sort: ArtifactCollectionSort = None,\n    limit: int = None,\n    offset: int = 0,\n) -> List[ArtifactCollection]:\n    \"\"\"\n    Query the Prefect API for artifacts. Only artifacts matching all criteria will\n    be returned.\n    Args:\n        artifact_filter: filter criteria for artifacts\n        flow_run_filter: filter criteria for flow runs\n        task_run_filter: filter criteria for task runs\n        sort: sort criteria for the artifacts\n        limit: limit for the artifact query\n        offset: offset for the artifact query\n    Returns:\n        a list of Artifact model representations of the artifacts\n    \"\"\"\n    body = {\n        \"artifacts\": (\n            artifact_filter.dict(json_compatible=True) if artifact_filter else None\n        ),\n        \"flow_runs\": (\n            flow_run_filter.dict(json_compatible=True) if flow_run_filter else None\n        ),\n        \"task_runs\": (\n            task_run_filter.dict(json_compatible=True) if task_run_filter else None\n        ),\n        \"sort\": sort,\n        \"limit\": limit,\n        \"offset\": offset,\n    }\n    response = await self._client.post(\"/artifacts/latest/filter\", json=body)\n    return pydantic.parse_obj_as(List[ArtifactCollection], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_artifact","title":"delete_artifact async","text":"

    Deletes an artifact with the provided id.

    Parameters:

    Name Type Description Default artifact_id UUID

    The id of the artifact to delete.

    required Source code in prefect/client/orchestration.py
    async def delete_artifact(self, artifact_id: UUID) -> None:\n    \"\"\"\n    Deletes an artifact with the provided id.\n\n    Args:\n        artifact_id: The id of the artifact to delete.\n    \"\"\"\n    try:\n        await self._client.delete(f\"/artifacts/{artifact_id}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_variable_by_name","title":"read_variable_by_name async","text":"

    Reads a variable by name. Returns None if no variable is found.

    Source code in prefect/client/orchestration.py
    async def read_variable_by_name(self, name: str) -> Optional[Variable]:\n    \"\"\"Reads a variable by name. Returns None if no variable is found.\"\"\"\n    try:\n        response = await self._client.get(f\"/variables/name/{name}\")\n        return pydantic.parse_obj_as(Variable, response.json())\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == status.HTTP_404_NOT_FOUND:\n            return None\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_variable_by_name","title":"delete_variable_by_name async","text":"

    Deletes a variable by name.

    Source code in prefect/client/orchestration.py
    async def delete_variable_by_name(self, name: str):\n    \"\"\"Deletes a variable by name.\"\"\"\n    try:\n        await self._client.delete(f\"/variables/name/{name}\")\n    except httpx.HTTPStatusError as e:\n        if e.response.status_code == 404:\n            raise prefect.exceptions.ObjectNotFound(http_exc=e) from e\n        else:\n            raise\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_variables","title":"read_variables async","text":"

    Reads all variables.

    Source code in prefect/client/orchestration.py
    async def read_variables(self, limit: int = None) -> List[Variable]:\n    \"\"\"Reads all variables.\"\"\"\n    response = await self._client.post(\"/variables/filter\", json={\"limit\": limit})\n    return pydantic.parse_obj_as(List[Variable], response.json())\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_worker_metadata","title":"read_worker_metadata async","text":"

    Reads worker metadata stored in Prefect collection registry.

    Source code in prefect/client/orchestration.py
    async def read_worker_metadata(self) -> Dict[str, Any]:\n    \"\"\"Reads worker metadata stored in Prefect collection registry.\"\"\"\n    response = await self._client.get(\"collections/views/aggregate-worker-metadata\")\n    response.raise_for_status()\n    return response.json()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_automation","title":"create_automation async","text":"

    Creates an automation in Prefect Cloud.

    Source code in prefect/client/orchestration.py
    async def create_automation(self, automation: Automation) -> UUID:\n    \"\"\"Creates an automation in Prefect Cloud.\"\"\"\n    if self.server_type != ServerType.CLOUD:\n        raise RuntimeError(\"Automations are only supported for Prefect Cloud.\")\n\n    response = await self._client.post(\n        \"/automations/\",\n        json=automation.dict(json_compatible=True),\n    )\n\n    return UUID(response.json()[\"id\"])\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.create_flow_run_input","title":"create_flow_run_input async","text":"

    Creates a flow run input.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The flow run id.

    required key str

    The input key.

    required value str

    The input value.

    required sender Optional[str]

    The sender of the input.

    None Source code in prefect/client/orchestration.py
    async def create_flow_run_input(\n    self, flow_run_id: UUID, key: str, value: str, sender: Optional[str] = None\n):\n    \"\"\"\n    Creates a flow run input.\n\n    Args:\n        flow_run_id: The flow run id.\n        key: The input key.\n        value: The input value.\n        sender: The sender of the input.\n    \"\"\"\n\n    # Initialize the input to ensure that the key is valid.\n    FlowRunInput(flow_run_id=flow_run_id, key=key, value=value)\n\n    response = await self._client.post(\n        f\"/flow_runs/{flow_run_id}/input\",\n        json={\"key\": key, \"value\": value, \"sender\": sender},\n    )\n    response.raise_for_status()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.read_flow_run_input","title":"read_flow_run_input async","text":"

    Reads a flow run input.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The flow run id.

    required key str

    The input key.

    required Source code in prefect/client/orchestration.py
    async def read_flow_run_input(self, flow_run_id: UUID, key: str) -> str:\n    \"\"\"\n    Reads a flow run input.\n\n    Args:\n        flow_run_id: The flow run id.\n        key: The input key.\n    \"\"\"\n    response = await self._client.get(f\"/flow_runs/{flow_run_id}/input/{key}\")\n    response.raise_for_status()\n    return response.content.decode()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.PrefectClient.delete_flow_run_input","title":"delete_flow_run_input async","text":"

    Deletes a flow run input.

    Parameters:

    Name Type Description Default flow_run_id UUID

    The flow run id.

    required key str

    The input key.

    required Source code in prefect/client/orchestration.py
    async def delete_flow_run_input(self, flow_run_id: UUID, key: str):\n    \"\"\"\n    Deletes a flow run input.\n\n    Args:\n        flow_run_id: The flow run id.\n        key: The input key.\n    \"\"\"\n    response = await self._client.delete(f\"/flow_runs/{flow_run_id}/input/{key}\")\n    response.raise_for_status()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/orchestration/#prefect.client.orchestration.get_client","title":"get_client","text":"

    Retrieve a HTTP client for communicating with the Prefect REST API.

    The client must be context managed; for example:

    async with get_client() as client:\n    await client.hello()\n
    Source code in prefect/client/orchestration.py
    def get_client(httpx_settings: Optional[dict] = None) -> \"PrefectClient\":\n    \"\"\"\n    Retrieve a HTTP client for communicating with the Prefect REST API.\n\n    The client must be context managed; for example:\n\n    ```python\n    async with get_client() as client:\n        await client.hello()\n    ```\n    \"\"\"\n    ctx = prefect.context.get_settings_context()\n    api = PREFECT_API_URL.value()\n\n    if not api:\n        # create an ephemeral API if none was provided\n        from prefect.server.api.server import create_app\n\n        api = create_app(ctx.settings, ephemeral=True)\n\n    return PrefectClient(\n        api,\n        api_key=PREFECT_API_KEY.value(),\n        httpx_settings=httpx_settings,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_1","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions","title":"prefect.client.schemas.actions","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.StateCreate","title":"StateCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a new state.

    Source code in prefect/client/schemas/actions.py
    class StateCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a new state.\"\"\"\n\n    type: StateType\n    name: Optional[str] = Field(default=None)\n    message: Optional[str] = Field(default=None, example=\"Run started\")\n    state_details: StateDetails = Field(default_factory=StateDetails)\n    data: Union[\"BaseResult[R]\", \"DataDocument[R]\", Any] = Field(\n        default=None,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowCreate","title":"FlowCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow.\"\"\"\n\n    name: str = FieldFrom(objects.Flow)\n    tags: List[str] = FieldFrom(objects.Flow)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowUpdate","title":"FlowUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow.\"\"\"\n\n    tags: List[str] = FieldFrom(objects.Flow)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentCreate","title":"DeploymentCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a deployment.

    Source code in prefect/client/schemas/actions.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None,\n)\n@copy_model_fields\nclass DeploymentCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a deployment.\"\"\"\n\n    @root_validator(pre=True)\n    def remove_old_fields(cls, values):\n        # 2.7.7 removed worker_pool_queue_id in lieu of worker_pool_name and\n        # worker_pool_queue_name. Those fields were later renamed to work_pool_name\n        # and work_queue_name. This validator removes old fields provided\n        # by older clients to avoid 422 errors.\n        values_copy = copy(values)\n        worker_pool_queue_id = values_copy.pop(\"worker_pool_queue_id\", None)\n        worker_pool_name = values_copy.pop(\"worker_pool_name\", None)\n        worker_pool_queue_name = values_copy.pop(\"worker_pool_queue_name\", None)\n        work_pool_queue_name = values_copy.pop(\"work_pool_queue_name\", None)\n        if worker_pool_queue_id:\n            warnings.warn(\n                (\n                    \"`worker_pool_queue_id` is no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        if worker_pool_name or worker_pool_queue_name or work_pool_queue_name:\n            warnings.warn(\n                (\n                    \"`worker_pool_name`, `worker_pool_queue_name`, and \"\n                    \"`work_pool_name` are\"\n                    \"no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        return values_copy\n\n    name: str = FieldFrom(objects.Deployment)\n    flow_id: UUID = FieldFrom(objects.Deployment)\n    is_schedule_active: Optional[bool] = FieldFrom(objects.Deployment)\n    enforce_parameter_schema: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n    parameter_openapi_schema: Optional[Dict[str, Any]] = FieldFrom(objects.Deployment)\n    parameters: Dict[str, Any] = FieldFrom(objects.Deployment)\n    tags: List[str] = FieldFrom(objects.Deployment)\n    pull_steps: Optional[List[dict]] = FieldFrom(objects.Deployment)\n\n    manifest_path: Optional[str] = FieldFrom(objects.Deployment)\n    work_queue_name: Optional[str] = FieldFrom(objects.Deployment)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the deployment's work pool.\",\n        example=\"my-work-pool\",\n    )\n    storage_document_id: Optional[UUID] = FieldFrom(objects.Deployment)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.Deployment)\n    schedule: Optional[SCHEDULE_TYPES] = FieldFrom(objects.Deployment)\n    description: Optional[str] = FieldFrom(objects.Deployment)\n    path: Optional[str] = FieldFrom(objects.Deployment)\n    version: Optional[str] = FieldFrom(objects.Deployment)\n    entrypoint: Optional[str] = FieldFrom(objects.Deployment)\n    infra_overrides: Optional[Dict[str, Any]] = FieldFrom(objects.Deployment)\n\n    def check_valid_configuration(self, base_job_template: dict):\n        \"\"\"Check that the combination of base_job_template defaults\n        and infra_overrides conforms to the specified schema.\n        \"\"\"\n        variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n        if variables_schema is not None:\n            # jsonschema considers required fields, even if that field has a default,\n            # to still be required. To get around this we remove the fields from\n            # required if there is a default present.\n            required = variables_schema.get(\"required\")\n            properties = variables_schema.get(\"properties\")\n            if required is not None and properties is not None:\n                for k, v in properties.items():\n                    if \"default\" in v and k in required:\n                        required.remove(k)\n\n            jsonschema.validate(self.infra_overrides, variables_schema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentCreate.check_valid_configuration","title":"check_valid_configuration","text":"

    Check that the combination of base_job_template defaults and infra_overrides conforms to the specified schema.

    Source code in prefect/client/schemas/actions.py
    def check_valid_configuration(self, base_job_template: dict):\n    \"\"\"Check that the combination of base_job_template defaults\n    and infra_overrides conforms to the specified schema.\n    \"\"\"\n    variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n    if variables_schema is not None:\n        # jsonschema considers required fields, even if that field has a default,\n        # to still be required. To get around this we remove the fields from\n        # required if there is a default present.\n        required = variables_schema.get(\"required\")\n        properties = variables_schema.get(\"properties\")\n        if required is not None and properties is not None:\n            for k, v in properties.items():\n                if \"default\" in v and k in required:\n                    required.remove(k)\n\n        jsonschema.validate(self.infra_overrides, variables_schema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentUpdate","title":"DeploymentUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a deployment.

    Source code in prefect/client/schemas/actions.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None,\n)\n@copy_model_fields\nclass DeploymentUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a deployment.\"\"\"\n\n    @root_validator(pre=True)\n    def remove_old_fields(cls, values):\n        # 2.7.7 removed worker_pool_queue_id in lieu of worker_pool_name and\n        # worker_pool_queue_name. Those fields were later renamed to work_pool_name\n        # and work_queue_name. This validator removes old fields provided\n        # by older clients to avoid 422 errors.\n        values_copy = copy(values)\n        worker_pool_queue_id = values_copy.pop(\"worker_pool_queue_id\", None)\n        worker_pool_name = values_copy.pop(\"worker_pool_name\", None)\n        worker_pool_queue_name = values_copy.pop(\"worker_pool_queue_name\", None)\n        work_pool_queue_name = values_copy.pop(\"work_pool_queue_name\", None)\n        if worker_pool_queue_id:\n            warnings.warn(\n                (\n                    \"`worker_pool_queue_id` is no longer supported for updating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        if worker_pool_name or worker_pool_queue_name or work_pool_queue_name:\n            warnings.warn(\n                (\n                    \"`worker_pool_name`, `worker_pool_queue_name`, and \"\n                    \"`work_pool_name` are\"\n                    \"no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        return values_copy\n\n    @validator(\"schedule\")\n    def return_none_schedule(cls, v):\n        if isinstance(v, NoSchedule):\n            return None\n        return v\n\n    version: Optional[str] = FieldFrom(objects.Deployment)\n    schedule: Optional[SCHEDULE_TYPES] = FieldFrom(objects.Deployment)\n    description: Optional[str] = FieldFrom(objects.Deployment)\n    is_schedule_active: bool = FieldFrom(objects.Deployment)\n    parameters: Optional[Dict[str, Any]] = Field(\n        default=None,\n        description=\"Parameters for flow runs scheduled by the deployment.\",\n    )\n    tags: List[str] = FieldFrom(objects.Deployment)\n    work_queue_name: Optional[str] = FieldFrom(objects.Deployment)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the deployment's work pool.\",\n        example=\"my-work-pool\",\n    )\n    path: Optional[str] = FieldFrom(objects.Deployment)\n    infra_overrides: Optional[Dict[str, Any]] = FieldFrom(objects.Deployment)\n    entrypoint: Optional[str] = FieldFrom(objects.Deployment)\n    manifest_path: Optional[str] = FieldFrom(objects.Deployment)\n    storage_document_id: Optional[UUID] = FieldFrom(objects.Deployment)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.Deployment)\n    enforce_parameter_schema: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n\n    def check_valid_configuration(self, base_job_template: dict):\n        \"\"\"Check that the combination of base_job_template defaults\n        and infra_overrides conforms to the specified schema.\n        \"\"\"\n        variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n        if variables_schema is not None:\n            # jsonschema considers required fields, even if that field has a default,\n            # to still be required. To get around this we remove the fields from\n            # required if there is a default present.\n            required = variables_schema.get(\"required\")\n            properties = variables_schema.get(\"properties\")\n            if required is not None and properties is not None:\n                for k, v in properties.items():\n                    if \"default\" in v and k in required:\n                        required.remove(k)\n\n        if variables_schema is not None:\n            jsonschema.validate(self.infra_overrides, variables_schema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentUpdate.check_valid_configuration","title":"check_valid_configuration","text":"

    Check that the combination of base_job_template defaults and infra_overrides conforms to the specified schema.

    Source code in prefect/client/schemas/actions.py
    def check_valid_configuration(self, base_job_template: dict):\n    \"\"\"Check that the combination of base_job_template defaults\n    and infra_overrides conforms to the specified schema.\n    \"\"\"\n    variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n    if variables_schema is not None:\n        # jsonschema considers required fields, even if that field has a default,\n        # to still be required. To get around this we remove the fields from\n        # required if there is a default present.\n        required = variables_schema.get(\"required\")\n        properties = variables_schema.get(\"properties\")\n        if required is not None and properties is not None:\n            for k, v in properties.items():\n                if \"default\" in v and k in required:\n                    required.remove(k)\n\n    if variables_schema is not None:\n        jsonschema.validate(self.infra_overrides, variables_schema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowRunUpdate","title":"FlowRunUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow run.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowRunUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow run.\"\"\"\n\n    name: Optional[str] = FieldFrom(objects.FlowRun)\n    flow_version: Optional[str] = FieldFrom(objects.FlowRun)\n    parameters: dict = FieldFrom(objects.FlowRun)\n    empirical_policy: objects.FlowRunPolicy = FieldFrom(objects.FlowRun)\n    tags: List[str] = FieldFrom(objects.FlowRun)\n    infrastructure_pid: Optional[str] = FieldFrom(objects.FlowRun)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.TaskRunCreate","title":"TaskRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a task run

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass TaskRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a task run\"\"\"\n\n    # TaskRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the task run to create\"\n    )\n\n    name: str = FieldFrom(objects.TaskRun)\n    flow_run_id: Optional[UUID] = FieldFrom(objects.TaskRun)\n    task_key: str = FieldFrom(objects.TaskRun)\n    dynamic_key: str = FieldFrom(objects.TaskRun)\n    cache_key: Optional[str] = FieldFrom(objects.TaskRun)\n    cache_expiration: Optional[objects.DateTimeTZ] = FieldFrom(objects.TaskRun)\n    task_version: Optional[str] = FieldFrom(objects.TaskRun)\n    empirical_policy: objects.TaskRunPolicy = FieldFrom(objects.TaskRun)\n    tags: List[str] = FieldFrom(objects.TaskRun)\n    task_inputs: Dict[\n        str,\n        List[\n            Union[\n                objects.TaskRunResult,\n                objects.Parameter,\n                objects.Constant,\n            ]\n        ],\n    ] = FieldFrom(objects.TaskRun)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.TaskRunUpdate","title":"TaskRunUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a task run

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass TaskRunUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a task run\"\"\"\n\n    name: str = FieldFrom(objects.TaskRun)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowRunCreate","title":"FlowRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run.\"\"\"\n\n    # FlowRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the flow run to create\"\n    )\n\n    name: str = FieldFrom(objects.FlowRun)\n    flow_id: UUID = FieldFrom(objects.FlowRun)\n    deployment_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    flow_version: Optional[str] = FieldFrom(objects.FlowRun)\n    parameters: dict = FieldFrom(objects.FlowRun)\n    context: dict = FieldFrom(objects.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    empirical_policy: objects.FlowRunPolicy = FieldFrom(objects.FlowRun)\n    tags: List[str] = FieldFrom(objects.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(objects.FlowRun)\n\n    class Config(ActionBaseModel.Config):\n        json_dumps = orjson_dumps_extra_compatible\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.DeploymentFlowRunCreate","title":"DeploymentFlowRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run from a deployment.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass DeploymentFlowRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run from a deployment.\"\"\"\n\n    # FlowRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the flow run to create\"\n    )\n\n    name: Optional[str] = FieldFrom(objects.FlowRun)\n    parameters: dict = FieldFrom(objects.FlowRun)\n    context: dict = FieldFrom(objects.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    empirical_policy: objects.FlowRunPolicy = FieldFrom(objects.FlowRun)\n    tags: List[str] = FieldFrom(objects.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(objects.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    work_queue_name: Optional[str] = FieldFrom(objects.FlowRun)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.SavedSearchCreate","title":"SavedSearchCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a saved search.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass SavedSearchCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a saved search.\"\"\"\n\n    name: str = FieldFrom(objects.SavedSearch)\n    filters: List[objects.SavedSearchFilter] = FieldFrom(objects.SavedSearch)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.ConcurrencyLimitCreate","title":"ConcurrencyLimitCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a concurrency limit.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass ConcurrencyLimitCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a concurrency limit.\"\"\"\n\n    tag: str = FieldFrom(objects.ConcurrencyLimit)\n    concurrency_limit: int = FieldFrom(objects.ConcurrencyLimit)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockTypeCreate","title":"BlockTypeCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block type.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockTypeCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block type.\"\"\"\n\n    name: str = FieldFrom(objects.BlockType)\n    slug: str = FieldFrom(objects.BlockType)\n    logo_url: Optional[objects.HttpUrl] = FieldFrom(objects.BlockType)\n    documentation_url: Optional[objects.HttpUrl] = FieldFrom(objects.BlockType)\n    description: Optional[str] = FieldFrom(objects.BlockType)\n    code_example: Optional[str] = FieldFrom(objects.BlockType)\n\n    # validators\n    _validate_slug_format = validator(\"slug\", allow_reuse=True)(\n        validate_block_type_slug\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockTypeUpdate","title":"BlockTypeUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a block type.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockTypeUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a block type.\"\"\"\n\n    logo_url: Optional[objects.HttpUrl] = FieldFrom(objects.BlockType)\n    documentation_url: Optional[objects.HttpUrl] = FieldFrom(objects.BlockType)\n    description: Optional[str] = FieldFrom(objects.BlockType)\n    code_example: Optional[str] = FieldFrom(objects.BlockType)\n\n    @classmethod\n    def updatable_fields(cls) -> set:\n        return get_class_fields_only(cls)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockSchemaCreate","title":"BlockSchemaCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block schema.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockSchemaCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block schema.\"\"\"\n\n    fields: dict = FieldFrom(objects.BlockSchema)\n    block_type_id: Optional[UUID] = FieldFrom(objects.BlockSchema)\n    capabilities: List[str] = FieldFrom(objects.BlockSchema)\n    version: str = FieldFrom(objects.BlockSchema)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockDocumentCreate","title":"BlockDocumentCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block document.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block document.\"\"\"\n\n    name: Optional[str] = FieldFrom(objects.BlockDocument)\n    data: dict = FieldFrom(objects.BlockDocument)\n    block_schema_id: UUID = FieldFrom(objects.BlockDocument)\n    block_type_id: UUID = FieldFrom(objects.BlockDocument)\n    is_anonymous: bool = FieldFrom(objects.BlockDocument)\n\n    _validate_name_format = validator(\"name\", allow_reuse=True)(\n        validate_block_document_name\n    )\n\n    @root_validator\n    def validate_name_is_present_if_not_anonymous(cls, values):\n        # TODO: We should find an elegant way to reuse this logic from the origin model\n        if not values.get(\"is_anonymous\") and not values.get(\"name\"):\n            raise ValueError(\"Names must be provided for block documents.\")\n        return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockDocumentUpdate","title":"BlockDocumentUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a block document.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a block document.\"\"\"\n\n    block_schema_id: Optional[UUID] = Field(\n        default=None, description=\"A block schema ID\"\n    )\n    data: dict = FieldFrom(objects.BlockDocument)\n    merge_existing_data: bool = True\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.BlockDocumentReferenceCreate","title":"BlockDocumentReferenceCreate","text":"

    Bases: ActionBaseModel

    Data used to create block document reference.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentReferenceCreate(ActionBaseModel):\n    \"\"\"Data used to create block document reference.\"\"\"\n\n    id: UUID = FieldFrom(objects.BlockDocumentReference)\n    parent_block_document_id: UUID = FieldFrom(objects.BlockDocumentReference)\n    reference_block_document_id: UUID = FieldFrom(objects.BlockDocumentReference)\n    name: str = FieldFrom(objects.BlockDocumentReference)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.LogCreate","title":"LogCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a log.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass LogCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a log.\"\"\"\n\n    name: str = FieldFrom(objects.Log)\n    level: int = FieldFrom(objects.Log)\n    message: str = FieldFrom(objects.Log)\n    timestamp: objects.DateTimeTZ = FieldFrom(objects.Log)\n    flow_run_id: Optional[UUID] = FieldFrom(objects.Log)\n    task_run_id: Optional[UUID] = FieldFrom(objects.Log)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.WorkPoolCreate","title":"WorkPoolCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a work pool.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass WorkPoolCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a work pool.\"\"\"\n\n    name: str = FieldFrom(objects.WorkPool)\n    description: Optional[str] = FieldFrom(objects.WorkPool)\n    type: str = Field(description=\"The work pool type.\", default=\"prefect-agent\")\n    base_job_template: Dict[str, Any] = FieldFrom(objects.WorkPool)\n    is_paused: bool = FieldFrom(objects.WorkPool)\n    concurrency_limit: Optional[int] = FieldFrom(objects.WorkPool)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.WorkPoolUpdate","title":"WorkPoolUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a work pool.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass WorkPoolUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a work pool.\"\"\"\n\n    description: Optional[str] = FieldFrom(objects.WorkPool)\n    is_paused: Optional[bool] = FieldFrom(objects.WorkPool)\n    base_job_template: Optional[Dict[str, Any]] = FieldFrom(objects.WorkPool)\n    concurrency_limit: Optional[int] = FieldFrom(objects.WorkPool)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.WorkQueueCreate","title":"WorkQueueCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a work queue.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass WorkQueueCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a work queue.\"\"\"\n\n    name: str = FieldFrom(objects.WorkQueue)\n    description: Optional[str] = FieldFrom(objects.WorkQueue)\n    is_paused: bool = FieldFrom(objects.WorkQueue)\n    concurrency_limit: Optional[int] = FieldFrom(objects.WorkQueue)\n    priority: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The queue's priority. Lower values are higher priority (1 is the highest).\"\n        ),\n    )\n\n    # DEPRECATED\n\n    filter: Optional[objects.QueueFilter] = Field(\n        None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.WorkQueueUpdate","title":"WorkQueueUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a work queue.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass WorkQueueUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a work queue.\"\"\"\n\n    name: str = FieldFrom(objects.WorkQueue)\n    description: Optional[str] = FieldFrom(objects.WorkQueue)\n    is_paused: bool = FieldFrom(objects.WorkQueue)\n    concurrency_limit: Optional[int] = FieldFrom(objects.WorkQueue)\n    priority: Optional[int] = FieldFrom(objects.WorkQueue)\n    last_polled: Optional[DateTimeTZ] = FieldFrom(objects.WorkQueue)\n\n    # DEPRECATED\n\n    filter: Optional[objects.QueueFilter] = Field(\n        None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowRunNotificationPolicyCreate","title":"FlowRunNotificationPolicyCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run notification policy.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowRunNotificationPolicyCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run notification policy.\"\"\"\n\n    is_active: bool = FieldFrom(objects.FlowRunNotificationPolicy)\n    state_names: List[str] = FieldFrom(objects.FlowRunNotificationPolicy)\n    tags: List[str] = FieldFrom(objects.FlowRunNotificationPolicy)\n    block_document_id: UUID = FieldFrom(objects.FlowRunNotificationPolicy)\n    message_template: Optional[str] = FieldFrom(objects.FlowRunNotificationPolicy)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.FlowRunNotificationPolicyUpdate","title":"FlowRunNotificationPolicyUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow run notification policy.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass FlowRunNotificationPolicyUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow run notification policy.\"\"\"\n\n    is_active: Optional[bool] = FieldFrom(objects.FlowRunNotificationPolicy)\n    state_names: Optional[List[str]] = FieldFrom(objects.FlowRunNotificationPolicy)\n    tags: Optional[List[str]] = FieldFrom(objects.FlowRunNotificationPolicy)\n    block_document_id: Optional[UUID] = FieldFrom(objects.FlowRunNotificationPolicy)\n    message_template: Optional[str] = FieldFrom(objects.FlowRunNotificationPolicy)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.ArtifactCreate","title":"ArtifactCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create an artifact.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass ArtifactCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create an artifact.\"\"\"\n\n    key: Optional[str] = FieldFrom(objects.Artifact)\n    type: Optional[str] = FieldFrom(objects.Artifact)\n    description: Optional[str] = FieldFrom(objects.Artifact)\n    data: Optional[Union[Dict[str, Any], Any]] = FieldFrom(objects.Artifact)\n    metadata_: Optional[Dict[str, str]] = FieldFrom(objects.Artifact)\n    flow_run_id: Optional[UUID] = FieldFrom(objects.Artifact)\n    task_run_id: Optional[UUID] = FieldFrom(objects.Artifact)\n\n    _validate_artifact_format = validator(\"key\", allow_reuse=True)(\n        validate_artifact_key\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.ArtifactUpdate","title":"ArtifactUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update an artifact.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass ArtifactUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update an artifact.\"\"\"\n\n    data: Optional[Union[Dict[str, Any], Any]] = FieldFrom(objects.Artifact)\n    description: Optional[str] = FieldFrom(objects.Artifact)\n    metadata_: Optional[Dict[str, str]] = FieldFrom(objects.Artifact)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.VariableCreate","title":"VariableCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a Variable.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass VariableCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a Variable.\"\"\"\n\n    name: str = FieldFrom(objects.Variable)\n    value: str = FieldFrom(objects.Variable)\n    tags: Optional[List[str]] = FieldFrom(objects.Variable)\n\n    # validators\n    _validate_name_format = validator(\"name\", allow_reuse=True)(validate_variable_name)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.VariableUpdate","title":"VariableUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a Variable.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass VariableUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a Variable.\"\"\"\n\n    name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the variable\",\n        example=\"my_variable\",\n        max_length=objects.MAX_VARIABLE_NAME_LENGTH,\n    )\n    value: Optional[str] = Field(\n        default=None,\n        description=\"The value of the variable\",\n        example=\"my-value\",\n        max_length=objects.MAX_VARIABLE_NAME_LENGTH,\n    )\n    tags: Optional[List[str]] = FieldFrom(objects.Variable)\n\n    # validators\n    _validate_name_format = validator(\"name\", allow_reuse=True)(validate_variable_name)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.GlobalConcurrencyLimitCreate","title":"GlobalConcurrencyLimitCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a global concurrency limit.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass GlobalConcurrencyLimitCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a global concurrency limit.\"\"\"\n\n    name: str = FieldFrom(objects.GlobalConcurrencyLimit)\n    limit: int = FieldFrom(objects.GlobalConcurrencyLimit)\n    active: Optional[bool] = FieldFrom(objects.GlobalConcurrencyLimit)\n    active_slots: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)\n    slot_decay_per_second: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.actions.GlobalConcurrencyLimitUpdate","title":"GlobalConcurrencyLimitUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a global concurrency limit.

    Source code in prefect/client/schemas/actions.py
    @copy_model_fields\nclass GlobalConcurrencyLimitUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a global concurrency limit.\"\"\"\n\n    name: Optional[str] = FieldFrom(objects.GlobalConcurrencyLimit)\n    limit: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)\n    active: Optional[bool] = FieldFrom(objects.GlobalConcurrencyLimit)\n    active_slots: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)\n    slot_decay_per_second: Optional[int] = FieldFrom(objects.GlobalConcurrencyLimit)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_2","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters","title":"prefect.client.schemas.filters","text":"

    Schemas that define Prefect REST API filtering operations.

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.Operator","title":"Operator","text":"

    Bases: AutoEnum

    Operators for combining filter criteria.

    Source code in prefect/client/schemas/filters.py
    class Operator(AutoEnum):\n    \"\"\"Operators for combining filter criteria.\"\"\"\n\n    and_ = AutoEnum.auto()\n    or_ = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.OperatorMixin","title":"OperatorMixin","text":"

    Base model for Prefect filters that combines criteria with a user-provided operator

    Source code in prefect/client/schemas/filters.py
    class OperatorMixin:\n    \"\"\"Base model for Prefect filters that combines criteria with a user-provided operator\"\"\"\n\n    operator: Operator = Field(\n        default=Operator.and_,\n        description=\"Operator for combining filter criteria. Defaults to 'and_'.\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowFilterId","title":"FlowFilterId","text":"

    Bases: PrefectBaseModel

    Filter by Flow.id.

    Source code in prefect/client/schemas/filters.py
    class FlowFilterId(PrefectBaseModel):\n    \"\"\"Filter by `Flow.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowFilterName","title":"FlowFilterName","text":"

    Bases: PrefectBaseModel

    Filter by Flow.name.

    Source code in prefect/client/schemas/filters.py
    class FlowFilterName(PrefectBaseModel):\n    \"\"\"Filter by `Flow.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of flow names to include\",\n        example=[\"my-flow-1\", \"my-flow-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowFilterTags","title":"FlowFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by Flow.tags.

    Source code in prefect/client/schemas/filters.py
    class FlowFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `Flow.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Flows will be returned only if their tags are a superset\"\n            \" of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include flows without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowFilter","title":"FlowFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter for flows. Only flows matching all criteria will be returned.

    Source code in prefect/client/schemas/filters.py
    class FlowFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter for flows. Only flows matching all criteria will be returned.\"\"\"\n\n    id: Optional[FlowFilterId] = Field(\n        default=None, description=\"Filter criteria for `Flow.id`\"\n    )\n    name: Optional[FlowFilterName] = Field(\n        default=None, description=\"Filter criteria for `Flow.name`\"\n    )\n    tags: Optional[FlowFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Flow.tags`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterId","title":"FlowRunFilterId","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.id.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterId(PrefectBaseModel):\n    \"\"\"Filter by FlowRun.id.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run ids to include\"\n    )\n    not_any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run ids to exclude\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterName","title":"FlowRunFilterName","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.name.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterName(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of flow run names to include\",\n        example=[\"my-flow-run-1\", \"my-flow-run-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterTags","title":"FlowRunFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by FlowRun.tags.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `FlowRun.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Flow runs will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include flow runs without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterDeploymentId","title":"FlowRunFilterDeploymentId","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by FlowRun.deployment_id.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterDeploymentId(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `FlowRun.deployment_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run deployment ids to include\"\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without deployment ids\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterWorkQueueName","title":"FlowRunFilterWorkQueueName","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by FlowRun.work_queue_name.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterWorkQueueName(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `FlowRun.work_queue_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"work_queue_1\", \"work_queue_2\"],\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without work queue names\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterStateType","title":"FlowRunFilterStateType","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.state_type.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterStateType(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.state_type`.\"\"\"\n\n    any_: Optional[List[StateType]] = Field(\n        default=None, description=\"A list of flow run state types to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterFlowVersion","title":"FlowRunFilterFlowVersion","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.flow_version.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterFlowVersion(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.flow_version`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run flow_versions to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterStartTime","title":"FlowRunFilterStartTime","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.start_time.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterStartTime(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs starting at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs starting at or after this time\",\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only return flow runs without a start time\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterExpectedStartTime","title":"FlowRunFilterExpectedStartTime","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.expected_start_time.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterExpectedStartTime(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.expected_start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs scheduled to start at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs scheduled to start at or after this time\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterNextScheduledStartTime","title":"FlowRunFilterNextScheduledStartTime","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.next_scheduled_start_time.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterNextScheduledStartTime(PrefectBaseModel):\n    \"\"\"Filter by `FlowRun.next_scheduled_start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include flow runs with a next_scheduled_start_time or before this\"\n            \" time\"\n        ),\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include flow runs with a next_scheduled_start_time at or after this\"\n            \" time\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterParentFlowRunId","title":"FlowRunFilterParentFlowRunId","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter for subflows of the given flow runs

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterParentFlowRunId(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter for subflows of the given flow runs\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run parents to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterParentTaskRunId","title":"FlowRunFilterParentTaskRunId","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by FlowRun.parent_task_run_id.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterParentTaskRunId(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `FlowRun.parent_task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run parent_task_run_ids to include\"\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without parent_task_run_id\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilterIdempotencyKey","title":"FlowRunFilterIdempotencyKey","text":"

    Bases: PrefectBaseModel

    Filter by FlowRun.idempotency_key.

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilterIdempotencyKey(PrefectBaseModel):\n    \"\"\"Filter by FlowRun.idempotency_key.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run idempotency keys to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run idempotency keys to exclude\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunFilter","title":"FlowRunFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter flow runs. Only flow runs matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class FlowRunFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter flow runs. Only flow runs matching all criteria will be returned\"\"\"\n\n    id: Optional[FlowRunFilterId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.id`\"\n    )\n    name: Optional[FlowRunFilterName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.name`\"\n    )\n    tags: Optional[FlowRunFilterTags] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.tags`\"\n    )\n    deployment_id: Optional[FlowRunFilterDeploymentId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.deployment_id`\"\n    )\n    work_queue_name: Optional[FlowRunFilterWorkQueueName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.work_queue_name\"\n    )\n    state: Optional[FlowRunFilterState] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.state`\"\n    )\n    flow_version: Optional[FlowRunFilterFlowVersion] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.flow_version`\"\n    )\n    start_time: Optional[FlowRunFilterStartTime] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.start_time`\"\n    )\n    expected_start_time: Optional[FlowRunFilterExpectedStartTime] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.expected_start_time`\"\n    )\n    next_scheduled_start_time: Optional[FlowRunFilterNextScheduledStartTime] = Field(\n        default=None,\n        description=\"Filter criteria for `FlowRun.next_scheduled_start_time`\",\n    )\n    parent_flow_run_id: Optional[FlowRunFilterParentFlowRunId] = Field(\n        default=None, description=\"Filter criteria for subflows of the given flow runs\"\n    )\n    parent_task_run_id: Optional[FlowRunFilterParentTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.parent_task_run_id`\"\n    )\n    idempotency_key: Optional[FlowRunFilterIdempotencyKey] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.idempotency_key`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterId","title":"TaskRunFilterId","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.id.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterId(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterName","title":"TaskRunFilterName","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.name.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterName(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of task run names to include\",\n        example=[\"my-task-run-1\", \"my-task-run-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterTags","title":"TaskRunFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by TaskRun.tags.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `TaskRun.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Task runs will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include task runs without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterStateType","title":"TaskRunFilterStateType","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.state_type.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterStateType(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.state_type`.\"\"\"\n\n    any_: Optional[List[StateType]] = Field(\n        default=None, description=\"A list of task run state types to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterSubFlowRuns","title":"TaskRunFilterSubFlowRuns","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.subflow_run.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterSubFlowRuns(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.subflow_run`.\"\"\"\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If true, only include task runs that are subflow run parents; if false,\"\n            \" exclude parent task runs\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilterStartTime","title":"TaskRunFilterStartTime","text":"

    Bases: PrefectBaseModel

    Filter by TaskRun.start_time.

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilterStartTime(PrefectBaseModel):\n    \"\"\"Filter by `TaskRun.start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include task runs starting at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include task runs starting at or after this time\",\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only return task runs without a start time\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.TaskRunFilter","title":"TaskRunFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter task runs. Only task runs matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class TaskRunFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter task runs. Only task runs matching all criteria will be returned\"\"\"\n\n    id: Optional[TaskRunFilterId] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.id`\"\n    )\n    name: Optional[TaskRunFilterName] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.name`\"\n    )\n    tags: Optional[TaskRunFilterTags] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.tags`\"\n    )\n    state: Optional[TaskRunFilterState] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.state`\"\n    )\n    start_time: Optional[TaskRunFilterStartTime] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.start_time`\"\n    )\n    subflow_runs: Optional[TaskRunFilterSubFlowRuns] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.subflow_run`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterId","title":"DeploymentFilterId","text":"

    Bases: PrefectBaseModel

    Filter by Deployment.id.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterId(PrefectBaseModel):\n    \"\"\"Filter by `Deployment.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of deployment ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterName","title":"DeploymentFilterName","text":"

    Bases: PrefectBaseModel

    Filter by Deployment.name.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterName(PrefectBaseModel):\n    \"\"\"Filter by `Deployment.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of deployment names to include\",\n        example=[\"my-deployment-1\", \"my-deployment-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterWorkQueueName","title":"DeploymentFilterWorkQueueName","text":"

    Bases: PrefectBaseModel

    Filter by Deployment.work_queue_name.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterWorkQueueName(PrefectBaseModel):\n    \"\"\"Filter by `Deployment.work_queue_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"work_queue_1\", \"work_queue_2\"],\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterIsScheduleActive","title":"DeploymentFilterIsScheduleActive","text":"

    Bases: PrefectBaseModel

    Filter by Deployment.is_schedule_active.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterIsScheduleActive(PrefectBaseModel):\n    \"\"\"Filter by `Deployment.is_schedule_active`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=\"Only returns where deployment schedule is/is not active\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilterTags","title":"DeploymentFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by Deployment.tags.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `Deployment.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Deployments will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include deployments without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.DeploymentFilter","title":"DeploymentFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter for deployments. Only deployments matching all criteria will be returned.

    Source code in prefect/client/schemas/filters.py
    class DeploymentFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter for deployments. Only deployments matching all criteria will be returned.\"\"\"\n\n    id: Optional[DeploymentFilterId] = Field(\n        default=None, description=\"Filter criteria for `Deployment.id`\"\n    )\n    name: Optional[DeploymentFilterName] = Field(\n        default=None, description=\"Filter criteria for `Deployment.name`\"\n    )\n    is_schedule_active: Optional[DeploymentFilterIsScheduleActive] = Field(\n        default=None, description=\"Filter criteria for `Deployment.is_schedule_active`\"\n    )\n    tags: Optional[DeploymentFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Deployment.tags`\"\n    )\n    work_queue_name: Optional[DeploymentFilterWorkQueueName] = Field(\n        default=None, description=\"Filter criteria for `Deployment.work_queue_name`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterName","title":"LogFilterName","text":"

    Bases: PrefectBaseModel

    Filter by Log.name.

    Source code in prefect/client/schemas/filters.py
    class LogFilterName(PrefectBaseModel):\n    \"\"\"Filter by `Log.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of log names to include\",\n        example=[\"prefect.logger.flow_runs\", \"prefect.logger.task_runs\"],\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterLevel","title":"LogFilterLevel","text":"

    Bases: PrefectBaseModel

    Filter by Log.level.

    Source code in prefect/client/schemas/filters.py
    class LogFilterLevel(PrefectBaseModel):\n    \"\"\"Filter by `Log.level`.\"\"\"\n\n    ge_: Optional[int] = Field(\n        default=None,\n        description=\"Include logs with a level greater than or equal to this level\",\n        example=20,\n    )\n\n    le_: Optional[int] = Field(\n        default=None,\n        description=\"Include logs with a level less than or equal to this level\",\n        example=50,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterTimestamp","title":"LogFilterTimestamp","text":"

    Bases: PrefectBaseModel

    Filter by Log.timestamp.

    Source code in prefect/client/schemas/filters.py
    class LogFilterTimestamp(PrefectBaseModel):\n    \"\"\"Filter by `Log.timestamp`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include logs with a timestamp at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include logs with a timestamp at or after this time\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterFlowRunId","title":"LogFilterFlowRunId","text":"

    Bases: PrefectBaseModel

    Filter by Log.flow_run_id.

    Source code in prefect/client/schemas/filters.py
    class LogFilterFlowRunId(PrefectBaseModel):\n    \"\"\"Filter by `Log.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilterTaskRunId","title":"LogFilterTaskRunId","text":"

    Bases: PrefectBaseModel

    Filter by Log.task_run_id.

    Source code in prefect/client/schemas/filters.py
    class LogFilterTaskRunId(PrefectBaseModel):\n    \"\"\"Filter by `Log.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.LogFilter","title":"LogFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter logs. Only logs matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class LogFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter logs. Only logs matching all criteria will be returned\"\"\"\n\n    level: Optional[LogFilterLevel] = Field(\n        default=None, description=\"Filter criteria for `Log.level`\"\n    )\n    timestamp: Optional[LogFilterTimestamp] = Field(\n        default=None, description=\"Filter criteria for `Log.timestamp`\"\n    )\n    flow_run_id: Optional[LogFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Log.flow_run_id`\"\n    )\n    task_run_id: Optional[LogFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Log.task_run_id`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FilterSet","title":"FilterSet","text":"

    Bases: PrefectBaseModel

    A collection of filters for common objects

    Source code in prefect/client/schemas/filters.py
    class FilterSet(PrefectBaseModel):\n    \"\"\"A collection of filters for common objects\"\"\"\n\n    flows: FlowFilter = Field(\n        default_factory=FlowFilter, description=\"Filters that apply to flows\"\n    )\n    flow_runs: FlowRunFilter = Field(\n        default_factory=FlowRunFilter, description=\"Filters that apply to flow runs\"\n    )\n    task_runs: TaskRunFilter = Field(\n        default_factory=TaskRunFilter, description=\"Filters that apply to task runs\"\n    )\n    deployments: DeploymentFilter = Field(\n        default_factory=DeploymentFilter,\n        description=\"Filters that apply to deployments\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockTypeFilterName","title":"BlockTypeFilterName","text":"

    Bases: PrefectBaseModel

    Filter by BlockType.name

    Source code in prefect/client/schemas/filters.py
    class BlockTypeFilterName(PrefectBaseModel):\n    \"\"\"Filter by `BlockType.name`\"\"\"\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockTypeFilterSlug","title":"BlockTypeFilterSlug","text":"

    Bases: PrefectBaseModel

    Filter by BlockType.slug

    Source code in prefect/client/schemas/filters.py
    class BlockTypeFilterSlug(PrefectBaseModel):\n    \"\"\"Filter by `BlockType.slug`\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of slugs to match\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockTypeFilter","title":"BlockTypeFilter","text":"

    Bases: PrefectBaseModel

    Filter BlockTypes

    Source code in prefect/client/schemas/filters.py
    class BlockTypeFilter(PrefectBaseModel):\n    \"\"\"Filter BlockTypes\"\"\"\n\n    name: Optional[BlockTypeFilterName] = Field(\n        default=None, description=\"Filter criteria for `BlockType.name`\"\n    )\n\n    slug: Optional[BlockTypeFilterSlug] = Field(\n        default=None, description=\"Filter criteria for `BlockType.slug`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilterBlockTypeId","title":"BlockSchemaFilterBlockTypeId","text":"

    Bases: PrefectBaseModel

    Filter by BlockSchema.block_type_id.

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilterBlockTypeId(PrefectBaseModel):\n    \"\"\"Filter by `BlockSchema.block_type_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block type ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilterId","title":"BlockSchemaFilterId","text":"

    Bases: PrefectBaseModel

    Filter by BlockSchema.id

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilterId(PrefectBaseModel):\n    \"\"\"Filter by BlockSchema.id\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilterCapabilities","title":"BlockSchemaFilterCapabilities","text":"

    Bases: PrefectBaseModel

    Filter by BlockSchema.capabilities

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilterCapabilities(PrefectBaseModel):\n    \"\"\"Filter by `BlockSchema.capabilities`\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"write-storage\", \"read-storage\"],\n        description=(\n            \"A list of block capabilities. Block entities will be returned only if an\"\n            \" associated block schema has a superset of the defined capabilities.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilterVersion","title":"BlockSchemaFilterVersion","text":"

    Bases: PrefectBaseModel

    Filter by BlockSchema.capabilities

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilterVersion(PrefectBaseModel):\n    \"\"\"Filter by `BlockSchema.capabilities`\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"2.0.0\", \"2.1.0\"],\n        description=\"A list of block schema versions.\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockSchemaFilter","title":"BlockSchemaFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter BlockSchemas

    Source code in prefect/client/schemas/filters.py
    class BlockSchemaFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter BlockSchemas\"\"\"\n\n    block_type_id: Optional[BlockSchemaFilterBlockTypeId] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.block_type_id`\"\n    )\n    block_capabilities: Optional[BlockSchemaFilterCapabilities] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.capabilities`\"\n    )\n    id: Optional[BlockSchemaFilterId] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.id`\"\n    )\n    version: Optional[BlockSchemaFilterVersion] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.version`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilterIsAnonymous","title":"BlockDocumentFilterIsAnonymous","text":"

    Bases: PrefectBaseModel

    Filter by BlockDocument.is_anonymous.

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilterIsAnonymous(PrefectBaseModel):\n    \"\"\"Filter by `BlockDocument.is_anonymous`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Filter block documents for only those that are or are not anonymous.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilterBlockTypeId","title":"BlockDocumentFilterBlockTypeId","text":"

    Bases: PrefectBaseModel

    Filter by BlockDocument.block_type_id.

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilterBlockTypeId(PrefectBaseModel):\n    \"\"\"Filter by `BlockDocument.block_type_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block type ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilterId","title":"BlockDocumentFilterId","text":"

    Bases: PrefectBaseModel

    Filter by BlockDocument.id.

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilterId(PrefectBaseModel):\n    \"\"\"Filter by `BlockDocument.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilterName","title":"BlockDocumentFilterName","text":"

    Bases: PrefectBaseModel

    Filter by BlockDocument.name.

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilterName(PrefectBaseModel):\n    \"\"\"Filter by `BlockDocument.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of block names to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match block names against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-block%\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.BlockDocumentFilter","title":"BlockDocumentFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter BlockDocuments. Only BlockDocuments matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class BlockDocumentFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter BlockDocuments. Only BlockDocuments matching all criteria will be returned\"\"\"\n\n    id: Optional[BlockDocumentFilterId] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.id`\"\n    )\n    is_anonymous: Optional[BlockDocumentFilterIsAnonymous] = Field(\n        # default is to exclude anonymous blocks\n        BlockDocumentFilterIsAnonymous(eq_=False),\n        description=(\n            \"Filter criteria for `BlockDocument.is_anonymous`. \"\n            \"Defaults to excluding anonymous blocks.\"\n        ),\n    )\n    block_type_id: Optional[BlockDocumentFilterBlockTypeId] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.block_type_id`\"\n    )\n    name: Optional[BlockDocumentFilterName] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.name`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunNotificationPolicyFilterIsActive","title":"FlowRunNotificationPolicyFilterIsActive","text":"

    Bases: PrefectBaseModel

    Filter by FlowRunNotificationPolicy.is_active.

    Source code in prefect/client/schemas/filters.py
    class FlowRunNotificationPolicyFilterIsActive(PrefectBaseModel):\n    \"\"\"Filter by `FlowRunNotificationPolicy.is_active`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Filter notification policies for only those that are or are not active.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.FlowRunNotificationPolicyFilter","title":"FlowRunNotificationPolicyFilter","text":"

    Bases: PrefectBaseModel

    Filter FlowRunNotificationPolicies.

    Source code in prefect/client/schemas/filters.py
    class FlowRunNotificationPolicyFilter(PrefectBaseModel):\n    \"\"\"Filter FlowRunNotificationPolicies.\"\"\"\n\n    is_active: Optional[FlowRunNotificationPolicyFilterIsActive] = Field(\n        default=FlowRunNotificationPolicyFilterIsActive(eq_=False),\n        description=\"Filter criteria for `FlowRunNotificationPolicy.is_active`. \",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkQueueFilterId","title":"WorkQueueFilterId","text":"

    Bases: PrefectBaseModel

    Filter by WorkQueue.id.

    Source code in prefect/client/schemas/filters.py
    class WorkQueueFilterId(PrefectBaseModel):\n    \"\"\"Filter by `WorkQueue.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None,\n        description=\"A list of work queue ids to include\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkQueueFilterName","title":"WorkQueueFilterName","text":"

    Bases: PrefectBaseModel

    Filter by WorkQueue.name.

    Source code in prefect/client/schemas/filters.py
    class WorkQueueFilterName(PrefectBaseModel):\n    \"\"\"Filter by `WorkQueue.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"wq-1\", \"wq-2\"],\n    )\n\n    startswith_: Optional[List[str]] = Field(\n        default=None,\n        description=(\n            \"A list of case-insensitive starts-with matches. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', and 'Marvin-robot', but not 'sad-marvin'.\"\n        ),\n        example=[\"marvin\", \"Marvin-robot\"],\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkQueueFilter","title":"WorkQueueFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter work queues. Only work queues matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class WorkQueueFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter work queues. Only work queues matching all criteria will be\n    returned\"\"\"\n\n    id: Optional[WorkQueueFilterId] = Field(\n        default=None, description=\"Filter criteria for `WorkQueue.id`\"\n    )\n\n    name: Optional[WorkQueueFilterName] = Field(\n        default=None, description=\"Filter criteria for `WorkQueue.name`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkPoolFilterId","title":"WorkPoolFilterId","text":"

    Bases: PrefectBaseModel

    Filter by WorkPool.id.

    Source code in prefect/client/schemas/filters.py
    class WorkPoolFilterId(PrefectBaseModel):\n    \"\"\"Filter by `WorkPool.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of work pool ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkPoolFilterName","title":"WorkPoolFilterName","text":"

    Bases: PrefectBaseModel

    Filter by WorkPool.name.

    Source code in prefect/client/schemas/filters.py
    class WorkPoolFilterName(PrefectBaseModel):\n    \"\"\"Filter by `WorkPool.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of work pool names to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkPoolFilterType","title":"WorkPoolFilterType","text":"

    Bases: PrefectBaseModel

    Filter by WorkPool.type.

    Source code in prefect/client/schemas/filters.py
    class WorkPoolFilterType(PrefectBaseModel):\n    \"\"\"Filter by `WorkPool.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of work pool types to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkerFilterWorkPoolId","title":"WorkerFilterWorkPoolId","text":"

    Bases: PrefectBaseModel

    Filter by Worker.worker_config_id.

    Source code in prefect/client/schemas/filters.py
    class WorkerFilterWorkPoolId(PrefectBaseModel):\n    \"\"\"Filter by `Worker.worker_config_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of work pool ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.WorkerFilterLastHeartbeatTime","title":"WorkerFilterLastHeartbeatTime","text":"

    Bases: PrefectBaseModel

    Filter by Worker.last_heartbeat_time.

    Source code in prefect/client/schemas/filters.py
    class WorkerFilterLastHeartbeatTime(PrefectBaseModel):\n    \"\"\"Filter by `Worker.last_heartbeat_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include processes whose last heartbeat was at or before this time\"\n        ),\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include processes whose last heartbeat was at or after this time\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterId","title":"ArtifactFilterId","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterId(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of artifact ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterKey","title":"ArtifactFilterKey","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.key.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterKey(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.key`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact keys to include\"\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match artifact keys against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-artifact-%\",\n    )\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If `true`, only include artifacts with a non-null key. If `false`, \"\n            \"only include artifacts with a null key.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterFlowRunId","title":"ArtifactFilterFlowRunId","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.flow_run_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterFlowRunId(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterTaskRunId","title":"ArtifactFilterTaskRunId","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.task_run_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterTaskRunId(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilterType","title":"ArtifactFilterType","text":"

    Bases: PrefectBaseModel

    Filter by Artifact.type.

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilterType(PrefectBaseModel):\n    \"\"\"Filter by `Artifact.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to exclude\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactFilter","title":"ArtifactFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter artifacts. Only artifacts matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class ArtifactFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter artifacts. Only artifacts matching all criteria will be returned\"\"\"\n\n    id: Optional[ArtifactFilterId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.id`\"\n    )\n    key: Optional[ArtifactFilterKey] = Field(\n        default=None, description=\"Filter criteria for `Artifact.key`\"\n    )\n    flow_run_id: Optional[ArtifactFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.flow_run_id`\"\n    )\n    task_run_id: Optional[ArtifactFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.task_run_id`\"\n    )\n    type: Optional[ArtifactFilterType] = Field(\n        default=None, description=\"Filter criteria for `Artifact.type`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterLatestId","title":"ArtifactCollectionFilterLatestId","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.latest_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterLatestId(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.latest_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of artifact ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterKey","title":"ArtifactCollectionFilterKey","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.key.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterKey(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.key`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact keys to include\"\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match artifact keys against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-artifact-%\",\n    )\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If `true`, only include artifacts with a non-null key. If `false`, \"\n            \"only include artifacts with a null key. Should return all rows in \"\n            \"the ArtifactCollection table if specified.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterFlowRunId","title":"ArtifactCollectionFilterFlowRunId","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.flow_run_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterFlowRunId(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterTaskRunId","title":"ArtifactCollectionFilterTaskRunId","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.task_run_id.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterTaskRunId(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilterType","title":"ArtifactCollectionFilterType","text":"

    Bases: PrefectBaseModel

    Filter by ArtifactCollection.type.

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilterType(PrefectBaseModel):\n    \"\"\"Filter by `ArtifactCollection.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to exclude\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.ArtifactCollectionFilter","title":"ArtifactCollectionFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter artifact collections. Only artifact collections matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class ArtifactCollectionFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter artifact collections. Only artifact collections matching all criteria will be returned\"\"\"\n\n    latest_id: Optional[ArtifactCollectionFilterLatestId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.id`\"\n    )\n    key: Optional[ArtifactCollectionFilterKey] = Field(\n        default=None, description=\"Filter criteria for `Artifact.key`\"\n    )\n    flow_run_id: Optional[ArtifactCollectionFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.flow_run_id`\"\n    )\n    task_run_id: Optional[ArtifactCollectionFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.task_run_id`\"\n    )\n    type: Optional[ArtifactCollectionFilterType] = Field(\n        default=None, description=\"Filter criteria for `Artifact.type`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilterId","title":"VariableFilterId","text":"

    Bases: PrefectBaseModel

    Filter by Variable.id.

    Source code in prefect/client/schemas/filters.py
    class VariableFilterId(PrefectBaseModel):\n    \"\"\"Filter by `Variable.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of variable ids to include\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilterName","title":"VariableFilterName","text":"

    Bases: PrefectBaseModel

    Filter by Variable.name.

    Source code in prefect/client/schemas/filters.py
    class VariableFilterName(PrefectBaseModel):\n    \"\"\"Filter by `Variable.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of variables names to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match variable names against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my_variable_%\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilterValue","title":"VariableFilterValue","text":"

    Bases: PrefectBaseModel

    Filter by Variable.value.

    Source code in prefect/client/schemas/filters.py
    class VariableFilterValue(PrefectBaseModel):\n    \"\"\"Filter by `Variable.value`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of variables value to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match variable value against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-value-%\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilterTags","title":"VariableFilterTags","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter by Variable.tags.

    Source code in prefect/client/schemas/filters.py
    class VariableFilterTags(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter by `Variable.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Variables will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include Variables without tags\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.filters.VariableFilter","title":"VariableFilter","text":"

    Bases: PrefectBaseModel, OperatorMixin

    Filter variables. Only variables matching all criteria will be returned

    Source code in prefect/client/schemas/filters.py
    class VariableFilter(PrefectBaseModel, OperatorMixin):\n    \"\"\"Filter variables. Only variables matching all criteria will be returned\"\"\"\n\n    id: Optional[VariableFilterId] = Field(\n        default=None, description=\"Filter criteria for `Variable.id`\"\n    )\n    name: Optional[VariableFilterName] = Field(\n        default=None, description=\"Filter criteria for `Variable.name`\"\n    )\n    value: Optional[VariableFilterValue] = Field(\n        default=None, description=\"Filter criteria for `Variable.value`\"\n    )\n    tags: Optional[VariableFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Variable.tags`\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_3","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects","title":"prefect.client.schemas.objects","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.StateType","title":"StateType","text":"

    Bases: AutoEnum

    Enumeration of state types.

    Source code in prefect/client/schemas/objects.py
    class StateType(AutoEnum):\n    \"\"\"Enumeration of state types.\"\"\"\n\n    SCHEDULED = AutoEnum.auto()\n    PENDING = AutoEnum.auto()\n    RUNNING = AutoEnum.auto()\n    COMPLETED = AutoEnum.auto()\n    FAILED = AutoEnum.auto()\n    CANCELLED = AutoEnum.auto()\n    CRASHED = AutoEnum.auto()\n    PAUSED = AutoEnum.auto()\n    CANCELLING = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkPoolStatus","title":"WorkPoolStatus","text":"

    Bases: AutoEnum

    Enumeration of work pool statuses.

    Source code in prefect/client/schemas/objects.py
    class WorkPoolStatus(AutoEnum):\n    \"\"\"Enumeration of work pool statuses.\"\"\"\n\n    READY = AutoEnum.auto()\n    NOT_READY = AutoEnum.auto()\n    PAUSED = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkerStatus","title":"WorkerStatus","text":"

    Bases: AutoEnum

    Enumeration of worker statuses.

    Source code in prefect/client/schemas/objects.py
    class WorkerStatus(AutoEnum):\n    \"\"\"Enumeration of worker statuses.\"\"\"\n\n    ONLINE = AutoEnum.auto()\n    OFFLINE = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.DeploymentStatus","title":"DeploymentStatus","text":"

    Bases: AutoEnum

    Enumeration of deployment statuses.

    Source code in prefect/client/schemas/objects.py
    class DeploymentStatus(AutoEnum):\n    \"\"\"Enumeration of deployment statuses.\"\"\"\n\n    READY = AutoEnum.auto()\n    NOT_READY = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State","title":"State","text":"

    Bases: ObjectBaseModel, Generic[R]

    The state of a run.

    Source code in prefect/client/schemas/objects.py
    class State(ObjectBaseModel, Generic[R]):\n    \"\"\"\n    The state of a run.\n    \"\"\"\n\n    type: StateType\n    name: Optional[str] = Field(default=None)\n    timestamp: DateTimeTZ = Field(default_factory=lambda: pendulum.now(\"UTC\"))\n    message: Optional[str] = Field(default=None, example=\"Run started\")\n    state_details: StateDetails = Field(default_factory=StateDetails)\n    data: Union[\"BaseResult[R]\", \"DataDocument[R]\", Any] = Field(\n        default=None,\n    )\n\n    @overload\n    def result(self: \"State[R]\", raise_on_failure: bool = True) -> R:\n        ...\n\n    @overload\n    def result(self: \"State[R]\", raise_on_failure: bool = False) -> Union[R, Exception]:\n        ...\n\n    def result(self, raise_on_failure: bool = True, fetch: Optional[bool] = None):\n        \"\"\"\n        Retrieve the result attached to this state.\n\n        Args:\n            raise_on_failure: a boolean specifying whether to raise an exception\n                if the state is of type `FAILED` and the underlying data is an exception\n            fetch: a boolean specifying whether to resolve references to persisted\n                results into data. For synchronous users, this defaults to `True`.\n                For asynchronous users, this defaults to `False` for backwards\n                compatibility.\n\n        Raises:\n            TypeError: If the state is failed but the result is not an exception.\n\n        Returns:\n            The result of the run\n\n        Examples:\n            >>> from prefect import flow, task\n            >>> @task\n            >>> def my_task(x):\n            >>>     return x\n\n            Get the result from a task future in a flow\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     future = my_task(\"hello\")\n            >>>     state = future.wait()\n            >>>     result = state.result()\n            >>>     print(result)\n            >>> my_flow()\n            hello\n\n            Get the result from a flow state\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     return \"hello\"\n            >>> my_flow(return_state=True).result()\n            hello\n\n            Get the result from a failed state\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     raise ValueError(\"oh no!\")\n            >>> state = my_flow(return_state=True)  # Error is wrapped in FAILED state\n            >>> state.result()  # Raises `ValueError`\n\n            Get the result from a failed state without erroring\n\n            >>> @flow\n            >>> def my_flow():\n            >>>     raise ValueError(\"oh no!\")\n            >>> state = my_flow(return_state=True)\n            >>> result = state.result(raise_on_failure=False)\n            >>> print(result)\n            ValueError(\"oh no!\")\n\n\n            Get the result from a flow state in an async context\n\n            >>> @flow\n            >>> async def my_flow():\n            >>>     return \"hello\"\n            >>> state = await my_flow(return_state=True)\n            >>> await state.result()\n            hello\n        \"\"\"\n        from prefect.states import get_state_result\n\n        return get_state_result(self, raise_on_failure=raise_on_failure, fetch=fetch)\n\n    def to_state_create(self):\n        \"\"\"\n        Convert this state to a `StateCreate` type which can be used to set the state of\n        a run in the API.\n\n        This method will drop this state's `data` if it is not a result type. Only\n        results should be sent to the API. Other data is only available locally.\n        \"\"\"\n        from prefect.client.schemas.actions import StateCreate\n        from prefect.results import BaseResult\n\n        return StateCreate(\n            type=self.type,\n            name=self.name,\n            message=self.message,\n            data=self.data if isinstance(self.data, BaseResult) else None,\n            state_details=self.state_details,\n        )\n\n    @validator(\"name\", always=True)\n    def default_name_from_type(cls, v, *, values, **kwargs):\n        \"\"\"If a name is not provided, use the type\"\"\"\n\n        # if `type` is not in `values` it means the `type` didn't pass its own\n        # validation check and an error will be raised after this function is called\n        if v is None and values.get(\"type\"):\n            v = \" \".join([v.capitalize() for v in values.get(\"type\").value.split(\"_\")])\n        return v\n\n    @root_validator\n    def default_scheduled_start_time(cls, values):\n        \"\"\"\n        TODO: This should throw an error instead of setting a default but is out of\n              scope for https://github.com/PrefectHQ/orion/pull/174/ and can be rolled\n              into work refactoring state initialization\n        \"\"\"\n        if values.get(\"type\") == StateType.SCHEDULED:\n            state_details = values.setdefault(\n                \"state_details\", cls.__fields__[\"state_details\"].get_default()\n            )\n            if not state_details.scheduled_time:\n                state_details.scheduled_time = pendulum.now(\"utc\")\n        return values\n\n    def is_scheduled(self) -> bool:\n        return self.type == StateType.SCHEDULED\n\n    def is_pending(self) -> bool:\n        return self.type == StateType.PENDING\n\n    def is_running(self) -> bool:\n        return self.type == StateType.RUNNING\n\n    def is_completed(self) -> bool:\n        return self.type == StateType.COMPLETED\n\n    def is_failed(self) -> bool:\n        return self.type == StateType.FAILED\n\n    def is_crashed(self) -> bool:\n        return self.type == StateType.CRASHED\n\n    def is_cancelled(self) -> bool:\n        return self.type == StateType.CANCELLED\n\n    def is_cancelling(self) -> bool:\n        return self.type == StateType.CANCELLING\n\n    def is_final(self) -> bool:\n        return self.type in {\n            StateType.CANCELLED,\n            StateType.FAILED,\n            StateType.COMPLETED,\n            StateType.CRASHED,\n        }\n\n    def is_paused(self) -> bool:\n        return self.type == StateType.PAUSED\n\n    def copy(self, *, update: dict = None, reset_fields: bool = False, **kwargs):\n        \"\"\"\n        Copying API models should return an object that could be inserted into the\n        database again. The 'timestamp' is reset using the default factory.\n        \"\"\"\n        update = update or {}\n        update.setdefault(\"timestamp\", self.__fields__[\"timestamp\"].get_default())\n        return super().copy(reset_fields=reset_fields, update=update, **kwargs)\n\n    def __repr__(self) -> str:\n        \"\"\"\n        Generates a complete state representation appropriate for introspection\n        and debugging, including the result:\n\n        `MyCompletedState(message=\"my message\", type=COMPLETED, result=...)`\n        \"\"\"\n        from prefect.deprecated.data_documents import DataDocument\n\n        if isinstance(self.data, DataDocument):\n            result = self.data.decode()\n        else:\n            result = self.data\n\n        display = dict(\n            message=repr(self.message),\n            type=str(self.type.value),\n            result=repr(result),\n        )\n\n        return f\"{self.name}({', '.join(f'{k}={v}' for k, v in display.items())})\"\n\n    def __str__(self) -> str:\n        \"\"\"\n        Generates a simple state representation appropriate for logging:\n\n        `MyCompletedState(\"my message\", type=COMPLETED)`\n        \"\"\"\n\n        display = []\n\n        if self.message:\n            display.append(repr(self.message))\n\n        if self.type.value.lower() != self.name.lower():\n            display.append(f\"type={self.type.value}\")\n\n        return f\"{self.name}({', '.join(display)})\"\n\n    def __hash__(self) -> int:\n        return hash(\n            (\n                getattr(self.state_details, \"flow_run_id\", None),\n                getattr(self.state_details, \"task_run_id\", None),\n                self.timestamp,\n                self.type,\n            )\n        )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State.result","title":"result","text":"

    Retrieve the result attached to this state.

    Parameters:

    Name Type Description Default raise_on_failure bool

    a boolean specifying whether to raise an exception if the state is of type FAILED and the underlying data is an exception

    True fetch Optional[bool]

    a boolean specifying whether to resolve references to persisted results into data. For synchronous users, this defaults to True. For asynchronous users, this defaults to False for backwards compatibility.

    None

    Raises:

    Type Description TypeError

    If the state is failed but the result is not an exception.

    Returns:

    Type Description

    The result of the run

    Examples:

    >>> from prefect import flow, task\n>>> @task\n>>> def my_task(x):\n>>>     return x\n

    Get the result from a task future in a flow

    >>> @flow\n>>> def my_flow():\n>>>     future = my_task(\"hello\")\n>>>     state = future.wait()\n>>>     result = state.result()\n>>>     print(result)\n>>> my_flow()\nhello\n

    Get the result from a flow state

    >>> @flow\n>>> def my_flow():\n>>>     return \"hello\"\n>>> my_flow(return_state=True).result()\nhello\n

    Get the result from a failed state

    >>> @flow\n>>> def my_flow():\n>>>     raise ValueError(\"oh no!\")\n>>> state = my_flow(return_state=True)  # Error is wrapped in FAILED state\n>>> state.result()  # Raises `ValueError`\n

    Get the result from a failed state without erroring

    >>> @flow\n>>> def my_flow():\n>>>     raise ValueError(\"oh no!\")\n>>> state = my_flow(return_state=True)\n>>> result = state.result(raise_on_failure=False)\n>>> print(result)\nValueError(\"oh no!\")\n

    Get the result from a flow state in an async context

    >>> @flow\n>>> async def my_flow():\n>>>     return \"hello\"\n>>> state = await my_flow(return_state=True)\n>>> await state.result()\nhello\n
    Source code in prefect/client/schemas/objects.py
    def result(self, raise_on_failure: bool = True, fetch: Optional[bool] = None):\n    \"\"\"\n    Retrieve the result attached to this state.\n\n    Args:\n        raise_on_failure: a boolean specifying whether to raise an exception\n            if the state is of type `FAILED` and the underlying data is an exception\n        fetch: a boolean specifying whether to resolve references to persisted\n            results into data. For synchronous users, this defaults to `True`.\n            For asynchronous users, this defaults to `False` for backwards\n            compatibility.\n\n    Raises:\n        TypeError: If the state is failed but the result is not an exception.\n\n    Returns:\n        The result of the run\n\n    Examples:\n        >>> from prefect import flow, task\n        >>> @task\n        >>> def my_task(x):\n        >>>     return x\n\n        Get the result from a task future in a flow\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     future = my_task(\"hello\")\n        >>>     state = future.wait()\n        >>>     result = state.result()\n        >>>     print(result)\n        >>> my_flow()\n        hello\n\n        Get the result from a flow state\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     return \"hello\"\n        >>> my_flow(return_state=True).result()\n        hello\n\n        Get the result from a failed state\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     raise ValueError(\"oh no!\")\n        >>> state = my_flow(return_state=True)  # Error is wrapped in FAILED state\n        >>> state.result()  # Raises `ValueError`\n\n        Get the result from a failed state without erroring\n\n        >>> @flow\n        >>> def my_flow():\n        >>>     raise ValueError(\"oh no!\")\n        >>> state = my_flow(return_state=True)\n        >>> result = state.result(raise_on_failure=False)\n        >>> print(result)\n        ValueError(\"oh no!\")\n\n\n        Get the result from a flow state in an async context\n\n        >>> @flow\n        >>> async def my_flow():\n        >>>     return \"hello\"\n        >>> state = await my_flow(return_state=True)\n        >>> await state.result()\n        hello\n    \"\"\"\n    from prefect.states import get_state_result\n\n    return get_state_result(self, raise_on_failure=raise_on_failure, fetch=fetch)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State.to_state_create","title":"to_state_create","text":"

    Convert this state to a StateCreate type which can be used to set the state of a run in the API.

    This method will drop this state's data if it is not a result type. Only results should be sent to the API. Other data is only available locally.

    Source code in prefect/client/schemas/objects.py
    def to_state_create(self):\n    \"\"\"\n    Convert this state to a `StateCreate` type which can be used to set the state of\n    a run in the API.\n\n    This method will drop this state's `data` if it is not a result type. Only\n    results should be sent to the API. Other data is only available locally.\n    \"\"\"\n    from prefect.client.schemas.actions import StateCreate\n    from prefect.results import BaseResult\n\n    return StateCreate(\n        type=self.type,\n        name=self.name,\n        message=self.message,\n        data=self.data if isinstance(self.data, BaseResult) else None,\n        state_details=self.state_details,\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State.default_name_from_type","title":"default_name_from_type","text":"

    If a name is not provided, use the type

    Source code in prefect/client/schemas/objects.py
    @validator(\"name\", always=True)\ndef default_name_from_type(cls, v, *, values, **kwargs):\n    \"\"\"If a name is not provided, use the type\"\"\"\n\n    # if `type` is not in `values` it means the `type` didn't pass its own\n    # validation check and an error will be raised after this function is called\n    if v is None and values.get(\"type\"):\n        v = \" \".join([v.capitalize() for v in values.get(\"type\").value.split(\"_\")])\n    return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.State.default_scheduled_start_time","title":"default_scheduled_start_time","text":"This should throw an error instead of setting a default but is out of

    scope for https://github.com/PrefectHQ/orion/pull/174/ and can be rolled into work refactoring state initialization

    Source code in prefect/client/schemas/objects.py
    @root_validator\ndef default_scheduled_start_time(cls, values):\n    \"\"\"\n    TODO: This should throw an error instead of setting a default but is out of\n          scope for https://github.com/PrefectHQ/orion/pull/174/ and can be rolled\n          into work refactoring state initialization\n    \"\"\"\n    if values.get(\"type\") == StateType.SCHEDULED:\n        state_details = values.setdefault(\n            \"state_details\", cls.__fields__[\"state_details\"].get_default()\n        )\n        if not state_details.scheduled_time:\n            state_details.scheduled_time = pendulum.now(\"utc\")\n    return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunPolicy","title":"FlowRunPolicy","text":"

    Bases: PrefectBaseModel

    Defines of how a flow run should be orchestrated.

    Source code in prefect/client/schemas/objects.py
    class FlowRunPolicy(PrefectBaseModel):\n    \"\"\"Defines of how a flow run should be orchestrated.\"\"\"\n\n    max_retries: int = Field(\n        default=0,\n        description=(\n            \"The maximum number of retries. Field is not used. Please use `retries`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retry_delay_seconds: float = Field(\n        default=0,\n        description=(\n            \"The delay between retries. Field is not used. Please use `retry_delay`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retries: Optional[int] = Field(default=None, description=\"The number of retries.\")\n    retry_delay: Optional[int] = Field(\n        default=None, description=\"The delay time between retries, in seconds.\"\n    )\n    pause_keys: Optional[set] = Field(\n        default_factory=set, description=\"Tracks pauses this run has observed.\"\n    )\n    resuming: Optional[bool] = Field(\n        default=False, description=\"Indicates if this run is resuming from a pause.\"\n    )\n\n    @root_validator\n    def populate_deprecated_fields(cls, values):\n        \"\"\"\n        If deprecated fields are provided, populate the corresponding new fields\n        to preserve orchestration behavior.\n        \"\"\"\n        if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n            values[\"retries\"] = values[\"max_retries\"]\n        if (\n            not values.get(\"retry_delay\", None)\n            and values.get(\"retry_delay_seconds\", 0) != 0\n        ):\n            values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n        return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunPolicy.populate_deprecated_fields","title":"populate_deprecated_fields","text":"

    If deprecated fields are provided, populate the corresponding new fields to preserve orchestration behavior.

    Source code in prefect/client/schemas/objects.py
    @root_validator\ndef populate_deprecated_fields(cls, values):\n    \"\"\"\n    If deprecated fields are provided, populate the corresponding new fields\n    to preserve orchestration behavior.\n    \"\"\"\n    if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n        values[\"retries\"] = values[\"max_retries\"]\n    if (\n        not values.get(\"retry_delay\", None)\n        and values.get(\"retry_delay_seconds\", 0) != 0\n    ):\n        values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n    return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRun","title":"FlowRun","text":"

    Bases: ObjectBaseModel

    Source code in prefect/client/schemas/objects.py
    class FlowRun(ObjectBaseModel):\n    name: str = Field(\n        default_factory=lambda: generate_slug(2),\n        description=(\n            \"The name of the flow run. Defaults to a random slug if not specified.\"\n        ),\n        example=\"my-flow-run\",\n    )\n    flow_id: UUID = Field(default=..., description=\"The id of the flow being run.\")\n    state_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the flow run's current state.\"\n    )\n    deployment_id: Optional[UUID] = Field(\n        default=None,\n        description=(\n            \"The id of the deployment associated with this flow run, if available.\"\n        ),\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None, description=\"The work queue that handled this flow run.\"\n    )\n    flow_version: Optional[str] = Field(\n        default=None,\n        description=\"The version of the flow executed in this flow run.\",\n        example=\"1.0\",\n    )\n    parameters: dict = Field(\n        default_factory=dict, description=\"Parameters for the flow run.\"\n    )\n    idempotency_key: Optional[str] = Field(\n        default=None,\n        description=(\n            \"An optional idempotency key for the flow run. Used to ensure the same flow\"\n            \" run is not created multiple times.\"\n        ),\n    )\n    context: dict = Field(\n        default_factory=dict,\n        description=\"Additional context for the flow run.\",\n        example={\"my_var\": \"my_val\"},\n    )\n    empirical_policy: FlowRunPolicy = Field(\n        default_factory=FlowRunPolicy,\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags on the flow run\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    parent_task_run_id: Optional[UUID] = Field(\n        default=None,\n        description=(\n            \"If the flow run is a subflow, the id of the 'dummy' task in the parent\"\n            \" flow used to track subflow state.\"\n        ),\n    )\n    run_count: int = Field(\n        default=0, description=\"The number of times the flow run was executed.\"\n    )\n    expected_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The flow run's expected start time.\",\n    )\n    next_scheduled_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The next time the flow run is scheduled to start.\",\n    )\n    start_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual start time.\"\n    )\n    end_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual end time.\"\n    )\n    total_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=(\n            \"Total run time. If the flow run was executed multiple times, the time of\"\n            \" each run will be summed.\"\n        ),\n    )\n    estimated_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"A real-time estimate of the total run time.\",\n    )\n    estimated_start_time_delta: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"The difference between actual and expected start time.\",\n    )\n    auto_scheduled: bool = Field(\n        default=False,\n        description=\"Whether or not the flow run was automatically scheduled.\",\n    )\n    infrastructure_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining infrastructure to use this flow run.\",\n    )\n    infrastructure_pid: Optional[str] = Field(\n        default=None,\n        description=\"The id of the flow run as returned by an infrastructure block.\",\n    )\n    created_by: Optional[CreatedBy] = Field(\n        default=None,\n        description=\"Optional information about the creator of this flow run.\",\n    )\n    work_queue_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the run's work pool queue.\"\n    )\n\n    work_pool_id: Optional[UUID] = Field(\n        description=\"The work pool with which the queue is associated.\"\n    )\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the flow run's work pool.\",\n        example=\"my-work-pool\",\n    )\n    state: Optional[State] = Field(\n        default=None,\n        description=\"The state of the flow run.\",\n        example=State(type=StateType.COMPLETED),\n    )\n\n    def __eq__(self, other: Any) -> bool:\n        \"\"\"\n        Check for \"equality\" to another flow run schema\n\n        Estimates times are rolling and will always change with repeated queries for\n        a flow run so we ignore them during equality checks.\n        \"\"\"\n        if isinstance(other, FlowRun):\n            exclude_fields = {\"estimated_run_time\", \"estimated_start_time_delta\"}\n            return self.dict(exclude=exclude_fields) == other.dict(\n                exclude=exclude_fields\n            )\n        return super().__eq__(other)\n\n    @validator(\"name\", pre=True)\n    def set_default_name(cls, name):\n        return name or generate_slug(2)\n\n    # These are server-side optimizations and should not be present on client models\n    # TODO: Deprecate these fields\n\n    state_type: Optional[StateType] = Field(\n        default=None, description=\"The type of the current flow run state.\"\n    )\n    state_name: Optional[str] = Field(\n        default=None, description=\"The name of the current flow run state.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.TaskRunPolicy","title":"TaskRunPolicy","text":"

    Bases: PrefectBaseModel

    Defines of how a task run should retry.

    Source code in prefect/client/schemas/objects.py
    class TaskRunPolicy(PrefectBaseModel):\n    \"\"\"Defines of how a task run should retry.\"\"\"\n\n    max_retries: int = Field(\n        default=0,\n        description=(\n            \"The maximum number of retries. Field is not used. Please use `retries`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retry_delay_seconds: float = Field(\n        default=0,\n        description=(\n            \"The delay between retries. Field is not used. Please use `retry_delay`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retries: Optional[int] = Field(default=None, description=\"The number of retries.\")\n    retry_delay: Union[None, int, List[int]] = Field(\n        default=None,\n        description=\"A delay time or list of delay times between retries, in seconds.\",\n    )\n    retry_jitter_factor: Optional[float] = Field(\n        default=None, description=\"Determines the amount a retry should jitter\"\n    )\n\n    @root_validator\n    def populate_deprecated_fields(cls, values):\n        \"\"\"\n        If deprecated fields are provided, populate the corresponding new fields\n        to preserve orchestration behavior.\n        \"\"\"\n        if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n            values[\"retries\"] = values[\"max_retries\"]\n\n        if (\n            not values.get(\"retry_delay\", None)\n            and values.get(\"retry_delay_seconds\", 0) != 0\n        ):\n            values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n\n        return values\n\n    @validator(\"retry_delay\")\n    def validate_configured_retry_delays(cls, v):\n        if isinstance(v, list) and (len(v) > 50):\n            raise ValueError(\"Can not configure more than 50 retry delays per task.\")\n        return v\n\n    @validator(\"retry_jitter_factor\")\n    def validate_jitter_factor(cls, v):\n        if v is not None and v < 0:\n            raise ValueError(\"`retry_jitter_factor` must be >= 0.\")\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.TaskRunPolicy.populate_deprecated_fields","title":"populate_deprecated_fields","text":"

    If deprecated fields are provided, populate the corresponding new fields to preserve orchestration behavior.

    Source code in prefect/client/schemas/objects.py
    @root_validator\ndef populate_deprecated_fields(cls, values):\n    \"\"\"\n    If deprecated fields are provided, populate the corresponding new fields\n    to preserve orchestration behavior.\n    \"\"\"\n    if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n        values[\"retries\"] = values[\"max_retries\"]\n\n    if (\n        not values.get(\"retry_delay\", None)\n        and values.get(\"retry_delay_seconds\", 0) != 0\n    ):\n        values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n\n    return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.TaskRunInput","title":"TaskRunInput","text":"

    Bases: PrefectBaseModel

    Base class for classes that represent inputs to task runs, which could include, constants, parameters, or other task runs.

    Source code in prefect/client/schemas/objects.py
    class TaskRunInput(PrefectBaseModel):\n    \"\"\"\n    Base class for classes that represent inputs to task runs, which\n    could include, constants, parameters, or other task runs.\n    \"\"\"\n\n    # freeze TaskRunInputs to allow them to be placed in sets\n    class Config:\n        frozen = True\n\n    input_type: str\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.TaskRunResult","title":"TaskRunResult","text":"

    Bases: TaskRunInput

    Represents a task run result input to another task run.

    Source code in prefect/client/schemas/objects.py
    class TaskRunResult(TaskRunInput):\n    \"\"\"Represents a task run result input to another task run.\"\"\"\n\n    input_type: Literal[\"task_run\"] = \"task_run\"\n    id: UUID\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Parameter","title":"Parameter","text":"

    Bases: TaskRunInput

    Represents a parameter input to a task run.

    Source code in prefect/client/schemas/objects.py
    class Parameter(TaskRunInput):\n    \"\"\"Represents a parameter input to a task run.\"\"\"\n\n    input_type: Literal[\"parameter\"] = \"parameter\"\n    name: str\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Constant","title":"Constant","text":"

    Bases: TaskRunInput

    Represents constant input value to a task run.

    Source code in prefect/client/schemas/objects.py
    class Constant(TaskRunInput):\n    \"\"\"Represents constant input value to a task run.\"\"\"\n\n    input_type: Literal[\"constant\"] = \"constant\"\n    type: str\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Workspace","title":"Workspace","text":"

    Bases: PrefectBaseModel

    A Prefect Cloud workspace.

    Expected payload for each workspace returned by the me/workspaces route.

    Source code in prefect/client/schemas/objects.py
    class Workspace(PrefectBaseModel):\n    \"\"\"\n    A Prefect Cloud workspace.\n\n    Expected payload for each workspace returned by the `me/workspaces` route.\n    \"\"\"\n\n    account_id: UUID = Field(..., description=\"The account id of the workspace.\")\n    account_name: str = Field(..., description=\"The account name.\")\n    account_handle: str = Field(..., description=\"The account's unique handle.\")\n    workspace_id: UUID = Field(..., description=\"The workspace id.\")\n    workspace_name: str = Field(..., description=\"The workspace name.\")\n    workspace_description: str = Field(..., description=\"Description of the workspace.\")\n    workspace_handle: str = Field(..., description=\"The workspace's unique handle.\")\n\n    class Config:\n        extra = \"ignore\"\n\n    @property\n    def handle(self) -> str:\n        \"\"\"\n        The full handle of the workspace as `account_handle` / `workspace_handle`\n        \"\"\"\n        return self.account_handle + \"/\" + self.workspace_handle\n\n    def api_url(self) -> str:\n        \"\"\"\n        Generate the API URL for accessing this workspace\n        \"\"\"\n        return (\n            f\"{PREFECT_CLOUD_API_URL.value()}\"\n            f\"/accounts/{self.account_id}\"\n            f\"/workspaces/{self.workspace_id}\"\n        )\n\n    def ui_url(self) -> str:\n        \"\"\"\n        Generate the UI URL for accessing this workspace\n        \"\"\"\n        return (\n            f\"{PREFECT_CLOUD_UI_URL.value()}\"\n            f\"/account/{self.account_id}\"\n            f\"/workspace/{self.workspace_id}\"\n        )\n\n    def __hash__(self):\n        return hash(self.handle)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Workspace.handle","title":"handle: str property","text":"

    The full handle of the workspace as account_handle / workspace_handle

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Workspace.api_url","title":"api_url","text":"

    Generate the API URL for accessing this workspace

    Source code in prefect/client/schemas/objects.py
    def api_url(self) -> str:\n    \"\"\"\n    Generate the API URL for accessing this workspace\n    \"\"\"\n    return (\n        f\"{PREFECT_CLOUD_API_URL.value()}\"\n        f\"/accounts/{self.account_id}\"\n        f\"/workspaces/{self.workspace_id}\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Workspace.ui_url","title":"ui_url","text":"

    Generate the UI URL for accessing this workspace

    Source code in prefect/client/schemas/objects.py
    def ui_url(self) -> str:\n    \"\"\"\n    Generate the UI URL for accessing this workspace\n    \"\"\"\n    return (\n        f\"{PREFECT_CLOUD_UI_URL.value()}\"\n        f\"/account/{self.account_id}\"\n        f\"/workspace/{self.workspace_id}\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockType","title":"BlockType","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block type

    Source code in prefect/client/schemas/objects.py
    class BlockType(ObjectBaseModel):\n    \"\"\"An ORM representation of a block type\"\"\"\n\n    name: str = Field(default=..., description=\"A block type's name\")\n    slug: str = Field(default=..., description=\"A block type's slug\")\n    logo_url: Optional[HttpUrl] = Field(\n        default=None, description=\"Web URL for the block type's logo\"\n    )\n    documentation_url: Optional[HttpUrl] = Field(\n        default=None, description=\"Web URL for the block type's documentation\"\n    )\n    description: Optional[str] = Field(\n        default=None,\n        description=\"A short blurb about the corresponding block's intended use\",\n    )\n    code_example: Optional[str] = Field(\n        default=None,\n        description=\"A code snippet demonstrating use of the corresponding block\",\n    )\n    is_protected: bool = Field(\n        default=False, description=\"Protected block types cannot be modified via API.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockDocument","title":"BlockDocument","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block document.

    Source code in prefect/client/schemas/objects.py
    class BlockDocument(ObjectBaseModel):\n    \"\"\"An ORM representation of a block document.\"\"\"\n\n    name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The block document's name. Not required for anonymous block documents.\"\n        ),\n    )\n    data: dict = Field(default_factory=dict, description=\"The block document's data\")\n    block_schema_id: UUID = Field(default=..., description=\"A block schema ID\")\n    block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The associated block schema\"\n    )\n    block_type_id: UUID = Field(default=..., description=\"A block type ID\")\n    block_type_name: Optional[str] = Field(None, description=\"A block type name\")\n    block_type: Optional[BlockType] = Field(\n        default=None, description=\"The associated block type\"\n    )\n    block_document_references: Dict[str, Dict[str, Any]] = Field(\n        default_factory=dict, description=\"Record of the block document's references\"\n    )\n    is_anonymous: bool = Field(\n        default=False,\n        description=(\n            \"Whether the block is anonymous (anonymous blocks are usually created by\"\n            \" Prefect automatically)\"\n        ),\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        # the BlockDocumentCreate subclass allows name=None\n        # and will inherit this validator\n        if v is not None:\n            raise_on_name_with_banned_characters(v)\n        return v\n\n    @root_validator\n    def validate_name_is_present_if_not_anonymous(cls, values):\n        # anonymous blocks may have no name prior to actually being\n        # stored in the database\n        if not values.get(\"is_anonymous\") and not values.get(\"name\"):\n            raise ValueError(\"Names must be provided for block documents.\")\n        return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Flow","title":"Flow","text":"

    Bases: ObjectBaseModel

    An ORM representation of flow data.

    Source code in prefect/client/schemas/objects.py
    class Flow(ObjectBaseModel):\n    \"\"\"An ORM representation of flow data.\"\"\"\n\n    name: str = Field(\n        default=..., description=\"The name of the flow\", example=\"my-flow\"\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of flow tags\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunnerSettings","title":"FlowRunnerSettings","text":"

    Bases: PrefectBaseModel

    An API schema for passing details about the flow runner.

    This schema is agnostic to the types and configuration provided by clients

    Source code in prefect/client/schemas/objects.py
    class FlowRunnerSettings(PrefectBaseModel):\n    \"\"\"\n    An API schema for passing details about the flow runner.\n\n    This schema is agnostic to the types and configuration provided by clients\n    \"\"\"\n\n    type: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The type of the flow runner which can be used by the client for\"\n            \" dispatching.\"\n        ),\n    )\n    config: Optional[dict] = Field(\n        default=None, description=\"The configuration for the given flow runner type.\"\n    )\n\n    # The following is required for composite compatibility in the ORM\n\n    def __init__(self, type: str = None, config: dict = None, **kwargs) -> None:\n        # Pydantic does not support positional arguments so they must be converted to\n        # keyword arguments\n        super().__init__(type=type, config=config, **kwargs)\n\n    def __composite_values__(self):\n        return self.type, self.config\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Deployment","title":"Deployment","text":"

    Bases: ObjectBaseModel

    An ORM representation of deployment data.

    Source code in prefect/client/schemas/objects.py
    class Deployment(ObjectBaseModel):\n    \"\"\"An ORM representation of deployment data.\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the deployment.\")\n    version: Optional[str] = Field(\n        default=None, description=\"An optional version for the deployment.\"\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"A description for the deployment.\"\n    )\n    flow_id: UUID = Field(\n        default=..., description=\"The flow id associated with the deployment.\"\n    )\n    schedule: Optional[SCHEDULE_TYPES] = Field(\n        default=None, description=\"A schedule for the deployment.\"\n    )\n    is_schedule_active: bool = Field(\n        default=True, description=\"Whether or not the deployment schedule is active.\"\n    )\n    infra_overrides: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Overrides to apply to the base infrastructure block at runtime.\",\n    )\n    parameters: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Parameters for flow runs scheduled by the deployment.\",\n    )\n    pull_steps: Optional[List[dict]] = Field(\n        default=None,\n        description=\"Pull steps for cloning and running this deployment.\",\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags for the deployment\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The work queue for the deployment. If no work queue is set, work will not\"\n            \" be scheduled.\"\n        ),\n    )\n    last_polled: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The last time the deployment was polled for status updates.\",\n    )\n    parameter_openapi_schema: Optional[Dict[str, Any]] = Field(\n        default=None,\n        description=\"The parameter schema of the flow, including defaults.\",\n    )\n    path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the working directory for the workflow, relative to remote\"\n            \" storage or an absolute path.\"\n        ),\n    )\n    entrypoint: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the entrypoint for the workflow, relative to the `path`.\"\n        ),\n    )\n    manifest_path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the flow's manifest file, relative to the chosen storage.\"\n        ),\n    )\n    storage_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining storage used for this flow.\",\n    )\n    infrastructure_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining infrastructure to use for flow runs.\",\n    )\n    created_by: Optional[CreatedBy] = Field(\n        default=None,\n        description=\"Optional information about the creator of this deployment.\",\n    )\n    updated_by: Optional[UpdatedBy] = Field(\n        default=None,\n        description=\"Optional information about the updater of this deployment.\",\n    )\n    work_queue_id: UUID = Field(\n        default=None,\n        description=(\n            \"The id of the work pool queue to which this deployment is assigned.\"\n        ),\n    )\n    enforce_parameter_schema: bool = Field(\n        default=False,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.ConcurrencyLimit","title":"ConcurrencyLimit","text":"

    Bases: ObjectBaseModel

    An ORM representation of a concurrency limit.

    Source code in prefect/client/schemas/objects.py
    class ConcurrencyLimit(ObjectBaseModel):\n    \"\"\"An ORM representation of a concurrency limit.\"\"\"\n\n    tag: str = Field(\n        default=..., description=\"A tag the concurrency limit is applied to.\"\n    )\n    concurrency_limit: int = Field(default=..., description=\"The concurrency limit.\")\n    active_slots: List[UUID] = Field(\n        default_factory=list,\n        description=\"A list of active run ids using a concurrency slot\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockSchema","title":"BlockSchema","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block schema.

    Source code in prefect/client/schemas/objects.py
    class BlockSchema(ObjectBaseModel):\n    \"\"\"An ORM representation of a block schema.\"\"\"\n\n    checksum: str = Field(default=..., description=\"The block schema's unique checksum\")\n    fields: dict = Field(\n        default_factory=dict, description=\"The block schema's field schema\"\n    )\n    block_type_id: Optional[UUID] = Field(default=..., description=\"A block type ID\")\n    block_type: Optional[BlockType] = Field(\n        default=None, description=\"The associated block type\"\n    )\n    capabilities: List[str] = Field(\n        default_factory=list,\n        description=\"A list of Block capabilities\",\n    )\n    version: str = Field(\n        default=DEFAULT_BLOCK_SCHEMA_VERSION,\n        description=\"Human readable identifier for the block schema\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockSchemaReference","title":"BlockSchemaReference","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block schema reference.

    Source code in prefect/client/schemas/objects.py
    class BlockSchemaReference(ObjectBaseModel):\n    \"\"\"An ORM representation of a block schema reference.\"\"\"\n\n    parent_block_schema_id: UUID = Field(\n        default=..., description=\"ID of block schema the reference is nested within\"\n    )\n    parent_block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The block schema the reference is nested within\"\n    )\n    reference_block_schema_id: UUID = Field(\n        default=..., description=\"ID of the nested block schema\"\n    )\n    reference_block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The nested block schema\"\n    )\n    name: str = Field(\n        default=..., description=\"The name that the reference is nested under\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.BlockDocumentReference","title":"BlockDocumentReference","text":"

    Bases: ObjectBaseModel

    An ORM representation of a block document reference.

    Source code in prefect/client/schemas/objects.py
    class BlockDocumentReference(ObjectBaseModel):\n    \"\"\"An ORM representation of a block document reference.\"\"\"\n\n    parent_block_document_id: UUID = Field(\n        default=..., description=\"ID of block document the reference is nested within\"\n    )\n    parent_block_document: Optional[BlockDocument] = Field(\n        default=None, description=\"The block document the reference is nested within\"\n    )\n    reference_block_document_id: UUID = Field(\n        default=..., description=\"ID of the nested block document\"\n    )\n    reference_block_document: Optional[BlockDocument] = Field(\n        default=None, description=\"The nested block document\"\n    )\n    name: str = Field(\n        default=..., description=\"The name that the reference is nested under\"\n    )\n\n    @root_validator\n    def validate_parent_and_ref_are_different(cls, values):\n        parent_id = values.get(\"parent_block_document_id\")\n        ref_id = values.get(\"reference_block_document_id\")\n        if parent_id and ref_id and parent_id == ref_id:\n            raise ValueError(\n                \"`parent_block_document_id` and `reference_block_document_id` cannot be\"\n                \" the same\"\n            )\n        return values\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.SavedSearchFilter","title":"SavedSearchFilter","text":"

    Bases: PrefectBaseModel

    A filter for a saved search model. Intended for use by the Prefect UI.

    Source code in prefect/client/schemas/objects.py
    class SavedSearchFilter(PrefectBaseModel):\n    \"\"\"A filter for a saved search model. Intended for use by the Prefect UI.\"\"\"\n\n    object: str = Field(default=..., description=\"The object over which to filter.\")\n    property: str = Field(\n        default=..., description=\"The property of the object on which to filter.\"\n    )\n    type: str = Field(default=..., description=\"The type of the property.\")\n    operation: str = Field(\n        default=...,\n        description=\"The operator to apply to the object. For example, `equals`.\",\n    )\n    value: Any = Field(\n        default=..., description=\"A JSON-compatible value for the filter.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.SavedSearch","title":"SavedSearch","text":"

    Bases: ObjectBaseModel

    An ORM representation of saved search data. Represents a set of filter criteria.

    Source code in prefect/client/schemas/objects.py
    class SavedSearch(ObjectBaseModel):\n    \"\"\"An ORM representation of saved search data. Represents a set of filter criteria.\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the saved search.\")\n    filters: List[SavedSearchFilter] = Field(\n        default_factory=list, description=\"The filter set for the saved search.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Log","title":"Log","text":"

    Bases: ObjectBaseModel

    An ORM representation of log data.

    Source code in prefect/client/schemas/objects.py
    class Log(ObjectBaseModel):\n    \"\"\"An ORM representation of log data.\"\"\"\n\n    name: str = Field(default=..., description=\"The logger name.\")\n    level: int = Field(default=..., description=\"The log level.\")\n    message: str = Field(default=..., description=\"The log message.\")\n    timestamp: DateTimeTZ = Field(default=..., description=\"The log timestamp.\")\n    flow_run_id: Optional[UUID] = Field(\n        default=None, description=\"The flow run ID associated with the log.\"\n    )\n    task_run_id: Optional[UUID] = Field(\n        default=None, description=\"The task run ID associated with the log.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.QueueFilter","title":"QueueFilter","text":"

    Bases: PrefectBaseModel

    Filter criteria definition for a work queue.

    Source code in prefect/client/schemas/objects.py
    class QueueFilter(PrefectBaseModel):\n    \"\"\"Filter criteria definition for a work queue.\"\"\"\n\n    tags: Optional[List[str]] = Field(\n        default=None,\n        description=\"Only include flow runs with these tags in the work queue.\",\n    )\n    deployment_ids: Optional[List[UUID]] = Field(\n        default=None,\n        description=\"Only include flow runs from these deployments in the work queue.\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkQueue","title":"WorkQueue","text":"

    Bases: ObjectBaseModel

    An ORM representation of a work queue

    Source code in prefect/client/schemas/objects.py
    class WorkQueue(ObjectBaseModel):\n    \"\"\"An ORM representation of a work queue\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the work queue.\")\n    description: Optional[str] = Field(\n        default=\"\", description=\"An optional description for the work queue.\"\n    )\n    is_paused: bool = Field(\n        default=False, description=\"Whether or not the work queue is paused.\"\n    )\n    concurrency_limit: Optional[conint(ge=0)] = Field(\n        default=None, description=\"An optional concurrency limit for the work queue.\"\n    )\n    priority: conint(ge=1) = Field(\n        default=1,\n        description=(\n            \"The queue's priority. Lower values are higher priority (1 is the highest).\"\n        ),\n    )\n    work_pool_name: Optional[str] = Field(default=None)\n    # Will be required after a future migration\n    work_pool_id: Optional[UUID] = Field(\n        description=\"The work pool with which the queue is associated.\"\n    )\n    filter: Optional[QueueFilter] = Field(\n        default=None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n    last_polled: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The last time an agent polled this queue for work.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkQueueHealthPolicy","title":"WorkQueueHealthPolicy","text":"

    Bases: PrefectBaseModel

    Source code in prefect/client/schemas/objects.py
    class WorkQueueHealthPolicy(PrefectBaseModel):\n    maximum_late_runs: Optional[int] = Field(\n        default=0,\n        description=(\n            \"The maximum number of late runs in the work queue before it is deemed\"\n            \" unhealthy. Defaults to `0`.\"\n        ),\n    )\n    maximum_seconds_since_last_polled: Optional[int] = Field(\n        default=60,\n        description=(\n            \"The maximum number of time in seconds elapsed since work queue has been\"\n            \" polled before it is deemed unhealthy. Defaults to `60`.\"\n        ),\n    )\n\n    def evaluate_health_status(\n        self, late_runs_count: int, last_polled: Optional[DateTimeTZ] = None\n    ) -> bool:\n        \"\"\"\n        Given empirical information about the state of the work queue, evaluate its health status.\n\n        Args:\n            late_runs: the count of late runs for the work queue.\n            last_polled: the last time the work queue was polled, if available.\n\n        Returns:\n            bool: whether or not the work queue is healthy.\n        \"\"\"\n        healthy = True\n        if (\n            self.maximum_late_runs is not None\n            and late_runs_count > self.maximum_late_runs\n        ):\n            healthy = False\n\n        if self.maximum_seconds_since_last_polled is not None:\n            if (\n                last_polled is None\n                or pendulum.now(\"UTC\").diff(last_polled).in_seconds()\n                > self.maximum_seconds_since_last_polled\n            ):\n                healthy = False\n\n        return healthy\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkQueueHealthPolicy.evaluate_health_status","title":"evaluate_health_status","text":"

    Given empirical information about the state of the work queue, evaluate its health status.

    Parameters:

    Name Type Description Default late_runs

    the count of late runs for the work queue.

    required last_polled Optional[DateTimeTZ]

    the last time the work queue was polled, if available.

    None

    Returns:

    Name Type Description bool bool

    whether or not the work queue is healthy.

    Source code in prefect/client/schemas/objects.py
    def evaluate_health_status(\n    self, late_runs_count: int, last_polled: Optional[DateTimeTZ] = None\n) -> bool:\n    \"\"\"\n    Given empirical information about the state of the work queue, evaluate its health status.\n\n    Args:\n        late_runs: the count of late runs for the work queue.\n        last_polled: the last time the work queue was polled, if available.\n\n    Returns:\n        bool: whether or not the work queue is healthy.\n    \"\"\"\n    healthy = True\n    if (\n        self.maximum_late_runs is not None\n        and late_runs_count > self.maximum_late_runs\n    ):\n        healthy = False\n\n    if self.maximum_seconds_since_last_polled is not None:\n        if (\n            last_polled is None\n            or pendulum.now(\"UTC\").diff(last_polled).in_seconds()\n            > self.maximum_seconds_since_last_polled\n        ):\n            healthy = False\n\n    return healthy\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunNotificationPolicy","title":"FlowRunNotificationPolicy","text":"

    Bases: ObjectBaseModel

    An ORM representation of a flow run notification.

    Source code in prefect/client/schemas/objects.py
    class FlowRunNotificationPolicy(ObjectBaseModel):\n    \"\"\"An ORM representation of a flow run notification.\"\"\"\n\n    is_active: bool = Field(\n        default=True, description=\"Whether the policy is currently active\"\n    )\n    state_names: List[str] = Field(\n        default=..., description=\"The flow run states that trigger notifications\"\n    )\n    tags: List[str] = Field(\n        default=...,\n        description=\"The flow run tags that trigger notifications (set [] to disable)\",\n    )\n    block_document_id: UUID = Field(\n        default=..., description=\"The block document ID used for sending notifications\"\n    )\n    message_template: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A templatable notification message. Use {braces} to add variables.\"\n            \" Valid variables include:\"\n            f\" {listrepr(sorted(FLOW_RUN_NOTIFICATION_TEMPLATE_KWARGS), sep=', ')}\"\n        ),\n        example=(\n            \"Flow run {flow_run_name} with id {flow_run_id} entered state\"\n            \" {flow_run_state_name}.\"\n        ),\n    )\n\n    @validator(\"message_template\")\n    def validate_message_template_variables(cls, v):\n        if v is not None:\n            try:\n                v.format(**{k: \"test\" for k in FLOW_RUN_NOTIFICATION_TEMPLATE_KWARGS})\n            except KeyError as exc:\n                raise ValueError(f\"Invalid template variable provided: '{exc.args[0]}'\")\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Agent","title":"Agent","text":"

    Bases: ObjectBaseModel

    An ORM representation of an agent

    Source code in prefect/client/schemas/objects.py
    class Agent(ObjectBaseModel):\n    \"\"\"An ORM representation of an agent\"\"\"\n\n    name: str = Field(\n        default_factory=lambda: generate_slug(2),\n        description=(\n            \"The name of the agent. If a name is not provided, it will be\"\n            \" auto-generated.\"\n        ),\n    )\n    work_queue_id: UUID = Field(\n        default=..., description=\"The work queue with which the agent is associated.\"\n    )\n    last_activity_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The last time this agent polled for work.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkPool","title":"WorkPool","text":"

    Bases: ObjectBaseModel

    An ORM representation of a work pool

    Source code in prefect/client/schemas/objects.py
    class WorkPool(ObjectBaseModel):\n    \"\"\"An ORM representation of a work pool\"\"\"\n\n    name: str = Field(\n        description=\"The name of the work pool.\",\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"A description of the work pool.\"\n    )\n    type: str = Field(description=\"The work pool type.\")\n    base_job_template: Dict[str, Any] = Field(\n        default_factory=dict, description=\"The work pool's base job template.\"\n    )\n    is_paused: bool = Field(\n        default=False,\n        description=\"Pausing the work pool stops the delivery of all work.\",\n    )\n    concurrency_limit: Optional[conint(ge=0)] = Field(\n        default=None, description=\"A concurrency limit for the work pool.\"\n    )\n    status: Optional[WorkPoolStatus] = Field(\n        default=None, description=\"The current status of the work pool.\"\n    )\n\n    # this required field has a default of None so that the custom validator\n    # below will be called and produce a more helpful error message\n    default_queue_id: UUID = Field(\n        None, description=\"The id of the pool's default queue.\"\n    )\n\n    @property\n    def is_push_pool(self) -> bool:\n        return self.type.endswith(\":push\")\n\n    @property\n    def is_managed_pool(self) -> bool:\n        return self.type.endswith(\":managed\")\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n\n    @validator(\"default_queue_id\", always=True)\n    def helpful_error_for_missing_default_queue_id(cls, v):\n        \"\"\"\n        Default queue ID is required because all pools must have a default queue\n        ID, but it represents a circular foreign key relationship to a\n        WorkQueue (which can't be created until the work pool exists).\n        Therefore, while this field can *technically* be null, it shouldn't be.\n        This should only be an issue when creating new pools, as reading\n        existing ones will always have this field populated. This custom error\n        message will help users understand that they should use the\n        `actions.WorkPoolCreate` model in that case.\n        \"\"\"\n        if v is None:\n            raise ValueError(\n                \"`default_queue_id` is a required field. If you are \"\n                \"creating a new WorkPool and don't have a queue \"\n                \"ID yet, use the `actions.WorkPoolCreate` model instead.\"\n            )\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.WorkPool.helpful_error_for_missing_default_queue_id","title":"helpful_error_for_missing_default_queue_id","text":"

    Default queue ID is required because all pools must have a default queue ID, but it represents a circular foreign key relationship to a WorkQueue (which can't be created until the work pool exists). Therefore, while this field can technically be null, it shouldn't be. This should only be an issue when creating new pools, as reading existing ones will always have this field populated. This custom error message will help users understand that they should use the actions.WorkPoolCreate model in that case.

    Source code in prefect/client/schemas/objects.py
    @validator(\"default_queue_id\", always=True)\ndef helpful_error_for_missing_default_queue_id(cls, v):\n    \"\"\"\n    Default queue ID is required because all pools must have a default queue\n    ID, but it represents a circular foreign key relationship to a\n    WorkQueue (which can't be created until the work pool exists).\n    Therefore, while this field can *technically* be null, it shouldn't be.\n    This should only be an issue when creating new pools, as reading\n    existing ones will always have this field populated. This custom error\n    message will help users understand that they should use the\n    `actions.WorkPoolCreate` model in that case.\n    \"\"\"\n    if v is None:\n        raise ValueError(\n            \"`default_queue_id` is a required field. If you are \"\n            \"creating a new WorkPool and don't have a queue \"\n            \"ID yet, use the `actions.WorkPoolCreate` model instead.\"\n        )\n    return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.Worker","title":"Worker","text":"

    Bases: ObjectBaseModel

    An ORM representation of a worker

    Source code in prefect/client/schemas/objects.py
    class Worker(ObjectBaseModel):\n    \"\"\"An ORM representation of a worker\"\"\"\n\n    name: str = Field(description=\"The name of the worker.\")\n    work_pool_id: UUID = Field(\n        description=\"The work pool with which the queue is associated.\"\n    )\n    last_heartbeat_time: datetime.datetime = Field(\n        None, description=\"The last time the worker process sent a heartbeat.\"\n    )\n    heartbeat_interval_seconds: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The number of seconds to expect between heartbeats sent by the worker.\"\n        ),\n    )\n    status: WorkerStatus = Field(\n        WorkerStatus.OFFLINE,\n        description=\"Current status of the worker.\",\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunInput","title":"FlowRunInput","text":"

    Bases: ObjectBaseModel

    Source code in prefect/client/schemas/objects.py
    class FlowRunInput(ObjectBaseModel):\n    flow_run_id: UUID = Field(description=\"The flow run ID associated with the input.\")\n    key: str = Field(description=\"The key of the input.\")\n    value: str = Field(description=\"The value of the input.\")\n    sender: Optional[str] = Field(description=\"The sender of the input.\")\n\n    @property\n    def decoded_value(self) -> Any:\n        \"\"\"\n        Decode the value of the input.\n\n        Returns:\n            Any: the decoded value\n        \"\"\"\n        return orjson.loads(self.value)\n\n    @validator(\"key\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_alphanumeric_dashes_only(v)\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.FlowRunInput.decoded_value","title":"decoded_value: Any property","text":"

    Decode the value of the input.

    Returns:

    Name Type Description Any Any

    the decoded value

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.objects.GlobalConcurrencyLimit","title":"GlobalConcurrencyLimit","text":"

    Bases: ObjectBaseModel

    An ORM representation of a global concurrency limit

    Source code in prefect/client/schemas/objects.py
    class GlobalConcurrencyLimit(ObjectBaseModel):\n    \"\"\"An ORM representation of a global concurrency limit\"\"\"\n\n    name: str = Field(description=\"The name of the global concurrency limit.\")\n    limit: int = Field(\n        description=(\n            \"The maximum number of slots that can be occupied on this concurrency\"\n            \" limit.\"\n        )\n    )\n    active: Optional[bool] = Field(\n        default=True,\n        description=\"Whether or not the concurrency limit is in an active state.\",\n    )\n    active_slots: Optional[int] = Field(\n        default=0,\n        description=\"Number of tasks currently using a concurrency slot.\",\n    )\n    slot_decay_per_second: Optional[int] = Field(\n        default=0,\n        description=(\n            \"Controls the rate at which slots are released when the concurrency limit\"\n            \" is used as a rate limit.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_4","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses","title":"prefect.client.schemas.responses","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.SetStateStatus","title":"SetStateStatus","text":"

    Bases: AutoEnum

    Enumerates return statuses for setting run states.

    Source code in prefect/client/schemas/responses.py
    class SetStateStatus(AutoEnum):\n    \"\"\"Enumerates return statuses for setting run states.\"\"\"\n\n    ACCEPT = AutoEnum.auto()\n    REJECT = AutoEnum.auto()\n    ABORT = AutoEnum.auto()\n    WAIT = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.StateAcceptDetails","title":"StateAcceptDetails","text":"

    Bases: PrefectBaseModel

    Details associated with an ACCEPT state transition.

    Source code in prefect/client/schemas/responses.py
    class StateAcceptDetails(PrefectBaseModel):\n    \"\"\"Details associated with an ACCEPT state transition.\"\"\"\n\n    type: Literal[\"accept_details\"] = Field(\n        default=\"accept_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.StateRejectDetails","title":"StateRejectDetails","text":"

    Bases: PrefectBaseModel

    Details associated with a REJECT state transition.

    Source code in prefect/client/schemas/responses.py
    class StateRejectDetails(PrefectBaseModel):\n    \"\"\"Details associated with a REJECT state transition.\"\"\"\n\n    type: Literal[\"reject_details\"] = Field(\n        default=\"reject_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition was rejected.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.StateAbortDetails","title":"StateAbortDetails","text":"

    Bases: PrefectBaseModel

    Details associated with an ABORT state transition.

    Source code in prefect/client/schemas/responses.py
    class StateAbortDetails(PrefectBaseModel):\n    \"\"\"Details associated with an ABORT state transition.\"\"\"\n\n    type: Literal[\"abort_details\"] = Field(\n        default=\"abort_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition was aborted.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.StateWaitDetails","title":"StateWaitDetails","text":"

    Bases: PrefectBaseModel

    Details associated with a WAIT state transition.

    Source code in prefect/client/schemas/responses.py
    class StateWaitDetails(PrefectBaseModel):\n    \"\"\"Details associated with a WAIT state transition.\"\"\"\n\n    type: Literal[\"wait_details\"] = Field(\n        default=\"wait_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    delay_seconds: int = Field(\n        default=...,\n        description=(\n            \"The length of time in seconds the client should wait before transitioning\"\n            \" states.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition should wait.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.HistoryResponseState","title":"HistoryResponseState","text":"

    Bases: PrefectBaseModel

    Represents a single state's history over an interval.

    Source code in prefect/client/schemas/responses.py
    class HistoryResponseState(PrefectBaseModel):\n    \"\"\"Represents a single state's history over an interval.\"\"\"\n\n    state_type: objects.StateType = Field(default=..., description=\"The state type.\")\n    state_name: str = Field(default=..., description=\"The state name.\")\n    count_runs: int = Field(\n        default=...,\n        description=\"The number of runs in the specified state during the interval.\",\n    )\n    sum_estimated_run_time: datetime.timedelta = Field(\n        default=...,\n        description=\"The total estimated run time of all runs during the interval.\",\n    )\n    sum_estimated_lateness: datetime.timedelta = Field(\n        default=...,\n        description=(\n            \"The sum of differences between actual and expected start time during the\"\n            \" interval.\"\n        ),\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.HistoryResponse","title":"HistoryResponse","text":"

    Bases: PrefectBaseModel

    Represents a history of aggregation states over an interval

    Source code in prefect/client/schemas/responses.py
    class HistoryResponse(PrefectBaseModel):\n    \"\"\"Represents a history of aggregation states over an interval\"\"\"\n\n    interval_start: DateTimeTZ = Field(\n        default=..., description=\"The start date of the interval.\"\n    )\n    interval_end: DateTimeTZ = Field(\n        default=..., description=\"The end date of the interval.\"\n    )\n    states: List[HistoryResponseState] = Field(\n        default=..., description=\"A list of state histories during the interval.\"\n    )\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.OrchestrationResult","title":"OrchestrationResult","text":"

    Bases: PrefectBaseModel

    A container for the output of state orchestration.

    Source code in prefect/client/schemas/responses.py
    class OrchestrationResult(PrefectBaseModel):\n    \"\"\"\n    A container for the output of state orchestration.\n    \"\"\"\n\n    state: Optional[objects.State]\n    status: SetStateStatus\n    details: StateResponseDetails\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.responses.FlowRunResponse","title":"FlowRunResponse","text":"

    Bases: ObjectBaseModel

    Source code in prefect/client/schemas/responses.py
    @copy_model_fields\nclass FlowRunResponse(ObjectBaseModel):\n    name: str = FieldFrom(objects.FlowRun)\n    flow_id: UUID = FieldFrom(objects.FlowRun)\n    state_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    deployment_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    work_queue_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    work_queue_name: Optional[str] = FieldFrom(objects.FlowRun)\n    flow_version: Optional[str] = FieldFrom(objects.FlowRun)\n    parameters: dict = FieldFrom(objects.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(objects.FlowRun)\n    context: dict = FieldFrom(objects.FlowRun)\n    empirical_policy: objects.FlowRunPolicy = FieldFrom(objects.FlowRun)\n    tags: List[str] = FieldFrom(objects.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    state_type: Optional[objects.StateType] = FieldFrom(objects.FlowRun)\n    state_name: Optional[str] = FieldFrom(objects.FlowRun)\n    run_count: int = FieldFrom(objects.FlowRun)\n    expected_start_time: Optional[DateTimeTZ] = FieldFrom(objects.FlowRun)\n    next_scheduled_start_time: Optional[DateTimeTZ] = FieldFrom(objects.FlowRun)\n    start_time: Optional[DateTimeTZ] = FieldFrom(objects.FlowRun)\n    end_time: Optional[DateTimeTZ] = FieldFrom(objects.FlowRun)\n    total_run_time: datetime.timedelta = FieldFrom(objects.FlowRun)\n    estimated_run_time: datetime.timedelta = FieldFrom(objects.FlowRun)\n    estimated_start_time_delta: datetime.timedelta = FieldFrom(objects.FlowRun)\n    auto_scheduled: bool = FieldFrom(objects.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    infrastructure_pid: Optional[str] = FieldFrom(objects.FlowRun)\n    created_by: Optional[CreatedBy] = FieldFrom(objects.FlowRun)\n    work_pool_id: Optional[UUID] = FieldFrom(objects.FlowRun)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the flow run's work pool.\",\n        example=\"my-work-pool\",\n    )\n    state: Optional[objects.State] = FieldFrom(objects.FlowRun)\n\n    def __eq__(self, other: Any) -> bool:\n        \"\"\"\n        Check for \"equality\" to another flow run schema\n\n        Estimates times are rolling and will always change with repeated queries for\n        a flow run so we ignore them during equality checks.\n        \"\"\"\n        if isinstance(other, FlowRunResponse):\n            exclude_fields = {\"estimated_run_time\", \"estimated_start_time_delta\"}\n            return self.dict(exclude=exclude_fields) == other.dict(\n                exclude=exclude_fields\n            )\n        return super().__eq__(other)\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_5","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules","title":"prefect.client.schemas.schedules","text":"

    Schedule schemas

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.IntervalSchedule","title":"IntervalSchedule","text":"

    Bases: PrefectBaseModel

    A schedule formed by adding interval increments to an anchor_date. If no anchor_date is supplied, the current UTC time is used. If a timezone-naive datetime is provided for anchor_date, it is assumed to be in the schedule's timezone (or UTC). Even if supplied with an IANA timezone, anchor dates are always stored as UTC offsets, so a timezone can be provided to determine localization behaviors like DST boundary handling. If none is provided it will be inferred from the anchor date.

    NOTE: If the IntervalSchedule anchor_date or timezone is provided in a DST-observing timezone, then the schedule will adjust itself appropriately. Intervals greater than 24 hours will follow DST conventions, while intervals of less than 24 hours will follow UTC intervals. For example, an hourly schedule will fire every UTC hour, even across DST boundaries. When clocks are set back, this will result in two runs that appear to both be scheduled for 1am local time, even though they are an hour apart in UTC time. For longer intervals, like a daily schedule, the interval schedule will adjust for DST boundaries so that the clock-hour remains constant. This means that a daily schedule that always fires at 9am will observe DST and continue to fire at 9am in the local time zone.

    Parameters:

    Name Type Description Default interval timedelta

    an interval to schedule on

    required anchor_date DateTimeTZ

    an anchor date to schedule increments against; if not provided, the current timestamp will be used

    required timezone str

    a valid timezone string

    required Source code in prefect/client/schemas/schedules.py
    class IntervalSchedule(PrefectBaseModel):\n    \"\"\"\n    A schedule formed by adding `interval` increments to an `anchor_date`. If no\n    `anchor_date` is supplied, the current UTC time is used.  If a\n    timezone-naive datetime is provided for `anchor_date`, it is assumed to be\n    in the schedule's timezone (or UTC). Even if supplied with an IANA timezone,\n    anchor dates are always stored as UTC offsets, so a `timezone` can be\n    provided to determine localization behaviors like DST boundary handling. If\n    none is provided it will be inferred from the anchor date.\n\n    NOTE: If the `IntervalSchedule` `anchor_date` or `timezone` is provided in a\n    DST-observing timezone, then the schedule will adjust itself appropriately.\n    Intervals greater than 24 hours will follow DST conventions, while intervals\n    of less than 24 hours will follow UTC intervals. For example, an hourly\n    schedule will fire every UTC hour, even across DST boundaries. When clocks\n    are set back, this will result in two runs that *appear* to both be\n    scheduled for 1am local time, even though they are an hour apart in UTC\n    time. For longer intervals, like a daily schedule, the interval schedule\n    will adjust for DST boundaries so that the clock-hour remains constant. This\n    means that a daily schedule that always fires at 9am will observe DST and\n    continue to fire at 9am in the local time zone.\n\n    Args:\n        interval (datetime.timedelta): an interval to schedule on\n        anchor_date (DateTimeTZ, optional): an anchor date to schedule increments against;\n            if not provided, the current timestamp will be used\n        timezone (str, optional): a valid timezone string\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n        exclude_none = True\n\n    interval: datetime.timedelta\n    anchor_date: DateTimeTZ = None\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n\n    @validator(\"interval\")\n    def interval_must_be_positive(cls, v):\n        if v.total_seconds() <= 0:\n            raise ValueError(\"The interval must be positive\")\n        return v\n\n    @validator(\"anchor_date\", always=True)\n    def default_anchor_date(cls, v):\n        if v is None:\n            return pendulum.now(\"UTC\")\n        return pendulum.instance(v)\n\n    @validator(\"timezone\", always=True)\n    def default_timezone(cls, v, *, values, **kwargs):\n        # if was provided, make sure its a valid IANA string\n        if v and v not in pendulum.tz.timezones:\n            raise ValueError(f'Invalid timezone: \"{v}\"')\n\n        # otherwise infer the timezone from the anchor date\n        elif v is None and values.get(\"anchor_date\"):\n            tz = values[\"anchor_date\"].tz.name\n            if tz in pendulum.tz.timezones:\n                return tz\n            # sometimes anchor dates have \"timezones\" that are UTC offsets\n            # like \"-04:00\". This happens when parsing ISO8601 strings.\n            # In this case we, the correct inferred localization is \"UTC\".\n            else:\n                return \"UTC\"\n\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.CronSchedule","title":"CronSchedule","text":"

    Bases: PrefectBaseModel

    Cron schedule

    NOTE: If the timezone is a DST-observing one, then the schedule will adjust itself appropriately. Cron's rules for DST are based on schedule times, not intervals. This means that an hourly cron schedule will fire on every new schedule hour, not every elapsed hour; for example, when clocks are set back this will result in a two-hour pause as the schedule will fire the first time 1am is reached and the first time 2am is reached, 120 minutes later. Longer schedules, such as one that fires at 9am every morning, will automatically adjust for DST.

    Parameters:

    Name Type Description Default cron str

    a valid cron string

    required timezone str

    a valid timezone string in IANA tzdata format (for example, America/New_York).

    required day_or bool

    Control how croniter handles day and day_of_week entries. Defaults to True, matching cron which connects those values using OR. If the switch is set to False, the values are connected using AND. This behaves like fcron and enables you to e.g. define a job that executes each 2nd friday of a month by setting the days of month and the weekday.

    required Source code in prefect/client/schemas/schedules.py
    class CronSchedule(PrefectBaseModel):\n    \"\"\"\n    Cron schedule\n\n    NOTE: If the timezone is a DST-observing one, then the schedule will adjust\n    itself appropriately. Cron's rules for DST are based on schedule times, not\n    intervals. This means that an hourly cron schedule will fire on every new\n    schedule hour, not every elapsed hour; for example, when clocks are set back\n    this will result in a two-hour pause as the schedule will fire *the first\n    time* 1am is reached and *the first time* 2am is reached, 120 minutes later.\n    Longer schedules, such as one that fires at 9am every morning, will\n    automatically adjust for DST.\n\n    Args:\n        cron (str): a valid cron string\n        timezone (str): a valid timezone string in IANA tzdata format (for example,\n            America/New_York).\n        day_or (bool, optional): Control how croniter handles `day` and `day_of_week`\n            entries. Defaults to True, matching cron which connects those values using\n            OR. If the switch is set to False, the values are connected using AND. This\n            behaves like fcron and enables you to e.g. define a job that executes each\n            2nd friday of a month by setting the days of month and the weekday.\n\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    cron: str = Field(default=..., example=\"0 0 * * *\")\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n    day_or: bool = Field(\n        default=True,\n        description=(\n            \"Control croniter behavior for handling day and day_of_week entries.\"\n        ),\n    )\n\n    @validator(\"timezone\")\n    def valid_timezone(cls, v):\n        if v and v not in pendulum.tz.timezones:\n            raise ValueError(\n                f'Invalid timezone: \"{v}\" (specify in IANA tzdata format, for example,'\n                \" America/New_York)\"\n            )\n        return v\n\n    @validator(\"cron\")\n    def valid_cron_string(cls, v):\n        # croniter allows \"random\" and \"hashed\" expressions\n        # which we do not support https://github.com/kiorky/croniter\n        if not croniter.is_valid(v):\n            raise ValueError(f'Invalid cron string: \"{v}\"')\n        elif any(c for c in v.split() if c.casefold() in [\"R\", \"H\", \"r\", \"h\"]):\n            raise ValueError(\n                f'Random and Hashed expressions are unsupported, received: \"{v}\"'\n            )\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.RRuleSchedule","title":"RRuleSchedule","text":"

    Bases: PrefectBaseModel

    RRule schedule, based on the iCalendar standard (RFC 5545) as implemented in dateutils.rrule.

    RRules are appropriate for any kind of calendar-date manipulation, including irregular intervals, repetition, exclusions, week day or day-of-month adjustments, and more.

    Note that as a calendar-oriented standard, RRuleSchedules are sensitive to to the initial timezone provided. A 9am daily schedule with a daylight saving time-aware start date will maintain a local 9am time through DST boundaries; a 9am daily schedule with a UTC start date will maintain a 9am UTC time.

    Parameters:

    Name Type Description Default rrule str

    a valid RRule string

    required timezone str

    a valid timezone string

    required Source code in prefect/client/schemas/schedules.py
    class RRuleSchedule(PrefectBaseModel):\n    \"\"\"\n    RRule schedule, based on the iCalendar standard\n    ([RFC 5545](https://datatracker.ietf.org/doc/html/rfc5545)) as\n    implemented in `dateutils.rrule`.\n\n    RRules are appropriate for any kind of calendar-date manipulation, including\n    irregular intervals, repetition, exclusions, week day or day-of-month\n    adjustments, and more.\n\n    Note that as a calendar-oriented standard, `RRuleSchedules` are sensitive to\n    to the initial timezone provided. A 9am daily schedule with a daylight saving\n    time-aware start date will maintain a local 9am time through DST boundaries;\n    a 9am daily schedule with a UTC start date will maintain a 9am UTC time.\n\n    Args:\n        rrule (str): a valid RRule string\n        timezone (str, optional): a valid timezone string\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    rrule: str\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n\n    @validator(\"rrule\")\n    def validate_rrule_str(cls, v):\n        # attempt to parse the rrule string as an rrule object\n        # this will error if the string is invalid\n        try:\n            dateutil.rrule.rrulestr(v, cache=True)\n        except ValueError as exc:\n            # rrules errors are a mix of cryptic and informative\n            # so reraise to be clear that the string was invalid\n            raise ValueError(f'Invalid RRule string \"{v}\": {exc}')\n        if len(v) > MAX_RRULE_LENGTH:\n            raise ValueError(\n                f'Invalid RRule string \"{v[:40]}...\"\\n'\n                f\"Max length is {MAX_RRULE_LENGTH}, got {len(v)}\"\n            )\n        return v\n\n    @classmethod\n    def from_rrule(cls, rrule: dateutil.rrule.rrule):\n        if isinstance(rrule, dateutil.rrule.rrule):\n            if rrule._dtstart.tzinfo is not None:\n                timezone = rrule._dtstart.tzinfo.name\n            else:\n                timezone = \"UTC\"\n            return RRuleSchedule(rrule=str(rrule), timezone=timezone)\n        elif isinstance(rrule, dateutil.rrule.rruleset):\n            dtstarts = [rr._dtstart for rr in rrule._rrule if rr._dtstart is not None]\n            unique_dstarts = set(pendulum.instance(d).in_tz(\"UTC\") for d in dtstarts)\n            unique_timezones = set(d.tzinfo for d in dtstarts if d.tzinfo is not None)\n\n            if len(unique_timezones) > 1:\n                raise ValueError(\n                    f\"rruleset has too many dtstart timezones: {unique_timezones}\"\n                )\n\n            if len(unique_dstarts) > 1:\n                raise ValueError(f\"rruleset has too many dtstarts: {unique_dstarts}\")\n\n            if unique_dstarts and unique_timezones:\n                timezone = dtstarts[0].tzinfo.name\n            else:\n                timezone = \"UTC\"\n\n            rruleset_string = \"\"\n            if rrule._rrule:\n                rruleset_string += \"\\n\".join(str(r) for r in rrule._rrule)\n            if rrule._exrule:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"\\n\".join(str(r) for r in rrule._exrule).replace(\n                    \"RRULE\", \"EXRULE\"\n                )\n            if rrule._rdate:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"RDATE:\" + \",\".join(\n                    rd.strftime(\"%Y%m%dT%H%M%SZ\") for rd in rrule._rdate\n                )\n            if rrule._exdate:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"EXDATE:\" + \",\".join(\n                    exd.strftime(\"%Y%m%dT%H%M%SZ\") for exd in rrule._exdate\n                )\n            return RRuleSchedule(rrule=rruleset_string, timezone=timezone)\n        else:\n            raise ValueError(f\"Invalid RRule object: {rrule}\")\n\n    def to_rrule(self) -> dateutil.rrule.rrule:\n        \"\"\"\n        Since rrule doesn't properly serialize/deserialize timezones, we localize dates\n        here\n        \"\"\"\n        rrule = dateutil.rrule.rrulestr(\n            self.rrule,\n            dtstart=DEFAULT_ANCHOR_DATE,\n            cache=True,\n        )\n        timezone = dateutil.tz.gettz(self.timezone)\n        if isinstance(rrule, dateutil.rrule.rrule):\n            kwargs = dict(dtstart=rrule._dtstart.replace(tzinfo=timezone))\n            if rrule._until:\n                kwargs.update(\n                    until=rrule._until.replace(tzinfo=timezone),\n                )\n            return rrule.replace(**kwargs)\n        elif isinstance(rrule, dateutil.rrule.rruleset):\n            # update rrules\n            localized_rrules = []\n            for rr in rrule._rrule:\n                kwargs = dict(dtstart=rr._dtstart.replace(tzinfo=timezone))\n                if rr._until:\n                    kwargs.update(\n                        until=rr._until.replace(tzinfo=timezone),\n                    )\n                localized_rrules.append(rr.replace(**kwargs))\n            rrule._rrule = localized_rrules\n\n            # update exrules\n            localized_exrules = []\n            for exr in rrule._exrule:\n                kwargs = dict(dtstart=exr._dtstart.replace(tzinfo=timezone))\n                if exr._until:\n                    kwargs.update(\n                        until=exr._until.replace(tzinfo=timezone),\n                    )\n                localized_exrules.append(exr.replace(**kwargs))\n            rrule._exrule = localized_exrules\n\n            # update rdates\n            localized_rdates = []\n            for rd in rrule._rdate:\n                localized_rdates.append(rd.replace(tzinfo=timezone))\n            rrule._rdate = localized_rdates\n\n            # update exdates\n            localized_exdates = []\n            for exd in rrule._exdate:\n                localized_exdates.append(exd.replace(tzinfo=timezone))\n            rrule._exdate = localized_exdates\n\n            return rrule\n\n    @validator(\"timezone\", always=True)\n    def valid_timezone(cls, v):\n        if v and v not in pytz.all_timezones_set:\n            raise ValueError(f'Invalid timezone: \"{v}\"')\n        elif v is None:\n            return \"UTC\"\n        return v\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.RRuleSchedule.to_rrule","title":"to_rrule","text":"

    Since rrule doesn't properly serialize/deserialize timezones, we localize dates here

    Source code in prefect/client/schemas/schedules.py
    def to_rrule(self) -> dateutil.rrule.rrule:\n    \"\"\"\n    Since rrule doesn't properly serialize/deserialize timezones, we localize dates\n    here\n    \"\"\"\n    rrule = dateutil.rrule.rrulestr(\n        self.rrule,\n        dtstart=DEFAULT_ANCHOR_DATE,\n        cache=True,\n    )\n    timezone = dateutil.tz.gettz(self.timezone)\n    if isinstance(rrule, dateutil.rrule.rrule):\n        kwargs = dict(dtstart=rrule._dtstart.replace(tzinfo=timezone))\n        if rrule._until:\n            kwargs.update(\n                until=rrule._until.replace(tzinfo=timezone),\n            )\n        return rrule.replace(**kwargs)\n    elif isinstance(rrule, dateutil.rrule.rruleset):\n        # update rrules\n        localized_rrules = []\n        for rr in rrule._rrule:\n            kwargs = dict(dtstart=rr._dtstart.replace(tzinfo=timezone))\n            if rr._until:\n                kwargs.update(\n                    until=rr._until.replace(tzinfo=timezone),\n                )\n            localized_rrules.append(rr.replace(**kwargs))\n        rrule._rrule = localized_rrules\n\n        # update exrules\n        localized_exrules = []\n        for exr in rrule._exrule:\n            kwargs = dict(dtstart=exr._dtstart.replace(tzinfo=timezone))\n            if exr._until:\n                kwargs.update(\n                    until=exr._until.replace(tzinfo=timezone),\n                )\n            localized_exrules.append(exr.replace(**kwargs))\n        rrule._exrule = localized_exrules\n\n        # update rdates\n        localized_rdates = []\n        for rd in rrule._rdate:\n            localized_rdates.append(rd.replace(tzinfo=timezone))\n        rrule._rdate = localized_rdates\n\n        # update exdates\n        localized_exdates = []\n        for exd in rrule._exdate:\n            localized_exdates.append(exd.replace(tzinfo=timezone))\n        rrule._exdate = localized_exdates\n\n        return rrule\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.schedules.construct_schedule","title":"construct_schedule","text":"

    Construct a schedule from the provided arguments.

    Parameters:

    Name Type Description Default interval Optional[Union[int, float, timedelta]]

    An interval on which to schedule runs. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None anchor_date Optional[Union[datetime, str]]

    The start date for an interval schedule.

    None cron Optional[str]

    A cron schedule for runs.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None timezone Optional[str]

    A timezone to use for the schedule. Defaults to UTC.

    None Source code in prefect/client/schemas/schedules.py
    def construct_schedule(\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    anchor_date: Optional[Union[datetime.datetime, str]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    timezone: Optional[str] = None,\n) -> SCHEDULE_TYPES:\n    \"\"\"\n    Construct a schedule from the provided arguments.\n\n    Args:\n        interval: An interval on which to schedule runs. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        anchor_date: The start date for an interval schedule.\n        cron: A cron schedule for runs.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        timezone: A timezone to use for the schedule. Defaults to UTC.\n    \"\"\"\n    num_schedules = sum(1 for entry in (interval, cron, rrule) if entry is not None)\n    if num_schedules > 1:\n        raise ValueError(\"Only one of interval, cron, or rrule can be provided.\")\n\n    if anchor_date and not interval:\n        raise ValueError(\n            \"An anchor date can only be provided with an interval schedule\"\n        )\n\n    if timezone and not (interval or cron or rrule):\n        raise ValueError(\n            \"A timezone can only be provided with interval, cron, or rrule\"\n        )\n\n    schedule = None\n    if interval:\n        if isinstance(interval, (int, float)):\n            interval = datetime.timedelta(seconds=interval)\n        schedule = IntervalSchedule(\n            interval=interval, anchor_date=anchor_date, timezone=timezone\n        )\n    elif cron:\n        schedule = CronSchedule(cron=cron, timezone=timezone)\n    elif rrule:\n        schedule = RRuleSchedule(rrule=rrule, timezone=timezone)\n\n    if schedule is None:\n        raise ValueError(\"Either interval, cron, or rrule must be provided\")\n\n    return schedule\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#_6","title":"schemas","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting","title":"prefect.client.schemas.sorting","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.FlowRunSort","title":"FlowRunSort","text":"

    Bases: AutoEnum

    Defines flow run sorting options.

    Source code in prefect/client/schemas/sorting.py
    class FlowRunSort(AutoEnum):\n    \"\"\"Defines flow run sorting options.\"\"\"\n\n    ID_DESC = AutoEnum.auto()\n    START_TIME_ASC = AutoEnum.auto()\n    START_TIME_DESC = AutoEnum.auto()\n    EXPECTED_START_TIME_ASC = AutoEnum.auto()\n    EXPECTED_START_TIME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n    NEXT_SCHEDULED_START_TIME_ASC = AutoEnum.auto()\n    END_TIME_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.TaskRunSort","title":"TaskRunSort","text":"

    Bases: AutoEnum

    Defines task run sorting options.

    Source code in prefect/client/schemas/sorting.py
    class TaskRunSort(AutoEnum):\n    \"\"\"Defines task run sorting options.\"\"\"\n\n    ID_DESC = AutoEnum.auto()\n    EXPECTED_START_TIME_ASC = AutoEnum.auto()\n    EXPECTED_START_TIME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n    NEXT_SCHEDULED_START_TIME_ASC = AutoEnum.auto()\n    END_TIME_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.LogSort","title":"LogSort","text":"

    Bases: AutoEnum

    Defines log sorting options.

    Source code in prefect/client/schemas/sorting.py
    class LogSort(AutoEnum):\n    \"\"\"Defines log sorting options.\"\"\"\n\n    TIMESTAMP_ASC = AutoEnum.auto()\n    TIMESTAMP_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.FlowSort","title":"FlowSort","text":"

    Bases: AutoEnum

    Defines flow sorting options.

    Source code in prefect/client/schemas/sorting.py
    class FlowSort(AutoEnum):\n    \"\"\"Defines flow sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.DeploymentSort","title":"DeploymentSort","text":"

    Bases: AutoEnum

    Defines deployment sorting options.

    Source code in prefect/client/schemas/sorting.py
    class DeploymentSort(AutoEnum):\n    \"\"\"Defines deployment sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.ArtifactSort","title":"ArtifactSort","text":"

    Bases: AutoEnum

    Defines artifact sorting options.

    Source code in prefect/client/schemas/sorting.py
    class ArtifactSort(AutoEnum):\n    \"\"\"Defines artifact sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    ID_DESC = AutoEnum.auto()\n    KEY_DESC = AutoEnum.auto()\n    KEY_ASC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.ArtifactCollectionSort","title":"ArtifactCollectionSort","text":"

    Bases: AutoEnum

    Defines artifact collection sorting options.

    Source code in prefect/client/schemas/sorting.py
    class ArtifactCollectionSort(AutoEnum):\n    \"\"\"Defines artifact collection sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    ID_DESC = AutoEnum.auto()\n    KEY_DESC = AutoEnum.auto()\n    KEY_ASC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.VariableSort","title":"VariableSort","text":"

    Bases: AutoEnum

    Defines variables sorting options.

    Source code in prefect/client/schemas/sorting.py
    class VariableSort(AutoEnum):\n    \"\"\"Defines variables sorting options.\"\"\"\n\n    CREATED_DESC = \"CREATED_DESC\"\n    UPDATED_DESC = \"UPDATED_DESC\"\n    NAME_DESC = \"NAME_DESC\"\n    NAME_ASC = \"NAME_ASC\"\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/schemas/#prefect.client.schemas.sorting.BlockDocumentSort","title":"BlockDocumentSort","text":"

    Bases: AutoEnum

    Defines block document sorting options.

    Source code in prefect/client/schemas/sorting.py
    class BlockDocumentSort(AutoEnum):\n    \"\"\"Defines block document sorting options.\"\"\"\n\n    NAME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    BLOCK_TYPE_AND_NAME_ASC = AutoEnum.auto()\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/utilities/","title":"utilities","text":"","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/utilities/#prefect.client.utilities","title":"prefect.client.utilities","text":"

    Utilities for working with clients.

    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/client/utilities/#prefect.client.utilities.inject_client","title":"inject_client","text":"

    Simple helper to provide a context managed client to a asynchronous function.

    The decorated function must take a client kwarg and if a client is passed when called it will be used instead of creating a new one, but it will not be context managed as it is assumed that the caller is managing the context.

    Source code in prefect/client/utilities.py
    def inject_client(fn):\n    \"\"\"\n    Simple helper to provide a context managed client to a asynchronous function.\n\n    The decorated function _must_ take a `client` kwarg and if a client is passed when\n    called it will be used instead of creating a new one, but it will not be context\n    managed as it is assumed that the caller is managing the context.\n    \"\"\"\n\n    @wraps(fn)\n    async def with_injected_client(*args, **kwargs):\n        from prefect.client.orchestration import get_client\n        from prefect.context import FlowRunContext, TaskRunContext\n\n        flow_run_context = FlowRunContext.get()\n        task_run_context = TaskRunContext.get()\n        client = None\n        client_context = asyncnullcontext()\n\n        if \"client\" in kwargs and kwargs[\"client\"] is not None:\n            # Client provided in kwargs\n            client = kwargs[\"client\"]\n        elif flow_run_context and flow_run_context.client._loop == get_running_loop():\n            # Client available from flow run context\n            client = flow_run_context.client\n        elif task_run_context and task_run_context.client._loop == get_running_loop():\n            # Client available from task run context\n            client = task_run_context.client\n\n        else:\n            # A new client is needed\n            client_context = get_client()\n\n        # Removes existing client to allow it to be set by setdefault below\n        kwargs.pop(\"client\", None)\n\n        async with client_context as new_client:\n            kwargs.setdefault(\"client\", new_client or client)\n            return await fn(*args, **kwargs)\n\n    return with_injected_client\n
    ","tags":["Python API","REST API"]},{"location":"api-ref/prefect/concurrency/asyncio/","title":"asyncio","text":"","tags":["Python API","concurrency","asyncio"]},{"location":"api-ref/prefect/concurrency/asyncio/#prefect.concurrency.asyncio","title":"prefect.concurrency.asyncio","text":"","tags":["Python API","concurrency","asyncio"]},{"location":"api-ref/prefect/concurrency/asyncio/#prefect.concurrency.asyncio.ConcurrencySlotAcquisitionError","title":"ConcurrencySlotAcquisitionError","text":"

    Bases: Exception

    Raised when an unhandlable occurs while acquiring concurrency slots.

    Source code in prefect/concurrency/asyncio.py
    class ConcurrencySlotAcquisitionError(Exception):\n    \"\"\"Raised when an unhandlable occurs while acquiring concurrency slots.\"\"\"\n
    ","tags":["Python API","concurrency","asyncio"]},{"location":"api-ref/prefect/concurrency/asyncio/#prefect.concurrency.asyncio.rate_limit","title":"rate_limit async","text":"

    Block execution until an occupy number of slots of the concurrency limits given in names are acquired. Requires that all given concurrency limits have a slot decay.

    Parameters:

    Name Type Description Default names Union[str, List[str]]

    The names of the concurrency limits to acquire slots from.

    required occupy int

    The number of slots to acquire and hold from each limit.

    1 Source code in prefect/concurrency/asyncio.py
    async def rate_limit(names: Union[str, List[str]], occupy: int = 1):\n    \"\"\"Block execution until an `occupy` number of slots of the concurrency\n    limits given in `names` are acquired. Requires that all given concurrency\n    limits have a slot decay.\n\n    Args:\n        names: The names of the concurrency limits to acquire slots from.\n        occupy: The number of slots to acquire and hold from each limit.\n    \"\"\"\n    names = names if isinstance(names, list) else [names]\n    limits = await _acquire_concurrency_slots(names, occupy, mode=\"rate_limit\")\n    _emit_concurrency_acquisition_events(limits, occupy)\n
    ","tags":["Python API","concurrency","asyncio"]},{"location":"api-ref/prefect/concurrency/common/","title":"common","text":"","tags":["Python API","concurrency","common"]},{"location":"api-ref/prefect/concurrency/common/#prefect.concurrency.common","title":"prefect.concurrency.common","text":"","tags":["Python API","concurrency","common"]},{"location":"api-ref/prefect/concurrency/events/","title":"events","text":"","tags":["Python API","concurrency"]},{"location":"api-ref/prefect/concurrency/events/#prefect.concurrency.events","title":"prefect.concurrency.events","text":"","tags":["Python API","concurrency"]},{"location":"api-ref/prefect/concurrency/services/","title":"services","text":"","tags":["Python API","concurrency"]},{"location":"api-ref/prefect/concurrency/services/#prefect.concurrency.services","title":"prefect.concurrency.services","text":"","tags":["Python API","concurrency"]},{"location":"api-ref/prefect/concurrency/sync/","title":"sync","text":"","tags":["Python API","concurrency","sync"]},{"location":"api-ref/prefect/concurrency/sync/#prefect.concurrency.sync","title":"prefect.concurrency.sync","text":"","tags":["Python API","concurrency","sync"]},{"location":"api-ref/prefect/concurrency/sync/#prefect.concurrency.sync.rate_limit","title":"rate_limit","text":"

    Block execution until an occupy number of slots of the concurrency limits given in names are acquired. Requires that all given concurrency limits have a slot decay.

    Parameters:

    Name Type Description Default names Union[str, List[str]]

    The names of the concurrency limits to acquire slots from.

    required occupy int

    The number of slots to acquire and hold from each limit.

    1 Source code in prefect/concurrency/sync.py
    def rate_limit(names: Union[str, List[str]], occupy: int = 1):\n    \"\"\"Block execution until an `occupy` number of slots of the concurrency\n    limits given in `names` are acquired. Requires that all given concurrency\n    limits have a slot decay.\n\n    Args:\n        names: The names of the concurrency limits to acquire slots from.\n        occupy: The number of slots to acquire and hold from each limit.\n    \"\"\"\n    names = names if isinstance(names, list) else [names]\n    limits = _call_async_function_from_sync(\n        _acquire_concurrency_slots, names, occupy, mode=\"rate_limit\"\n    )\n    _emit_concurrency_acquisition_events(limits, occupy)\n
    ","tags":["Python API","concurrency","sync"]},{"location":"api-ref/prefect/deployments/base/","title":"base","text":"","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base","title":"prefect.deployments.base","text":"

    Core primitives for managing Prefect projects. Projects provide a minimally opinionated build system for managing flows and deployments.

    To get started, follow along with the deloyments tutorial.

    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.configure_project_by_recipe","title":"configure_project_by_recipe","text":"

    Given a recipe name, returns a dictionary representing base configuration options.

    Parameters:

    Name Type Description Default recipe str

    the name of the recipe to use

    required formatting_kwargs dict

    additional keyword arguments to format the recipe

    {}

    Raises:

    Type Description ValueError

    if provided recipe name does not exist.

    Source code in prefect/deployments/base.py
    def configure_project_by_recipe(recipe: str, **formatting_kwargs) -> dict:\n    \"\"\"\n    Given a recipe name, returns a dictionary representing base configuration options.\n\n    Args:\n        recipe (str): the name of the recipe to use\n        formatting_kwargs (dict, optional): additional keyword arguments to format the recipe\n\n    Raises:\n        ValueError: if provided recipe name does not exist.\n    \"\"\"\n    # load the recipe\n    recipe_path = Path(__file__).parent / \"recipes\" / recipe / \"prefect.yaml\"\n\n    if not recipe_path.exists():\n        raise ValueError(f\"Unknown recipe {recipe!r} provided.\")\n\n    with recipe_path.open(mode=\"r\") as f:\n        config = yaml.safe_load(f)\n\n    config = apply_values(\n        template=config, values=formatting_kwargs, remove_notset=False\n    )\n\n    return config\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.create_default_prefect_yaml","title":"create_default_prefect_yaml","text":"

    Creates default prefect.yaml file in the provided path if one does not already exist; returns boolean specifying whether a file was created.

    Parameters:

    Name Type Description Default name str

    the name of the project; if not provided, the current directory name will be used

    None contents dict

    a dictionary of contents to write to the file; if not provided, defaults will be used

    None Source code in prefect/deployments/base.py
    def create_default_prefect_yaml(\n    path: str, name: str = None, contents: dict = None\n) -> bool:\n    \"\"\"\n    Creates default `prefect.yaml` file in the provided path if one does not already exist;\n    returns boolean specifying whether a file was created.\n\n    Args:\n        name (str, optional): the name of the project; if not provided, the current directory name\n            will be used\n        contents (dict, optional): a dictionary of contents to write to the file; if not provided,\n            defaults will be used\n    \"\"\"\n    path = Path(path)\n    prefect_file = path / \"prefect.yaml\"\n    if prefect_file.exists():\n        return False\n    default_file = Path(__file__).parent / \"templates\" / \"prefect.yaml\"\n\n    with default_file.open(mode=\"r\") as df:\n        default_contents = yaml.safe_load(df)\n\n    import prefect\n\n    contents[\"prefect-version\"] = prefect.__version__\n    contents[\"name\"] = name\n\n    with prefect_file.open(mode=\"w\") as f:\n        # write header\n        f.write(\n            \"# Welcome to your prefect.yaml file! You can use this file for storing and\"\n            \" managing\\n# configuration for deploying your flows. We recommend\"\n            \" committing this file to source\\n# control along with your flow code.\\n\\n\"\n        )\n\n        f.write(\"# Generic metadata about this project\\n\")\n        yaml.dump({\"name\": contents[\"name\"]}, f, sort_keys=False)\n        yaml.dump({\"prefect-version\": contents[\"prefect-version\"]}, f, sort_keys=False)\n        f.write(\"\\n\")\n\n        # build\n        f.write(\"# build section allows you to manage and build docker images\\n\")\n        yaml.dump(\n            {\"build\": contents.get(\"build\", default_contents.get(\"build\"))},\n            f,\n            sort_keys=False,\n        )\n        f.write(\"\\n\")\n\n        # push\n        f.write(\n            \"# push section allows you to manage if and how this project is uploaded to\"\n            \" remote locations\\n\"\n        )\n        yaml.dump(\n            {\"push\": contents.get(\"push\", default_contents.get(\"push\"))},\n            f,\n            sort_keys=False,\n        )\n        f.write(\"\\n\")\n\n        # pull\n        f.write(\n            \"# pull section allows you to provide instructions for cloning this project\"\n            \" in remote locations\\n\"\n        )\n        yaml.dump(\n            {\"pull\": contents.get(\"pull\", default_contents.get(\"pull\"))},\n            f,\n            sort_keys=False,\n        )\n        f.write(\"\\n\")\n\n        # deployments\n        f.write(\n            \"# the deployments section allows you to provide configuration for\"\n            \" deploying flows\\n\"\n        )\n        yaml.dump(\n            {\n                \"deployments\": contents.get(\n                    \"deployments\", default_contents.get(\"deployments\")\n                )\n            },\n            f,\n            sort_keys=False,\n        )\n    return True\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.find_prefect_directory","title":"find_prefect_directory","text":"

    Given a path, recurses upward looking for .prefect/ directories.

    Once found, returns absolute path to the ./prefect directory, which is assumed to reside within the root for the current project.

    If one is never found, None is returned.

    Source code in prefect/deployments/base.py
    def find_prefect_directory(path: Path = None) -> Optional[Path]:\n    \"\"\"\n    Given a path, recurses upward looking for .prefect/ directories.\n\n    Once found, returns absolute path to the ./prefect directory, which is assumed to reside within the\n    root for the current project.\n\n    If one is never found, `None` is returned.\n    \"\"\"\n    path = Path(path or \".\").resolve()\n    parent = path.parent.resolve()\n    while path != parent:\n        prefect_dir = path.joinpath(\".prefect\")\n        if prefect_dir.is_dir():\n            return prefect_dir\n\n        path = parent.resolve()\n        parent = path.parent.resolve()\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.initialize_project","title":"initialize_project","text":"

    Initializes a basic project structure with base files. If no name is provided, the name of the current directory is used. If no recipe is provided, one is inferred.

    Parameters:

    Name Type Description Default name str

    the name of the project; if not provided, the current directory name

    None recipe str

    the name of the recipe to use; if not provided, one is inferred

    None inputs dict

    a dictionary of inputs to use when formatting the recipe

    None

    Returns:

    Type Description List[str]

    List[str]: a list of files / directories that were created

    Source code in prefect/deployments/base.py
    def initialize_project(\n    name: str = None, recipe: str = None, inputs: dict = None\n) -> List[str]:\n    \"\"\"\n    Initializes a basic project structure with base files.  If no name is provided, the name\n    of the current directory is used.  If no recipe is provided, one is inferred.\n\n    Args:\n        name (str, optional): the name of the project; if not provided, the current directory name\n        recipe (str, optional): the name of the recipe to use; if not provided, one is inferred\n        inputs (dict, optional): a dictionary of inputs to use when formatting the recipe\n\n    Returns:\n        List[str]: a list of files / directories that were created\n    \"\"\"\n    # determine if in git repo or use directory name as a default\n    is_git_based = False\n    formatting_kwargs = {\"directory\": str(Path(\".\").absolute().resolve())}\n    dir_name = os.path.basename(os.getcwd())\n\n    remote_url = _get_git_remote_origin_url()\n    if remote_url:\n        formatting_kwargs[\"repository\"] = remote_url\n        is_git_based = True\n        branch = _get_git_branch()\n        formatting_kwargs[\"branch\"] = branch or \"main\"\n\n    formatting_kwargs[\"name\"] = dir_name\n\n    has_dockerfile = Path(\"Dockerfile\").exists()\n\n    if has_dockerfile:\n        formatting_kwargs[\"dockerfile\"] = \"Dockerfile\"\n    elif recipe is not None and \"docker\" in recipe:\n        formatting_kwargs[\"dockerfile\"] = \"auto\"\n\n    # hand craft a pull step\n    if is_git_based and recipe is None:\n        if has_dockerfile:\n            recipe = \"docker-git\"\n        else:\n            recipe = \"git\"\n    elif recipe is None and has_dockerfile:\n        recipe = \"docker\"\n    elif recipe is None:\n        recipe = \"local\"\n\n    formatting_kwargs.update(inputs or {})\n    configuration = configure_project_by_recipe(recipe=recipe, **formatting_kwargs)\n\n    project_name = name or dir_name\n\n    files = []\n    if create_default_ignore_file(\".\"):\n        files.append(\".prefectignore\")\n    if create_default_prefect_yaml(\".\", name=project_name, contents=configuration):\n        files.append(\"prefect.yaml\")\n    if set_prefect_hidden_dir():\n        files.append(\".prefect/\")\n\n    return files\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.register_flow","title":"register_flow async","text":"

    Register a flow with this project from an entrypoint.

    Parameters:

    Name Type Description Default entrypoint str

    the entrypoint to the flow to register

    required force bool

    whether or not to overwrite an existing flow with the same name

    False

    Raises:

    Type Description ValueError

    if force is False and registration would overwrite an existing flow

    Source code in prefect/deployments/base.py
    async def register_flow(entrypoint: str, force: bool = False):\n    \"\"\"\n    Register a flow with this project from an entrypoint.\n\n    Args:\n        entrypoint (str): the entrypoint to the flow to register\n        force (bool, optional): whether or not to overwrite an existing flow with the same name\n\n    Raises:\n        ValueError: if `force` is `False` and registration would overwrite an existing flow\n    \"\"\"\n    try:\n        fpath, obj_name = entrypoint.rsplit(\":\", 1)\n    except ValueError as exc:\n        if str(exc) == \"not enough values to unpack (expected 2, got 1)\":\n            missing_flow_name_msg = (\n                \"Your flow entrypoint must include the name of the function that is\"\n                f\" the entrypoint to your flow.\\nTry {entrypoint}:<flow_name> as your\"\n                f\" entrypoint. If you meant to specify '{entrypoint}' as the deployment\"\n                f\" name, try `prefect deploy -n {entrypoint}`.\"\n            )\n            raise ValueError(missing_flow_name_msg)\n        else:\n            raise exc\n\n    flow = await run_sync_in_worker_thread(load_flow_from_entrypoint, entrypoint)\n\n    fpath = Path(fpath).absolute()\n    prefect_dir = find_prefect_directory()\n    if not prefect_dir:\n        raise FileNotFoundError(\n            \"No .prefect directory could be found - run `prefect project\"\n            \" init` to create one.\"\n        )\n\n    entrypoint = f\"{fpath.relative_to(prefect_dir.parent)!s}:{obj_name}\"\n\n    flows_file = prefect_dir / \"flows.json\"\n    if flows_file.exists():\n        with flows_file.open(mode=\"r\") as f:\n            flows = json.load(f)\n    else:\n        flows = {}\n\n    ## quality control\n    if flow.name in flows and flows[flow.name] != entrypoint:\n        if not force:\n            raise ValueError(\n                f\"Conflicting entry found for flow with name {flow.name!r}.\\nExisting\"\n                f\" entrypoint: {flows[flow.name]}\\nAttempted entrypoint:\"\n                f\" {entrypoint}\\n\\nYou can try removing the existing entry for\"\n                f\" {flow.name!r} from your [yellow]~/.prefect/flows.json[/yellow].\"\n            )\n\n    flows[flow.name] = entrypoint\n\n    with flows_file.open(mode=\"w\") as f:\n        json.dump(flows, f, sort_keys=True, indent=2)\n\n    return flow\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/base/#prefect.deployments.base.set_prefect_hidden_dir","title":"set_prefect_hidden_dir","text":"

    Creates default .prefect/ directory if one does not already exist. Returns boolean specifying whether or not a directory was created.

    If a path is provided, the directory will be created in that location.

    Source code in prefect/deployments/base.py
    def set_prefect_hidden_dir(path: str = None) -> bool:\n    \"\"\"\n    Creates default `.prefect/` directory if one does not already exist.\n    Returns boolean specifying whether or not a directory was created.\n\n    If a path is provided, the directory will be created in that location.\n    \"\"\"\n    path = Path(path or \".\") / \".prefect\"\n\n    # use exists so that we dont accidentally overwrite a file\n    if path.exists():\n        return False\n    path.mkdir(mode=0o0700)\n    return True\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/deployments/","title":"deployments","text":"","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments","title":"prefect.deployments.deployments","text":"

    Objects for specifying deployments and utilities for loading flows from deployments.

    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment","title":"Deployment","text":"

    Bases: BaseModel

    A Prefect Deployment definition, used for specifying and building deployments.

    Parameters:

    Name Type Description Default name

    A name for the deployment (required).

    required version

    An optional version for the deployment; defaults to the flow's version

    required description

    An optional description of the deployment; defaults to the flow's description

    required tags

    An optional list of tags to associate with this deployment; note that tags are used only for organizational purposes. For delegating work to agents, see work_queue_name.

    required schedule

    A schedule to run this deployment on, once registered

    required is_schedule_active

    Whether or not the schedule is active

    required work_queue_name

    The work queue that will handle this deployment's runs

    required work_pool_name

    The work pool for the deployment

    required flow_name

    The name of the flow this deployment encapsulates

    required parameters

    A dictionary of parameter values to pass to runs created from this deployment

    required infrastructure

    An optional infrastructure block used to configure infrastructure for runs; if not provided, will default to running this deployment in Agent subprocesses

    required infra_overrides

    A dictionary of dot delimited infrastructure overrides that will be applied at runtime; for example env.CONFIG_KEY=config_value or namespace='prefect'

    required storage

    An optional remote storage block used to store and retrieve this workflow; if not provided, will default to referencing this flow by its local path

    required path

    The path to the working directory for the workflow, relative to remote storage or, if stored on a local filesystem, an absolute path

    required entrypoint

    The path to the entrypoint for the workflow, always relative to the path

    required parameter_openapi_schema

    The parameter schema of the flow, including defaults.

    required enforce_parameter_schema

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    required
    Create a new deployment using configuration defaults for an imported flow:\n\n>>> from my_project.flows import my_flow\n>>> from prefect.deployments import Deployment\n>>>\n>>> deployment = Deployment.build_from_flow(\n...     flow=my_flow,\n...     name=\"example\",\n...     version=\"1\",\n...     tags=[\"demo\"],\n>>> )\n>>> deployment.apply()\n\nCreate a new deployment with custom storage and an infrastructure override:\n\n>>> from my_project.flows import my_flow\n>>> from prefect.deployments import Deployment\n>>> from prefect.filesystems import S3\n\n>>> storage = S3.load(\"dev-bucket\") # load a pre-defined block\n>>> deployment = Deployment.build_from_flow(\n...     flow=my_flow,\n...     name=\"s3-example\",\n...     version=\"2\",\n...     tags=[\"aws\"],\n...     storage=storage,\n...     infra_overrides=dict(\"env.PREFECT_LOGGING_LEVEL\"=\"DEBUG\"),\n>>> )\n>>> deployment.apply()\n
    Source code in prefect/deployments/deployments.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None and x != DEFAULT_AGENT_WORK_POOL_NAME,\n)\nclass Deployment(BaseModel):\n    \"\"\"\n    A Prefect Deployment definition, used for specifying and building deployments.\n\n    Args:\n        name: A name for the deployment (required).\n        version: An optional version for the deployment; defaults to the flow's version\n        description: An optional description of the deployment; defaults to the flow's\n            description\n        tags: An optional list of tags to associate with this deployment; note that tags\n            are used only for organizational purposes. For delegating work to agents,\n            see `work_queue_name`.\n        schedule: A schedule to run this deployment on, once registered\n        is_schedule_active: Whether or not the schedule is active\n        work_queue_name: The work queue that will handle this deployment's runs\n        work_pool_name: The work pool for the deployment\n        flow_name: The name of the flow this deployment encapsulates\n        parameters: A dictionary of parameter values to pass to runs created from this\n            deployment\n        infrastructure: An optional infrastructure block used to configure\n            infrastructure for runs; if not provided, will default to running this\n            deployment in Agent subprocesses\n        infra_overrides: A dictionary of dot delimited infrastructure overrides that\n            will be applied at runtime; for example `env.CONFIG_KEY=config_value` or\n            `namespace='prefect'`\n        storage: An optional remote storage block used to store and retrieve this\n            workflow; if not provided, will default to referencing this flow by its\n            local path\n        path: The path to the working directory for the workflow, relative to remote\n            storage or, if stored on a local filesystem, an absolute path\n        entrypoint: The path to the entrypoint for the workflow, always relative to the\n            `path`\n        parameter_openapi_schema: The parameter schema of the flow, including defaults.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n\n    Examples:\n\n        Create a new deployment using configuration defaults for an imported flow:\n\n        >>> from my_project.flows import my_flow\n        >>> from prefect.deployments import Deployment\n        >>>\n        >>> deployment = Deployment.build_from_flow(\n        ...     flow=my_flow,\n        ...     name=\"example\",\n        ...     version=\"1\",\n        ...     tags=[\"demo\"],\n        >>> )\n        >>> deployment.apply()\n\n        Create a new deployment with custom storage and an infrastructure override:\n\n        >>> from my_project.flows import my_flow\n        >>> from prefect.deployments import Deployment\n        >>> from prefect.filesystems import S3\n\n        >>> storage = S3.load(\"dev-bucket\") # load a pre-defined block\n        >>> deployment = Deployment.build_from_flow(\n        ...     flow=my_flow,\n        ...     name=\"s3-example\",\n        ...     version=\"2\",\n        ...     tags=[\"aws\"],\n        ...     storage=storage,\n        ...     infra_overrides=dict(\"env.PREFECT_LOGGING_LEVEL\"=\"DEBUG\"),\n        >>> )\n        >>> deployment.apply()\n\n    \"\"\"\n\n    class Config:\n        json_encoders = {SecretDict: lambda v: v.dict()}\n        validate_assignment = True\n        extra = \"forbid\"\n\n    @property\n    def _editable_fields(self) -> List[str]:\n        editable_fields = [\n            \"name\",\n            \"description\",\n            \"version\",\n            \"work_queue_name\",\n            \"work_pool_name\",\n            \"tags\",\n            \"parameters\",\n            \"schedule\",\n            \"is_schedule_active\",\n            \"infra_overrides\",\n        ]\n\n        # if infrastructure is baked as a pre-saved block, then\n        # editing its fields will not update anything\n        if self.infrastructure._block_document_id:\n            return editable_fields\n        else:\n            return editable_fields + [\"infrastructure\"]\n\n    @property\n    def location(self) -> str:\n        \"\"\"\n        The 'location' that this deployment points to is given by `path` alone\n        in the case of no remote storage, and otherwise by `storage.basepath / path`.\n\n        The underlying flow entrypoint is interpreted relative to this location.\n        \"\"\"\n        location = \"\"\n        if self.storage:\n            location = (\n                self.storage.basepath + \"/\"\n                if not self.storage.basepath.endswith(\"/\")\n                else \"\"\n            )\n        if self.path:\n            location += self.path\n        return location\n\n    @sync_compatible\n    async def to_yaml(self, path: Path) -> None:\n        yaml_dict = self._yaml_dict()\n        schema = self.schema()\n\n        with open(path, \"w\") as f:\n            # write header\n            f.write(\n                \"###\\n### A complete description of a Prefect Deployment for flow\"\n                f\" {self.flow_name!r}\\n###\\n\"\n            )\n\n            # write editable fields\n            for field in self._editable_fields:\n                # write any comments\n                if schema[\"properties\"][field].get(\"yaml_comment\"):\n                    f.write(f\"# {schema['properties'][field]['yaml_comment']}\\n\")\n                # write the field\n                yaml.dump({field: yaml_dict[field]}, f, sort_keys=False)\n\n            # write non-editable fields\n            f.write(\"\\n###\\n### DO NOT EDIT BELOW THIS LINE\\n###\\n\")\n            yaml.dump(\n                {k: v for k, v in yaml_dict.items() if k not in self._editable_fields},\n                f,\n                sort_keys=False,\n            )\n\n    def _yaml_dict(self) -> dict:\n        \"\"\"\n        Returns a YAML-compatible representation of this deployment as a dictionary.\n        \"\"\"\n        # avoids issues with UUIDs showing up in YAML\n        all_fields = json.loads(\n            self.json(\n                exclude={\n                    \"storage\": {\"_filesystem\", \"filesystem\", \"_remote_file_system\"}\n                }\n            )\n        )\n        if all_fields[\"storage\"]:\n            all_fields[\"storage\"][\n                \"_block_type_slug\"\n            ] = self.storage.get_block_type_slug()\n        if all_fields[\"infrastructure\"]:\n            all_fields[\"infrastructure\"][\n                \"_block_type_slug\"\n            ] = self.infrastructure.get_block_type_slug()\n        return all_fields\n\n    # top level metadata\n    name: str = Field(..., description=\"The name of the deployment.\")\n    description: Optional[str] = Field(\n        default=None, description=\"An optional description of the deployment.\"\n    )\n    version: Optional[str] = Field(\n        default=None, description=\"An optional version for the deployment.\"\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"One of more tags to apply to this deployment.\",\n    )\n    schedule: SCHEDULE_TYPES = None\n    is_schedule_active: Optional[bool] = Field(\n        default=None, description=\"Whether or not the schedule is active.\"\n    )\n    flow_name: Optional[str] = Field(default=None, description=\"The name of the flow.\")\n    work_queue_name: Optional[str] = Field(\n        \"default\",\n        description=\"The work queue for the deployment.\",\n        yaml_comment=\"The work queue that will handle this deployment's runs\",\n    )\n    work_pool_name: Optional[str] = Field(\n        default=None, description=\"The work pool for the deployment\"\n    )\n    # flow data\n    parameters: Dict[str, Any] = Field(default_factory=dict)\n    manifest_path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the flow's manifest file, relative to the chosen storage.\"\n        ),\n    )\n    infrastructure: Infrastructure = Field(default_factory=Process)\n    infra_overrides: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Overrides to apply to the base infrastructure block at runtime.\",\n    )\n    storage: Optional[Block] = Field(\n        None,\n        help=\"The remote storage to use for this workflow.\",\n    )\n    path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the working directory for the workflow, relative to remote\"\n            \" storage or an absolute path.\"\n        ),\n    )\n    entrypoint: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the entrypoint for the workflow, relative to the `path`.\"\n        ),\n    )\n    parameter_openapi_schema: ParameterSchema = Field(\n        default_factory=ParameterSchema,\n        description=\"The parameter schema of the flow, including defaults.\",\n    )\n    timestamp: datetime = Field(default_factory=partial(pendulum.now, \"UTC\"))\n    triggers: List[DeploymentTrigger] = Field(\n        default_factory=list,\n        description=\"The triggers that should cause this deployment to run.\",\n    )\n    # defaults to None to allow for backwards compatibility\n    enforce_parameter_schema: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Whether or not the Prefect API should enforce the parameter schema for\"\n            \" this deployment.\"\n        ),\n    )\n\n    @validator(\"infrastructure\", pre=True)\n    def infrastructure_must_have_capabilities(cls, value):\n        if isinstance(value, dict):\n            if \"_block_type_slug\" in value:\n                # Replace private attribute with public for dispatch\n                value[\"block_type_slug\"] = value.pop(\"_block_type_slug\")\n            block = Block(**value)\n        elif value is None:\n            return value\n        else:\n            block = value\n\n        if \"run-infrastructure\" not in block.get_block_capabilities():\n            raise ValueError(\n                \"Infrastructure block must have 'run-infrastructure' capabilities.\"\n            )\n        return block\n\n    @validator(\"storage\", pre=True)\n    def storage_must_have_capabilities(cls, value):\n        if isinstance(value, dict):\n            block_type = Block.get_block_class_from_key(value.pop(\"_block_type_slug\"))\n            block = block_type(**value)\n        elif value is None:\n            return value\n        else:\n            block = value\n\n        capabilities = block.get_block_capabilities()\n        if \"get-directory\" not in capabilities:\n            raise ValueError(\n                \"Remote Storage block must have 'get-directory' capabilities.\"\n            )\n        return block\n\n    @validator(\"parameter_openapi_schema\", pre=True)\n    def handle_openapi_schema(cls, value):\n        \"\"\"\n        This method ensures setting a value of `None` is handled gracefully.\n        \"\"\"\n        if value is None:\n            return ParameterSchema()\n        return value\n\n    @validator(\"triggers\")\n    def validate_automation_names(cls, field_value, values, field, config):\n        \"\"\"Ensure that each trigger has a name for its automation if none is provided.\"\"\"\n        for i, trigger in enumerate(field_value, start=1):\n            if trigger.name is None:\n                trigger.name = f\"{values['name']}__automation_{i}\"\n\n        return field_value\n\n    @classmethod\n    @sync_compatible\n    async def load_from_yaml(cls, path: str):\n        data = yaml.safe_load(await anyio.Path(path).read_bytes())\n        # load blocks from server to ensure secret values are properly hydrated\n        if data.get(\"storage\"):\n            block_doc_name = data[\"storage\"].get(\"_block_document_name\")\n            # if no doc name, this block is not stored on the server\n            if block_doc_name:\n                block_slug = data[\"storage\"][\"_block_type_slug\"]\n                block = await Block.load(f\"{block_slug}/{block_doc_name}\")\n                data[\"storage\"] = block\n\n        if data.get(\"infrastructure\"):\n            block_doc_name = data[\"infrastructure\"].get(\"_block_document_name\")\n            # if no doc name, this block is not stored on the server\n            if block_doc_name:\n                block_slug = data[\"infrastructure\"][\"_block_type_slug\"]\n                block = await Block.load(f\"{block_slug}/{block_doc_name}\")\n                data[\"infrastructure\"] = block\n\n            return cls(**data)\n\n    @sync_compatible\n    async def load(self) -> bool:\n        \"\"\"\n        Queries the API for a deployment with this name for this flow, and if found,\n        prepopulates any settings that were not set at initialization.\n\n        Returns a boolean specifying whether a load was successful or not.\n\n        Raises:\n            - ValueError: if both name and flow name are not set\n        \"\"\"\n        if not self.name or not self.flow_name:\n            raise ValueError(\"Both a deployment name and flow name must be provided.\")\n        async with get_client() as client:\n            try:\n                deployment = await client.read_deployment_by_name(\n                    f\"{self.flow_name}/{self.name}\"\n                )\n                if deployment.storage_document_id:\n                    Block._from_block_document(\n                        await client.read_block_document(deployment.storage_document_id)\n                    )\n\n                excluded_fields = self.__fields_set__.union(\n                    {\n                        \"infrastructure\",\n                        \"storage\",\n                        \"timestamp\",\n                        \"triggers\",\n                        \"enforce_parameter_schema\",\n                    }\n                )\n                for field in set(self.__fields__.keys()) - excluded_fields:\n                    new_value = getattr(deployment, field)\n                    setattr(self, field, new_value)\n\n                if \"infrastructure\" not in self.__fields_set__:\n                    if deployment.infrastructure_document_id:\n                        self.infrastructure = Block._from_block_document(\n                            await client.read_block_document(\n                                deployment.infrastructure_document_id\n                            )\n                        )\n                if \"storage\" not in self.__fields_set__:\n                    if deployment.storage_document_id:\n                        self.storage = Block._from_block_document(\n                            await client.read_block_document(\n                                deployment.storage_document_id\n                            )\n                        )\n            except ObjectNotFound:\n                return False\n        return True\n\n    @sync_compatible\n    async def update(self, ignore_none: bool = False, **kwargs):\n        \"\"\"\n        Performs an in-place update with the provided settings.\n\n        Args:\n            ignore_none: if True, all `None` values are ignored when performing the\n                update\n        \"\"\"\n        unknown_keys = set(kwargs.keys()) - set(self.dict().keys())\n        if unknown_keys:\n            raise ValueError(\n                f\"Received unexpected attributes: {', '.join(unknown_keys)}\"\n            )\n        for key, value in kwargs.items():\n            if ignore_none and value is None:\n                continue\n            setattr(self, key, value)\n\n    @sync_compatible\n    async def upload_to_storage(\n        self, storage_block: str = None, ignore_file: str = \".prefectignore\"\n    ) -> Optional[int]:\n        \"\"\"\n        Uploads the workflow this deployment represents using a provided storage block;\n        if no block is provided, defaults to configuring self for local storage.\n\n        Args:\n            storage_block: a string reference a remote storage block slug `$type/$name`;\n                if provided, used to upload the workflow's project\n            ignore_file: an optional path to a `.prefectignore` file that specifies\n                filename patterns to ignore when uploading to remote storage; if not\n                provided, looks for `.prefectignore` in the current working directory\n        \"\"\"\n        file_count = None\n        if storage_block:\n            storage = await Block.load(storage_block)\n\n            if \"put-directory\" not in storage.get_block_capabilities():\n                raise BlockMissingCapabilities(\n                    f\"Storage block {storage!r} missing 'put-directory' capability.\"\n                )\n\n            self.storage = storage\n\n            # upload current directory to storage location\n            file_count = await self.storage.put_directory(\n                ignore_file=ignore_file, to_path=self.path\n            )\n        elif self.storage:\n            if \"put-directory\" not in self.storage.get_block_capabilities():\n                raise BlockMissingCapabilities(\n                    f\"Storage block {self.storage!r} missing 'put-directory'\"\n                    \" capability.\"\n                )\n\n            file_count = await self.storage.put_directory(\n                ignore_file=ignore_file, to_path=self.path\n            )\n\n        # persists storage now in case it contains secret values\n        if self.storage and not self.storage._block_document_id:\n            await self.storage._save(is_anonymous=True)\n\n        return file_count\n\n    @sync_compatible\n    async def apply(\n        self, upload: bool = False, work_queue_concurrency: int = None\n    ) -> UUID:\n        \"\"\"\n        Registers this deployment with the API and returns the deployment's ID.\n\n        Args:\n            upload: if True, deployment files are automatically uploaded to remote\n                storage\n            work_queue_concurrency: If provided, sets the concurrency limit on the\n                deployment's work queue\n        \"\"\"\n        if not self.name or not self.flow_name:\n            raise ValueError(\"Both a deployment name and flow name must be set.\")\n        async with get_client() as client:\n            # prep IDs\n            flow_id = await client.create_flow_from_name(self.flow_name)\n\n            infrastructure_document_id = self.infrastructure._block_document_id\n            if not infrastructure_document_id:\n                # if not building off a block, will create an anonymous block\n                self.infrastructure = self.infrastructure.copy()\n                infrastructure_document_id = await self.infrastructure._save(\n                    is_anonymous=True,\n                )\n\n            if upload:\n                await self.upload_to_storage()\n\n            if self.work_queue_name and work_queue_concurrency is not None:\n                try:\n                    res = await client.create_work_queue(\n                        name=self.work_queue_name, work_pool_name=self.work_pool_name\n                    )\n                except ObjectAlreadyExists:\n                    res = await client.read_work_queue_by_name(\n                        name=self.work_queue_name, work_pool_name=self.work_pool_name\n                    )\n                await client.update_work_queue(\n                    res.id, concurrency_limit=work_queue_concurrency\n                )\n\n            # we assume storage was already saved\n            storage_document_id = getattr(self.storage, \"_block_document_id\", None)\n            deployment_id = await client.create_deployment(\n                flow_id=flow_id,\n                name=self.name,\n                work_queue_name=self.work_queue_name,\n                work_pool_name=self.work_pool_name,\n                version=self.version,\n                schedule=self.schedule,\n                is_schedule_active=self.is_schedule_active,\n                parameters=self.parameters,\n                description=self.description,\n                tags=self.tags,\n                manifest_path=self.manifest_path,  # allows for backwards YAML compat\n                path=self.path,\n                entrypoint=self.entrypoint,\n                infra_overrides=self.infra_overrides,\n                storage_document_id=storage_document_id,\n                infrastructure_document_id=infrastructure_document_id,\n                parameter_openapi_schema=self.parameter_openapi_schema.dict(),\n                enforce_parameter_schema=self.enforce_parameter_schema,\n            )\n\n            if client.server_type == ServerType.CLOUD:\n                # The triggers defined in the deployment spec are, essentially,\n                # anonymous and attempting truly sync them with cloud is not\n                # feasible. Instead, we remove all automations that are owned\n                # by the deployment, meaning that they were created via this\n                # mechanism below, and then recreate them.\n                await client.delete_resource_owned_automations(\n                    f\"prefect.deployment.{deployment_id}\"\n                )\n                for trigger in self.triggers:\n                    trigger.set_deployment_id(deployment_id)\n                    await client.create_automation(trigger.as_automation())\n\n            return deployment_id\n\n    @classmethod\n    @sync_compatible\n    async def build_from_flow(\n        cls,\n        flow: Flow,\n        name: str,\n        output: str = None,\n        skip_upload: bool = False,\n        ignore_file: str = \".prefectignore\",\n        apply: bool = False,\n        load_existing: bool = True,\n        **kwargs,\n    ) -> \"Deployment\":\n        \"\"\"\n        Configure a deployment for a given flow.\n\n        Args:\n            flow: A flow function to deploy\n            name: A name for the deployment\n            output (optional): if provided, the full deployment specification will be\n                written as a YAML file in the location specified by `output`\n            skip_upload: if True, deployment files are not automatically uploaded to\n                remote storage\n            ignore_file: an optional path to a `.prefectignore` file that specifies\n                filename patterns to ignore when uploading to remote storage; if not\n                provided, looks for `.prefectignore` in the current working directory\n            apply: if True, the deployment is automatically registered with the API\n            load_existing: if True, load any settings that may already be configured for\n                the named deployment server-side (e.g., schedules, default parameter\n                values, etc.)\n            **kwargs: other keyword arguments to pass to the constructor for the\n                `Deployment` class\n        \"\"\"\n        if not name:\n            raise ValueError(\"A deployment name must be provided.\")\n\n        # note that `deployment.load` only updates settings that were *not*\n        # provided at initialization\n        deployment = cls(name=name, **kwargs)\n        deployment.flow_name = flow.name\n        if not deployment.entrypoint:\n            ## first see if an entrypoint can be determined\n            flow_file = getattr(flow, \"__globals__\", {}).get(\"__file__\")\n            mod_name = getattr(flow, \"__module__\", None)\n            if not flow_file:\n                if not mod_name:\n                    # todo, check if the file location was manually set already\n                    raise ValueError(\"Could not determine flow's file location.\")\n                module = importlib.import_module(mod_name)\n                flow_file = getattr(module, \"__file__\", None)\n                if not flow_file:\n                    raise ValueError(\"Could not determine flow's file location.\")\n\n            # set entrypoint\n            entry_path = Path(flow_file).absolute().relative_to(Path(\".\").absolute())\n            deployment.entrypoint = f\"{entry_path}:{flow.fn.__name__}\"\n\n        if load_existing:\n            await deployment.load()\n\n        # set a few attributes for this flow object\n        deployment.parameter_openapi_schema = parameter_schema(flow)\n\n        # ensure the ignore file exists\n        if not Path(ignore_file).exists():\n            Path(ignore_file).touch()\n\n        if not deployment.version:\n            deployment.version = flow.version\n        if not deployment.description:\n            deployment.description = flow.description\n\n        # proxy for whether infra is docker-based\n        is_docker_based = hasattr(deployment.infrastructure, \"image\")\n\n        if not deployment.storage and not is_docker_based and not deployment.path:\n            deployment.path = str(Path(\".\").absolute())\n        elif not deployment.storage and is_docker_based:\n            # only update if a path is not already set\n            if not deployment.path:\n                deployment.path = \"/opt/prefect/flows\"\n\n        if not skip_upload:\n            if (\n                deployment.storage\n                and \"put-directory\" in deployment.storage.get_block_capabilities()\n            ):\n                await deployment.upload_to_storage(ignore_file=ignore_file)\n\n        if output:\n            await deployment.to_yaml(output)\n\n        if apply:\n            await deployment.apply()\n\n        return deployment\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.location","title":"location: str property","text":"

    The 'location' that this deployment points to is given by path alone in the case of no remote storage, and otherwise by storage.basepath / path.

    The underlying flow entrypoint is interpreted relative to this location.

    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.apply","title":"apply async","text":"

    Registers this deployment with the API and returns the deployment's ID.

    Parameters:

    Name Type Description Default upload bool

    if True, deployment files are automatically uploaded to remote storage

    False work_queue_concurrency int

    If provided, sets the concurrency limit on the deployment's work queue

    None Source code in prefect/deployments/deployments.py
    @sync_compatible\nasync def apply(\n    self, upload: bool = False, work_queue_concurrency: int = None\n) -> UUID:\n    \"\"\"\n    Registers this deployment with the API and returns the deployment's ID.\n\n    Args:\n        upload: if True, deployment files are automatically uploaded to remote\n            storage\n        work_queue_concurrency: If provided, sets the concurrency limit on the\n            deployment's work queue\n    \"\"\"\n    if not self.name or not self.flow_name:\n        raise ValueError(\"Both a deployment name and flow name must be set.\")\n    async with get_client() as client:\n        # prep IDs\n        flow_id = await client.create_flow_from_name(self.flow_name)\n\n        infrastructure_document_id = self.infrastructure._block_document_id\n        if not infrastructure_document_id:\n            # if not building off a block, will create an anonymous block\n            self.infrastructure = self.infrastructure.copy()\n            infrastructure_document_id = await self.infrastructure._save(\n                is_anonymous=True,\n            )\n\n        if upload:\n            await self.upload_to_storage()\n\n        if self.work_queue_name and work_queue_concurrency is not None:\n            try:\n                res = await client.create_work_queue(\n                    name=self.work_queue_name, work_pool_name=self.work_pool_name\n                )\n            except ObjectAlreadyExists:\n                res = await client.read_work_queue_by_name(\n                    name=self.work_queue_name, work_pool_name=self.work_pool_name\n                )\n            await client.update_work_queue(\n                res.id, concurrency_limit=work_queue_concurrency\n            )\n\n        # we assume storage was already saved\n        storage_document_id = getattr(self.storage, \"_block_document_id\", None)\n        deployment_id = await client.create_deployment(\n            flow_id=flow_id,\n            name=self.name,\n            work_queue_name=self.work_queue_name,\n            work_pool_name=self.work_pool_name,\n            version=self.version,\n            schedule=self.schedule,\n            is_schedule_active=self.is_schedule_active,\n            parameters=self.parameters,\n            description=self.description,\n            tags=self.tags,\n            manifest_path=self.manifest_path,  # allows for backwards YAML compat\n            path=self.path,\n            entrypoint=self.entrypoint,\n            infra_overrides=self.infra_overrides,\n            storage_document_id=storage_document_id,\n            infrastructure_document_id=infrastructure_document_id,\n            parameter_openapi_schema=self.parameter_openapi_schema.dict(),\n            enforce_parameter_schema=self.enforce_parameter_schema,\n        )\n\n        if client.server_type == ServerType.CLOUD:\n            # The triggers defined in the deployment spec are, essentially,\n            # anonymous and attempting truly sync them with cloud is not\n            # feasible. Instead, we remove all automations that are owned\n            # by the deployment, meaning that they were created via this\n            # mechanism below, and then recreate them.\n            await client.delete_resource_owned_automations(\n                f\"prefect.deployment.{deployment_id}\"\n            )\n            for trigger in self.triggers:\n                trigger.set_deployment_id(deployment_id)\n                await client.create_automation(trigger.as_automation())\n\n        return deployment_id\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.build_from_flow","title":"build_from_flow async classmethod","text":"

    Configure a deployment for a given flow.

    Parameters:

    Name Type Description Default flow Flow

    A flow function to deploy

    required name str

    A name for the deployment

    required output optional

    if provided, the full deployment specification will be written as a YAML file in the location specified by output

    None skip_upload bool

    if True, deployment files are not automatically uploaded to remote storage

    False ignore_file str

    an optional path to a .prefectignore file that specifies filename patterns to ignore when uploading to remote storage; if not provided, looks for .prefectignore in the current working directory

    '.prefectignore' apply bool

    if True, the deployment is automatically registered with the API

    False load_existing bool

    if True, load any settings that may already be configured for the named deployment server-side (e.g., schedules, default parameter values, etc.)

    True **kwargs

    other keyword arguments to pass to the constructor for the Deployment class

    {} Source code in prefect/deployments/deployments.py
    @classmethod\n@sync_compatible\nasync def build_from_flow(\n    cls,\n    flow: Flow,\n    name: str,\n    output: str = None,\n    skip_upload: bool = False,\n    ignore_file: str = \".prefectignore\",\n    apply: bool = False,\n    load_existing: bool = True,\n    **kwargs,\n) -> \"Deployment\":\n    \"\"\"\n    Configure a deployment for a given flow.\n\n    Args:\n        flow: A flow function to deploy\n        name: A name for the deployment\n        output (optional): if provided, the full deployment specification will be\n            written as a YAML file in the location specified by `output`\n        skip_upload: if True, deployment files are not automatically uploaded to\n            remote storage\n        ignore_file: an optional path to a `.prefectignore` file that specifies\n            filename patterns to ignore when uploading to remote storage; if not\n            provided, looks for `.prefectignore` in the current working directory\n        apply: if True, the deployment is automatically registered with the API\n        load_existing: if True, load any settings that may already be configured for\n            the named deployment server-side (e.g., schedules, default parameter\n            values, etc.)\n        **kwargs: other keyword arguments to pass to the constructor for the\n            `Deployment` class\n    \"\"\"\n    if not name:\n        raise ValueError(\"A deployment name must be provided.\")\n\n    # note that `deployment.load` only updates settings that were *not*\n    # provided at initialization\n    deployment = cls(name=name, **kwargs)\n    deployment.flow_name = flow.name\n    if not deployment.entrypoint:\n        ## first see if an entrypoint can be determined\n        flow_file = getattr(flow, \"__globals__\", {}).get(\"__file__\")\n        mod_name = getattr(flow, \"__module__\", None)\n        if not flow_file:\n            if not mod_name:\n                # todo, check if the file location was manually set already\n                raise ValueError(\"Could not determine flow's file location.\")\n            module = importlib.import_module(mod_name)\n            flow_file = getattr(module, \"__file__\", None)\n            if not flow_file:\n                raise ValueError(\"Could not determine flow's file location.\")\n\n        # set entrypoint\n        entry_path = Path(flow_file).absolute().relative_to(Path(\".\").absolute())\n        deployment.entrypoint = f\"{entry_path}:{flow.fn.__name__}\"\n\n    if load_existing:\n        await deployment.load()\n\n    # set a few attributes for this flow object\n    deployment.parameter_openapi_schema = parameter_schema(flow)\n\n    # ensure the ignore file exists\n    if not Path(ignore_file).exists():\n        Path(ignore_file).touch()\n\n    if not deployment.version:\n        deployment.version = flow.version\n    if not deployment.description:\n        deployment.description = flow.description\n\n    # proxy for whether infra is docker-based\n    is_docker_based = hasattr(deployment.infrastructure, \"image\")\n\n    if not deployment.storage and not is_docker_based and not deployment.path:\n        deployment.path = str(Path(\".\").absolute())\n    elif not deployment.storage and is_docker_based:\n        # only update if a path is not already set\n        if not deployment.path:\n            deployment.path = \"/opt/prefect/flows\"\n\n    if not skip_upload:\n        if (\n            deployment.storage\n            and \"put-directory\" in deployment.storage.get_block_capabilities()\n        ):\n            await deployment.upload_to_storage(ignore_file=ignore_file)\n\n    if output:\n        await deployment.to_yaml(output)\n\n    if apply:\n        await deployment.apply()\n\n    return deployment\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.handle_openapi_schema","title":"handle_openapi_schema","text":"

    This method ensures setting a value of None is handled gracefully.

    Source code in prefect/deployments/deployments.py
    @validator(\"parameter_openapi_schema\", pre=True)\ndef handle_openapi_schema(cls, value):\n    \"\"\"\n    This method ensures setting a value of `None` is handled gracefully.\n    \"\"\"\n    if value is None:\n        return ParameterSchema()\n    return value\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.load","title":"load async","text":"

    Queries the API for a deployment with this name for this flow, and if found, prepopulates any settings that were not set at initialization.

    Returns a boolean specifying whether a load was successful or not.

    Raises:

    Type Description -ValueError

    if both name and flow name are not set

    Source code in prefect/deployments/deployments.py
    @sync_compatible\nasync def load(self) -> bool:\n    \"\"\"\n    Queries the API for a deployment with this name for this flow, and if found,\n    prepopulates any settings that were not set at initialization.\n\n    Returns a boolean specifying whether a load was successful or not.\n\n    Raises:\n        - ValueError: if both name and flow name are not set\n    \"\"\"\n    if not self.name or not self.flow_name:\n        raise ValueError(\"Both a deployment name and flow name must be provided.\")\n    async with get_client() as client:\n        try:\n            deployment = await client.read_deployment_by_name(\n                f\"{self.flow_name}/{self.name}\"\n            )\n            if deployment.storage_document_id:\n                Block._from_block_document(\n                    await client.read_block_document(deployment.storage_document_id)\n                )\n\n            excluded_fields = self.__fields_set__.union(\n                {\n                    \"infrastructure\",\n                    \"storage\",\n                    \"timestamp\",\n                    \"triggers\",\n                    \"enforce_parameter_schema\",\n                }\n            )\n            for field in set(self.__fields__.keys()) - excluded_fields:\n                new_value = getattr(deployment, field)\n                setattr(self, field, new_value)\n\n            if \"infrastructure\" not in self.__fields_set__:\n                if deployment.infrastructure_document_id:\n                    self.infrastructure = Block._from_block_document(\n                        await client.read_block_document(\n                            deployment.infrastructure_document_id\n                        )\n                    )\n            if \"storage\" not in self.__fields_set__:\n                if deployment.storage_document_id:\n                    self.storage = Block._from_block_document(\n                        await client.read_block_document(\n                            deployment.storage_document_id\n                        )\n                    )\n        except ObjectNotFound:\n            return False\n    return True\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.update","title":"update async","text":"

    Performs an in-place update with the provided settings.

    Parameters:

    Name Type Description Default ignore_none bool

    if True, all None values are ignored when performing the update

    False Source code in prefect/deployments/deployments.py
    @sync_compatible\nasync def update(self, ignore_none: bool = False, **kwargs):\n    \"\"\"\n    Performs an in-place update with the provided settings.\n\n    Args:\n        ignore_none: if True, all `None` values are ignored when performing the\n            update\n    \"\"\"\n    unknown_keys = set(kwargs.keys()) - set(self.dict().keys())\n    if unknown_keys:\n        raise ValueError(\n            f\"Received unexpected attributes: {', '.join(unknown_keys)}\"\n        )\n    for key, value in kwargs.items():\n        if ignore_none and value is None:\n            continue\n        setattr(self, key, value)\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.upload_to_storage","title":"upload_to_storage async","text":"

    Uploads the workflow this deployment represents using a provided storage block; if no block is provided, defaults to configuring self for local storage.

    Parameters:

    Name Type Description Default storage_block str

    a string reference a remote storage block slug $type/$name; if provided, used to upload the workflow's project

    None ignore_file str

    an optional path to a .prefectignore file that specifies filename patterns to ignore when uploading to remote storage; if not provided, looks for .prefectignore in the current working directory

    '.prefectignore' Source code in prefect/deployments/deployments.py
    @sync_compatible\nasync def upload_to_storage(\n    self, storage_block: str = None, ignore_file: str = \".prefectignore\"\n) -> Optional[int]:\n    \"\"\"\n    Uploads the workflow this deployment represents using a provided storage block;\n    if no block is provided, defaults to configuring self for local storage.\n\n    Args:\n        storage_block: a string reference a remote storage block slug `$type/$name`;\n            if provided, used to upload the workflow's project\n        ignore_file: an optional path to a `.prefectignore` file that specifies\n            filename patterns to ignore when uploading to remote storage; if not\n            provided, looks for `.prefectignore` in the current working directory\n    \"\"\"\n    file_count = None\n    if storage_block:\n        storage = await Block.load(storage_block)\n\n        if \"put-directory\" not in storage.get_block_capabilities():\n            raise BlockMissingCapabilities(\n                f\"Storage block {storage!r} missing 'put-directory' capability.\"\n            )\n\n        self.storage = storage\n\n        # upload current directory to storage location\n        file_count = await self.storage.put_directory(\n            ignore_file=ignore_file, to_path=self.path\n        )\n    elif self.storage:\n        if \"put-directory\" not in self.storage.get_block_capabilities():\n            raise BlockMissingCapabilities(\n                f\"Storage block {self.storage!r} missing 'put-directory'\"\n                \" capability.\"\n            )\n\n        file_count = await self.storage.put_directory(\n            ignore_file=ignore_file, to_path=self.path\n        )\n\n    # persists storage now in case it contains secret values\n    if self.storage and not self.storage._block_document_id:\n        await self.storage._save(is_anonymous=True)\n\n    return file_count\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.Deployment.validate_automation_names","title":"validate_automation_names","text":"

    Ensure that each trigger has a name for its automation if none is provided.

    Source code in prefect/deployments/deployments.py
    @validator(\"triggers\")\ndef validate_automation_names(cls, field_value, values, field, config):\n    \"\"\"Ensure that each trigger has a name for its automation if none is provided.\"\"\"\n    for i, trigger in enumerate(field_value, start=1):\n        if trigger.name is None:\n            trigger.name = f\"{values['name']}__automation_{i}\"\n\n    return field_value\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.load_deployments_from_yaml","title":"load_deployments_from_yaml","text":"

    Load deployments from a yaml file.

    Source code in prefect/deployments/deployments.py
    def load_deployments_from_yaml(\n    path: str,\n) -> PrefectObjectRegistry:\n    \"\"\"\n    Load deployments from a yaml file.\n    \"\"\"\n    with open(path, \"r\") as f:\n        contents = f.read()\n\n    # Parse into a yaml tree to retrieve separate documents\n    nodes = yaml.compose_all(contents)\n\n    with PrefectObjectRegistry(capture_failures=True) as registry:\n        for node in nodes:\n            with tmpchdir(path):\n                deployment_dict = yaml.safe_load(yaml.serialize(node))\n                # The return value is not necessary, just instantiating the Deployment\n                # is enough to get it recorded on the registry\n                parse_obj_as(Deployment, deployment_dict)\n\n    return registry\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.load_flow_from_flow_run","title":"load_flow_from_flow_run async","text":"

    Load a flow from the location/script provided in a deployment's storage document.

    If ignore_storage=True is provided, no pull from remote storage occurs. This flag is largely for testing, and assumes the flow is already available locally.

    Source code in prefect/deployments/deployments.py
    @inject_client\nasync def load_flow_from_flow_run(\n    flow_run: FlowRun,\n    client: PrefectClient,\n    ignore_storage: bool = False,\n    storage_base_path: Optional[str] = None,\n) -> Flow:\n    \"\"\"\n    Load a flow from the location/script provided in a deployment's storage document.\n\n    If `ignore_storage=True` is provided, no pull from remote storage occurs.  This flag\n    is largely for testing, and assumes the flow is already available locally.\n    \"\"\"\n    deployment = await client.read_deployment(flow_run.deployment_id)\n    logger = flow_run_logger(flow_run)\n\n    runner_storage_base_path = storage_base_path or os.environ.get(\n        \"PREFECT__STORAGE_BASE_PATH\"\n    )\n\n    if not ignore_storage and not deployment.pull_steps:\n        sys.path.insert(0, \".\")\n        if deployment.storage_document_id:\n            storage_document = await client.read_block_document(\n                deployment.storage_document_id\n            )\n            storage_block = Block._from_block_document(storage_document)\n        else:\n            basepath = deployment.path or Path(deployment.manifest_path).parent\n            if runner_storage_base_path:\n                basepath = str(basepath).replace(\n                    \"$STORAGE_BASE_PATH\", runner_storage_base_path\n                )\n            storage_block = LocalFileSystem(basepath=basepath)\n\n        from_path = (\n            str(deployment.path).replace(\"$STORAGE_BASE_PATH\", runner_storage_base_path)\n            if runner_storage_base_path and deployment.path\n            else deployment.path\n        )\n        logger.info(f\"Downloading flow code from storage at {from_path!r}\")\n        await storage_block.get_directory(from_path=from_path, local_path=\".\")\n\n    if deployment.pull_steps:\n        logger.debug(f\"Running {len(deployment.pull_steps)} deployment pull steps\")\n        output = await run_steps(deployment.pull_steps)\n        if output.get(\"directory\"):\n            logger.debug(f\"Changing working directory to {output['directory']!r}\")\n            os.chdir(output[\"directory\"])\n\n    import_path = relative_path_to_current_platform(deployment.entrypoint)\n    logger.debug(f\"Importing flow code from '{import_path}'\")\n\n    # for backwards compat\n    if deployment.manifest_path:\n        with open(deployment.manifest_path, \"r\") as f:\n            import_path = json.load(f)[\"import_path\"]\n            import_path = (\n                Path(deployment.manifest_path).parent / import_path\n            ).absolute()\n    flow = await run_sync_in_worker_thread(load_flow_from_entrypoint, str(import_path))\n    return flow\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/deployments/#prefect.deployments.deployments.run_deployment","title":"run_deployment async","text":"

    Create a flow run for a deployment and return it after completion or a timeout.

    This function will return when the created flow run enters any terminal state or the timeout is reached. If the timeout is reached and the flow run has not reached a terminal state, it will still be returned. When using a timeout, we suggest checking the state of the flow run if completion is important moving forward.

    Parameters:

    Name Type Description Default name Union[str, UUID]

    The deployment id or deployment name in the form: <slugified-flow-name>/<slugified-deployment-name>

    required parameters Optional[dict]

    Parameter overrides for this flow run. Merged with the deployment defaults.

    None scheduled_time Optional[datetime]

    The time to schedule the flow run for, defaults to scheduling the flow run to start now.

    None flow_run_name Optional[str]

    A name for the created flow run

    None timeout Optional[float]

    The amount of time to wait for the flow run to complete before returning. Setting timeout to 0 will return the flow run immediately. Setting timeout to None will allow this function to poll indefinitely. Defaults to None

    None poll_interval Optional[float]

    The number of seconds between polls

    5 tags Optional[Iterable[str]]

    A list of tags to associate with this flow run; note that tags are used only for organizational purposes.

    None idempotency_key Optional[str]

    A unique value to recognize retries of the same run, and prevent creating multiple flow runs.

    None work_queue_name Optional[str]

    The name of a work queue to use for this run. Defaults to the default work queue for the deployment.

    None Source code in prefect/deployments/deployments.py
    @sync_compatible\n@inject_client\nasync def run_deployment(\n    name: Union[str, UUID],\n    client: Optional[PrefectClient] = None,\n    parameters: Optional[dict] = None,\n    scheduled_time: Optional[datetime] = None,\n    flow_run_name: Optional[str] = None,\n    timeout: Optional[float] = None,\n    poll_interval: Optional[float] = 5,\n    tags: Optional[Iterable[str]] = None,\n    idempotency_key: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n):\n    \"\"\"\n    Create a flow run for a deployment and return it after completion or a timeout.\n\n    This function will return when the created flow run enters any terminal state or\n    the timeout is reached. If the timeout is reached and the flow run has not reached\n    a terminal state, it will still be returned. When using a timeout, we suggest\n    checking the state of the flow run if completion is important moving forward.\n\n    Args:\n        name: The deployment id or deployment name in the form:\n            `<slugified-flow-name>/<slugified-deployment-name>`\n        parameters: Parameter overrides for this flow run. Merged with the deployment\n            defaults.\n        scheduled_time: The time to schedule the flow run for, defaults to scheduling\n            the flow run to start now.\n        flow_run_name: A name for the created flow run\n        timeout: The amount of time to wait for the flow run to complete before\n            returning. Setting `timeout` to 0 will return the flow run immediately.\n            Setting `timeout` to None will allow this function to poll indefinitely.\n            Defaults to None\n        poll_interval: The number of seconds between polls\n        tags: A list of tags to associate with this flow run; note that tags are used\n            only for organizational purposes.\n        idempotency_key: A unique value to recognize retries of the same run, and\n            prevent creating multiple flow runs.\n        work_queue_name: The name of a work queue to use for this run. Defaults to\n            the default work queue for the deployment.\n    \"\"\"\n    if timeout is not None and timeout < 0:\n        raise ValueError(\"`timeout` cannot be negative\")\n\n    if scheduled_time is None:\n        scheduled_time = pendulum.now(\"UTC\")\n\n    parameters = parameters or {}\n\n    deployment_id = None\n\n    if isinstance(name, UUID):\n        deployment_id = name\n    else:\n        try:\n            deployment_id = UUID(name)\n        except ValueError:\n            pass\n\n    if deployment_id:\n        deployment = await client.read_deployment(deployment_id=deployment_id)\n    else:\n        deployment = await client.read_deployment_by_name(name)\n\n    flow_run_ctx = FlowRunContext.get()\n    task_run_ctx = TaskRunContext.get()\n    if flow_run_ctx or task_run_ctx:\n        # This was called from a flow. Link the flow run as a subflow.\n        from prefect.engine import (\n            Pending,\n            _dynamic_key_for_task_run,\n            collect_task_run_inputs,\n        )\n\n        task_inputs = {\n            k: await collect_task_run_inputs(v) for k, v in parameters.items()\n        }\n\n        if deployment_id:\n            flow = await client.read_flow(deployment.flow_id)\n            deployment_name = f\"{flow.name}/{deployment.name}\"\n        else:\n            deployment_name = name\n\n        # Generate a task in the parent flow run to represent the result of the subflow\n        dummy_task = Task(\n            name=deployment_name,\n            fn=lambda: None,\n            version=deployment.version,\n        )\n        # Override the default task key to include the deployment name\n        dummy_task.task_key = f\"{__name__}.run_deployment.{slugify(deployment_name)}\"\n        flow_run_id = (\n            flow_run_ctx.flow_run.id\n            if flow_run_ctx\n            else task_run_ctx.task_run.flow_run_id\n        )\n        dynamic_key = (\n            _dynamic_key_for_task_run(flow_run_ctx, dummy_task)\n            if flow_run_ctx\n            else task_run_ctx.task_run.dynamic_key\n        )\n        parent_task_run = await client.create_task_run(\n            task=dummy_task,\n            flow_run_id=flow_run_id,\n            dynamic_key=dynamic_key,\n            task_inputs=task_inputs,\n            state=Pending(),\n        )\n        parent_task_run_id = parent_task_run.id\n    else:\n        parent_task_run_id = None\n\n    flow_run = await client.create_flow_run_from_deployment(\n        deployment.id,\n        parameters=parameters,\n        state=Scheduled(scheduled_time=scheduled_time),\n        name=flow_run_name,\n        tags=tags,\n        idempotency_key=idempotency_key,\n        parent_task_run_id=parent_task_run_id,\n        work_queue_name=work_queue_name,\n    )\n\n    flow_run_id = flow_run.id\n\n    if timeout == 0:\n        return flow_run\n\n    with anyio.move_on_after(timeout):\n        while True:\n            flow_run = await client.read_flow_run(flow_run_id)\n            flow_state = flow_run.state\n            if flow_state and flow_state.is_final():\n                return flow_run\n            await anyio.sleep(poll_interval)\n\n    return flow_run\n
    ","tags":["Python API","flow runs","deployments"]},{"location":"api-ref/prefect/deployments/runner/","title":"runner","text":"","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner","title":"prefect.deployments.runner","text":"

    Objects for creating and configuring deployments for flows using serve functionality.

    Example
    import time\nfrom prefect import flow, serve\n\n\n@flow\ndef slow_flow(sleep: int = 60):\n    \"Sleepy flow - sleeps the provided amount of time (in seconds).\"\n    time.sleep(sleep)\n\n\n@flow\ndef fast_flow():\n    \"Fastest flow this side of the Mississippi.\"\n    return\n\n\nif __name__ == \"__main__\":\n    # to_deployment creates RunnerDeployment instances\n    slow_deploy = slow_flow.to_deployment(name=\"sleeper\", interval=45)\n    fast_deploy = fast_flow.to_deployment(name=\"fast\")\n\n    serve(slow_deploy, fast_deploy)\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.DeploymentApplyError","title":"DeploymentApplyError","text":"

    Bases: RuntimeError

    Raised when an error occurs while applying a deployment.

    Source code in prefect/deployments/runner.py
    class DeploymentApplyError(RuntimeError):\n    \"\"\"\n    Raised when an error occurs while applying a deployment.\n    \"\"\"\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.DeploymentImage","title":"DeploymentImage","text":"

    Configuration used to build and push a Docker image for a deployment.

    Attributes:

    Name Type Description name

    The name of the Docker image to build, including the registry and repository.

    tag

    The tag to apply to the built image.

    dockerfile

    The path to the Dockerfile to use for building the image. If not provided, a default Dockerfile will be generated.

    **build_kwargs

    Additional keyword arguments to pass to the Docker build request. See the docker-py documentation for more information.

    Source code in prefect/deployments/runner.py
    class DeploymentImage:\n    \"\"\"\n    Configuration used to build and push a Docker image for a deployment.\n\n    Attributes:\n        name: The name of the Docker image to build, including the registry and\n            repository.\n        tag: The tag to apply to the built image.\n        dockerfile: The path to the Dockerfile to use for building the image. If\n            not provided, a default Dockerfile will be generated.\n        **build_kwargs: Additional keyword arguments to pass to the Docker build request.\n            See the [`docker-py` documentation](https://docker-py.readthedocs.io/en/stable/images.html#docker.models.images.ImageCollection.build)\n            for more information.\n\n    \"\"\"\n\n    def __init__(self, name, tag=None, dockerfile=\"auto\", **build_kwargs):\n        image_name, image_tag = parse_image_tag(name)\n        if tag and image_tag:\n            raise ValueError(\n                f\"Only one tag can be provided - both {image_tag!r} and {tag!r} were\"\n                \" provided as tags.\"\n            )\n        namespace, repository = split_repository_path(image_name)\n        # if the provided image name does not include a namespace (registry URL or user/org name),\n        # use the default namespace\n        if not namespace:\n            namespace = PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE.value()\n        # join the namespace and repository to create the full image name\n        # ignore namespace if it is None\n        self.name = \"/\".join(filter(None, [namespace, repository]))\n        self.tag = tag or image_tag or slugify(pendulum.now(\"utc\").isoformat())\n        self.dockerfile = dockerfile\n        self.build_kwargs = build_kwargs\n\n    @property\n    def reference(self):\n        return f\"{self.name}:{self.tag}\"\n\n    def build(self):\n        full_image_name = self.reference\n        build_kwargs = self.build_kwargs.copy()\n        build_kwargs[\"context\"] = Path.cwd()\n        build_kwargs[\"tag\"] = full_image_name\n        build_kwargs[\"pull\"] = build_kwargs.get(\"pull\", True)\n\n        if self.dockerfile == \"auto\":\n            with generate_default_dockerfile():\n                build_image(**build_kwargs)\n        else:\n            build_kwargs[\"dockerfile\"] = self.dockerfile\n            build_image(**build_kwargs)\n\n    def push(self):\n        with docker_client() as client:\n            events = client.api.push(\n                repository=self.name, tag=self.tag, stream=True, decode=True\n            )\n            for event in events:\n                if \"error\" in event:\n                    raise PushError(event[\"error\"])\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment","title":"RunnerDeployment","text":"

    Bases: BaseModel

    A Prefect RunnerDeployment definition, used for specifying and building deployments.

    Attributes:

    Name Type Description name str

    A name for the deployment (required).

    version Optional[str]

    An optional version for the deployment; defaults to the flow's version

    description Optional[str]

    An optional description of the deployment; defaults to the flow's description

    tags List[str]

    An optional list of tags to associate with this deployment; note that tags are used only for organizational purposes. For delegating work to agents, see work_queue_name.

    schedule Optional[SCHEDULE_TYPES]

    A schedule to run this deployment on, once registered

    is_schedule_active Optional[bool]

    Whether or not the schedule is active

    parameters Dict[str, Any]

    A dictionary of parameter values to pass to runs created from this deployment

    path Dict[str, Any]

    The path to the working directory for the workflow, relative to remote storage or, if stored on a local filesystem, an absolute path

    entrypoint Optional[str]

    The path to the entrypoint for the workflow, always relative to the path

    parameter_openapi_schema Optional[str]

    The parameter schema of the flow, including defaults.

    enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    job_variables Dict[str, Any]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    Source code in prefect/deployments/runner.py
    class RunnerDeployment(BaseModel):\n    \"\"\"\n    A Prefect RunnerDeployment definition, used for specifying and building deployments.\n\n    Attributes:\n        name: A name for the deployment (required).\n        version: An optional version for the deployment; defaults to the flow's version\n        description: An optional description of the deployment; defaults to the flow's\n            description\n        tags: An optional list of tags to associate with this deployment; note that tags\n            are used only for organizational purposes. For delegating work to agents,\n            see `work_queue_name`.\n        schedule: A schedule to run this deployment on, once registered\n        is_schedule_active: Whether or not the schedule is active\n        parameters: A dictionary of parameter values to pass to runs created from this\n            deployment\n        path: The path to the working directory for the workflow, relative to remote\n            storage or, if stored on a local filesystem, an absolute path\n        entrypoint: The path to the entrypoint for the workflow, always relative to the\n            `path`\n        parameter_openapi_schema: The parameter schema of the flow, including defaults.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n    \"\"\"\n\n    class Config:\n        arbitrary_types_allowed = True\n\n    name: str = Field(..., description=\"The name of the deployment.\")\n    flow_name: Optional[str] = Field(\n        None, description=\"The name of the underlying flow; typically inferred.\"\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"An optional description of the deployment.\"\n    )\n    version: Optional[str] = Field(\n        default=None, description=\"An optional version for the deployment.\"\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"One of more tags to apply to this deployment.\",\n    )\n    schedule: Optional[SCHEDULE_TYPES] = None\n    is_schedule_active: Optional[bool] = Field(\n        default=None, description=\"Whether or not the schedule is active.\"\n    )\n    parameters: Dict[str, Any] = Field(default_factory=dict)\n    entrypoint: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the entrypoint for the workflow, relative to the `path`.\"\n        ),\n    )\n    triggers: List[DeploymentTrigger] = Field(\n        default_factory=list,\n        description=\"The triggers that should cause this deployment to run.\",\n    )\n    enforce_parameter_schema: bool = Field(\n        default=False,\n        description=(\n            \"Whether or not the Prefect API should enforce the parameter schema for\"\n            \" this deployment.\"\n        ),\n    )\n    storage: Optional[RunnerStorage] = Field(\n        default=None,\n        description=(\n            \"The storage object used to retrieve flow code for this deployment.\"\n        ),\n    )\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The name of the work pool to use for this deployment. Only used when\"\n            \" the deployment is registered with a built runner.\"\n        ),\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The name of the work queue to use for this deployment. Only used when\"\n            \" the deployment is registered with a built runner.\"\n        ),\n    )\n    job_variables: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=(\n            \"Job variables used to override the default values of a work pool\"\n            \" base job template. Only used when the deployment is registered with\"\n            \" a built runner.\"\n        ),\n    )\n    _path: Optional[str] = PrivateAttr(\n        default=None,\n    )\n    _parameter_openapi_schema: ParameterSchema = PrivateAttr(\n        default_factory=ParameterSchema,\n    )\n\n    @validator(\"triggers\", allow_reuse=True)\n    def validate_automation_names(cls, field_value, values, field, config):\n        \"\"\"Ensure that each trigger has a name for its automation if none is provided.\"\"\"\n        for i, trigger in enumerate(field_value, start=1):\n            if trigger.name is None:\n                trigger.name = f\"{values['name']}__automation_{i}\"\n\n        return field_value\n\n    @sync_compatible\n    async def apply(\n        self, work_pool_name: Optional[str] = None, image: Optional[str] = None\n    ) -> UUID:\n        \"\"\"\n        Registers this deployment with the API and returns the deployment's ID.\n\n        Args:\n            work_pool_name: The name of the work pool to use for this\n                deployment.\n            image: The registry, name, and tag of the Docker image to\n                use for this deployment. Only used when the deployment is\n                deployed to a work pool.\n\n        Returns:\n            The ID of the created deployment.\n        \"\"\"\n        work_pool_name = work_pool_name or self.work_pool_name\n\n        if image and not work_pool_name:\n            raise ValueError(\n                \"An image can only be provided when registering a deployment with a\"\n                \" work pool.\"\n            )\n\n        if self.work_queue_name and not work_pool_name:\n            raise ValueError(\n                \"A work queue can only be provided when registering a deployment with\"\n                \" a work pool.\"\n            )\n\n        if self.job_variables and not work_pool_name:\n            raise ValueError(\n                \"Job variables can only be provided when registering a deployment\"\n                \" with a work pool.\"\n            )\n\n        async with get_client() as client:\n            flow_id = await client.create_flow_from_name(self.flow_name)\n\n            create_payload = dict(\n                flow_id=flow_id,\n                name=self.name,\n                work_queue_name=self.work_queue_name,\n                work_pool_name=work_pool_name,\n                version=self.version,\n                schedule=self.schedule,\n                is_schedule_active=self.is_schedule_active,\n                parameters=self.parameters,\n                description=self.description,\n                tags=self.tags,\n                path=self._path,\n                entrypoint=self.entrypoint,\n                storage_document_id=None,\n                infrastructure_document_id=None,\n                parameter_openapi_schema=self._parameter_openapi_schema.dict(),\n                enforce_parameter_schema=self.enforce_parameter_schema,\n            )\n\n            if work_pool_name:\n                create_payload[\"infra_overrides\"] = self.job_variables\n                if image:\n                    create_payload[\"infra_overrides\"][\"image\"] = image\n                create_payload[\"path\"] = None if self.storage else self._path\n                create_payload[\"pull_steps\"] = (\n                    [self.storage.to_pull_step()] if self.storage else []\n                )\n\n            try:\n                deployment_id = await client.create_deployment(**create_payload)\n            except Exception as exc:\n                if isinstance(exc, PrefectHTTPStatusError):\n                    detail = exc.response.json().get(\"detail\")\n                    if detail:\n                        raise DeploymentApplyError(detail) from exc\n                raise DeploymentApplyError(\n                    f\"Error while applying deployment: {str(exc)}\"\n                ) from exc\n\n            if client.server_type == ServerType.CLOUD:\n                # The triggers defined in the deployment spec are, essentially,\n                # anonymous and attempting truly sync them with cloud is not\n                # feasible. Instead, we remove all automations that are owned\n                # by the deployment, meaning that they were created via this\n                # mechanism below, and then recreate them.\n                await client.delete_resource_owned_automations(\n                    f\"prefect.deployment.{deployment_id}\"\n                )\n                for trigger in self.triggers:\n                    trigger.set_deployment_id(deployment_id)\n                    await client.create_automation(trigger.as_automation())\n\n            return deployment_id\n\n    @staticmethod\n    def _construct_schedule(\n        interval: Optional[Union[int, float, timedelta]] = None,\n        anchor_date: Optional[Union[datetime, str]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        timezone: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n    ) -> Optional[SCHEDULE_TYPES]:\n        \"\"\"\n        Construct a schedule from the provided arguments.\n\n        This is a single path for all serve schedules. If schedule is provided,\n        it is returned. Otherwise, the other arguments are used to construct a schedule.\n\n        Args:\n            interval: An interval on which to schedule runs. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            anchor_date: The start date for an interval schedule.\n            cron: A cron schedule for runs.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            timezone: A timezone to use for the schedule.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n        \"\"\"\n        num_schedules = sum(\n            1 for entry in (interval, cron, rrule, schedule) if entry is not None\n        )\n        if num_schedules > 1:\n            raise ValueError(\n                \"Only one of interval, cron, rrule, or schedule can be provided.\"\n            )\n\n        if schedule:\n            return schedule\n        elif interval or cron or rrule:\n            return construct_schedule(\n                interval=interval,\n                cron=cron,\n                rrule=rrule,\n                timezone=timezone,\n                anchor_date=anchor_date,\n            )\n        else:\n            return None\n\n    def _set_defaults_from_flow(self, flow: \"Flow\"):\n        self._parameter_openapi_schema = parameter_schema(flow)\n\n        if not self.version:\n            self.version = flow.version\n        if not self.description:\n            self.description = flow.description\n\n    @classmethod\n    def from_flow(\n        cls,\n        flow: \"Flow\",\n        name: str,\n        interval: Optional[Union[int, float, timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        work_pool_name: Optional[str] = None,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[Dict[str, Any]] = None,\n    ) -> \"RunnerDeployment\":\n        \"\"\"\n        Configure a deployment for a given flow.\n\n        Args:\n            flow: A flow function to deploy\n            name: A name for the deployment\n            interval: An interval on which to execute the current flow. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this flow.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            triggers: A list of triggers that should kick of a run of this flow.\n            parameters: A dictionary of default parameter values to pass to runs of this flow.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for this deployment.\n            work_pool_name: The name of the work pool to use for this deployment.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n                available settings.\n        \"\"\"\n        schedule = cls._construct_schedule(\n            interval=interval, cron=cron, rrule=rrule, schedule=schedule\n        )\n\n        job_variables = job_variables or {}\n\n        deployment = cls(\n            name=Path(name).stem,\n            flow_name=flow.name,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags or [],\n            triggers=triggers or [],\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n\n        if not deployment.entrypoint:\n            no_file_location_error = (\n                \"Flows defined interactively cannot be deployed. Check out the\"\n                \" quickstart guide for help getting started:\"\n                \" https://docs.prefect.io/latest/getting-started/quickstart\"\n            )\n            ## first see if an entrypoint can be determined\n            flow_file = getattr(flow, \"__globals__\", {}).get(\"__file__\")\n            mod_name = getattr(flow, \"__module__\", None)\n            if not flow_file:\n                if not mod_name:\n                    raise ValueError(no_file_location_error)\n                try:\n                    module = importlib.import_module(mod_name)\n                    flow_file = getattr(module, \"__file__\", None)\n                except ModuleNotFoundError as exc:\n                    if \"__prefect_loader__\" in str(exc):\n                        raise ValueError(\n                            \"Cannot create a RunnerDeployment from a flow that has been\"\n                            \" loaded from an entrypoint. To deploy a flow via\"\n                            \" entrypoint, use RunnerDeployment.from_entrypoint instead.\"\n                        )\n                    raise ValueError(no_file_location_error)\n                if not flow_file:\n                    raise ValueError(no_file_location_error)\n\n            # set entrypoint\n            entry_path = Path(flow_file).absolute().relative_to(Path.cwd().absolute())\n            deployment.entrypoint = f\"{entry_path}:{flow.fn.__name__}\"\n\n        if not deployment._path:\n            deployment._path = \".\"\n\n        cls._set_defaults_from_flow(deployment, flow)\n\n        return deployment\n\n    @classmethod\n    def from_entrypoint(\n        cls,\n        entrypoint: str,\n        name: str,\n        interval: Optional[Union[int, float, timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        work_pool_name: Optional[str] = None,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[Dict[str, Any]] = None,\n    ) -> \"RunnerDeployment\":\n        \"\"\"\n        Configure a deployment for a given flow located at a given entrypoint.\n\n        Args:\n            entrypoint:  The path to a file containing a flow and the name of the flow function in\n                the format `./path/to/file.py:flow_func_name`.\n            name: A name for the deployment\n            interval: An interval on which to execute the current flow. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this flow.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            triggers: A list of triggers that should kick of a run of this flow.\n            parameters: A dictionary of default parameter values to pass to runs of this flow.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for this deployment.\n            work_pool_name: The name of the work pool to use for this deployment.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n                available settings.\n        \"\"\"\n        from prefect.flows import load_flow_from_entrypoint\n\n        job_variables = job_variables or {}\n        flow = load_flow_from_entrypoint(entrypoint)\n\n        schedule = cls._construct_schedule(\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n        )\n\n        deployment = cls(\n            name=Path(name).stem,\n            flow_name=flow.name,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags or [],\n            triggers=triggers or [],\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            entrypoint=entrypoint,\n            enforce_parameter_schema=enforce_parameter_schema,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n        deployment._path = str(Path.cwd())\n\n        cls._set_defaults_from_flow(deployment, flow)\n\n        return deployment\n\n    @classmethod\n    @sync_compatible\n    async def from_storage(\n        cls,\n        storage: RunnerStorage,\n        entrypoint: str,\n        name: str,\n        interval: Optional[Union[int, float, timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n        work_pool_name: Optional[str] = None,\n        work_queue_name: Optional[str] = None,\n        job_variables: Optional[Dict[str, Any]] = None,\n    ):\n        \"\"\"\n        Create a RunnerDeployment from a flow located at a given entrypoint and stored in a\n        local storage location.\n\n        Args:\n            entrypoint:  The path to a file containing a flow and the name of the flow function in\n                the format `./path/to/file.py:flow_func_name`.\n            name: A name for the deployment\n            storage: A storage object to use for retrieving flow code. If not provided, a\n                URL must be provided.\n            interval: An interval on which to execute the current flow. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this flow.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            triggers: A list of triggers that should kick of a run of this flow.\n            parameters: A dictionary of default parameter values to pass to runs of this flow.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n            enforce_parameter_schema: Whether or not the Prefect API should enforce the\n                parameter schema for this deployment.\n            work_pool_name: The name of the work pool to use for this deployment.\n            work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n                If not provided the default work queue for the work pool will be used.\n            job_variables: Settings used to override the values specified default base job template\n                of the chosen work pool. Refer to the base job template of the chosen work pool for\n                available settings.\n        \"\"\"\n        from prefect.flows import load_flow_from_entrypoint\n\n        schedule = cls._construct_schedule(\n            interval=interval, cron=cron, rrule=rrule, schedule=schedule\n        )\n        job_variables = job_variables or {}\n\n        with tempfile.TemporaryDirectory() as tmpdir:\n            storage.set_base_path(Path(tmpdir))\n            await storage.pull_code()\n\n            full_entrypoint = str(storage.destination / entrypoint)\n            flow = await from_async.wait_for_call_in_new_thread(\n                create_call(load_flow_from_entrypoint, full_entrypoint)\n            )\n\n        deployment = cls(\n            name=Path(name).stem,\n            flow_name=flow.name,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            tags=tags or [],\n            triggers=triggers or [],\n            parameters=parameters or {},\n            description=description,\n            version=version,\n            entrypoint=entrypoint,\n            enforce_parameter_schema=enforce_parameter_schema,\n            storage=storage,\n            work_pool_name=work_pool_name,\n            work_queue_name=work_queue_name,\n            job_variables=job_variables,\n        )\n        deployment._path = str(storage.destination).replace(\n            tmpdir, \"$STORAGE_BASE_PATH\"\n        )\n\n        cls._set_defaults_from_flow(deployment, flow)\n\n        return deployment\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.apply","title":"apply async","text":"

    Registers this deployment with the API and returns the deployment's ID.

    Parameters:

    Name Type Description Default work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None image Optional[str]

    The registry, name, and tag of the Docker image to use for this deployment. Only used when the deployment is deployed to a work pool.

    None

    Returns:

    Type Description UUID

    The ID of the created deployment.

    Source code in prefect/deployments/runner.py
    @sync_compatible\nasync def apply(\n    self, work_pool_name: Optional[str] = None, image: Optional[str] = None\n) -> UUID:\n    \"\"\"\n    Registers this deployment with the API and returns the deployment's ID.\n\n    Args:\n        work_pool_name: The name of the work pool to use for this\n            deployment.\n        image: The registry, name, and tag of the Docker image to\n            use for this deployment. Only used when the deployment is\n            deployed to a work pool.\n\n    Returns:\n        The ID of the created deployment.\n    \"\"\"\n    work_pool_name = work_pool_name or self.work_pool_name\n\n    if image and not work_pool_name:\n        raise ValueError(\n            \"An image can only be provided when registering a deployment with a\"\n            \" work pool.\"\n        )\n\n    if self.work_queue_name and not work_pool_name:\n        raise ValueError(\n            \"A work queue can only be provided when registering a deployment with\"\n            \" a work pool.\"\n        )\n\n    if self.job_variables and not work_pool_name:\n        raise ValueError(\n            \"Job variables can only be provided when registering a deployment\"\n            \" with a work pool.\"\n        )\n\n    async with get_client() as client:\n        flow_id = await client.create_flow_from_name(self.flow_name)\n\n        create_payload = dict(\n            flow_id=flow_id,\n            name=self.name,\n            work_queue_name=self.work_queue_name,\n            work_pool_name=work_pool_name,\n            version=self.version,\n            schedule=self.schedule,\n            is_schedule_active=self.is_schedule_active,\n            parameters=self.parameters,\n            description=self.description,\n            tags=self.tags,\n            path=self._path,\n            entrypoint=self.entrypoint,\n            storage_document_id=None,\n            infrastructure_document_id=None,\n            parameter_openapi_schema=self._parameter_openapi_schema.dict(),\n            enforce_parameter_schema=self.enforce_parameter_schema,\n        )\n\n        if work_pool_name:\n            create_payload[\"infra_overrides\"] = self.job_variables\n            if image:\n                create_payload[\"infra_overrides\"][\"image\"] = image\n            create_payload[\"path\"] = None if self.storage else self._path\n            create_payload[\"pull_steps\"] = (\n                [self.storage.to_pull_step()] if self.storage else []\n            )\n\n        try:\n            deployment_id = await client.create_deployment(**create_payload)\n        except Exception as exc:\n            if isinstance(exc, PrefectHTTPStatusError):\n                detail = exc.response.json().get(\"detail\")\n                if detail:\n                    raise DeploymentApplyError(detail) from exc\n            raise DeploymentApplyError(\n                f\"Error while applying deployment: {str(exc)}\"\n            ) from exc\n\n        if client.server_type == ServerType.CLOUD:\n            # The triggers defined in the deployment spec are, essentially,\n            # anonymous and attempting truly sync them with cloud is not\n            # feasible. Instead, we remove all automations that are owned\n            # by the deployment, meaning that they were created via this\n            # mechanism below, and then recreate them.\n            await client.delete_resource_owned_automations(\n                f\"prefect.deployment.{deployment_id}\"\n            )\n            for trigger in self.triggers:\n                trigger.set_deployment_id(deployment_id)\n                await client.create_automation(trigger.as_automation())\n\n        return deployment_id\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.from_entrypoint","title":"from_entrypoint classmethod","text":"

    Configure a deployment for a given flow located at a given entrypoint.

    Parameters:

    Name Type Description Default entrypoint str

    The path to a file containing a flow and the name of the flow function in the format ./path/to/file.py:flow_func_name.

    required name str

    A name for the deployment

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the current flow. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this flow.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object of when to execute runs of this flow. Used for advanced scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that should kick of a run of this flow.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this flow.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    False work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[Dict[str, Any]]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    None Source code in prefect/deployments/runner.py
    @classmethod\ndef from_entrypoint(\n    cls,\n    entrypoint: str,\n    name: str,\n    interval: Optional[Union[int, float, timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    work_pool_name: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[Dict[str, Any]] = None,\n) -> \"RunnerDeployment\":\n    \"\"\"\n    Configure a deployment for a given flow located at a given entrypoint.\n\n    Args:\n        entrypoint:  The path to a file containing a flow and the name of the flow function in\n            the format `./path/to/file.py:flow_func_name`.\n        name: A name for the deployment\n        interval: An interval on which to execute the current flow. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this flow.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        schedule: A schedule object of when to execute runs of this flow. Used for\n            advanced scheduling options like timezone.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        triggers: A list of triggers that should kick of a run of this flow.\n        parameters: A dictionary of default parameter values to pass to runs of this flow.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n    \"\"\"\n    from prefect.flows import load_flow_from_entrypoint\n\n    job_variables = job_variables or {}\n    flow = load_flow_from_entrypoint(entrypoint)\n\n    schedule = cls._construct_schedule(\n        interval=interval,\n        cron=cron,\n        rrule=rrule,\n        schedule=schedule,\n    )\n\n    deployment = cls(\n        name=Path(name).stem,\n        flow_name=flow.name,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        tags=tags or [],\n        triggers=triggers or [],\n        parameters=parameters or {},\n        description=description,\n        version=version,\n        entrypoint=entrypoint,\n        enforce_parameter_schema=enforce_parameter_schema,\n        work_pool_name=work_pool_name,\n        work_queue_name=work_queue_name,\n        job_variables=job_variables,\n    )\n    deployment._path = str(Path.cwd())\n\n    cls._set_defaults_from_flow(deployment, flow)\n\n    return deployment\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.from_flow","title":"from_flow classmethod","text":"

    Configure a deployment for a given flow.

    Parameters:

    Name Type Description Default flow Flow

    A flow function to deploy

    required name str

    A name for the deployment

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the current flow. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this flow.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object of when to execute runs of this flow. Used for advanced scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that should kick of a run of this flow.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this flow.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    False work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[Dict[str, Any]]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    None Source code in prefect/deployments/runner.py
    @classmethod\ndef from_flow(\n    cls,\n    flow: \"Flow\",\n    name: str,\n    interval: Optional[Union[int, float, timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    work_pool_name: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[Dict[str, Any]] = None,\n) -> \"RunnerDeployment\":\n    \"\"\"\n    Configure a deployment for a given flow.\n\n    Args:\n        flow: A flow function to deploy\n        name: A name for the deployment\n        interval: An interval on which to execute the current flow. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this flow.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        schedule: A schedule object of when to execute runs of this flow. Used for\n            advanced scheduling options like timezone.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        triggers: A list of triggers that should kick of a run of this flow.\n        parameters: A dictionary of default parameter values to pass to runs of this flow.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n    \"\"\"\n    schedule = cls._construct_schedule(\n        interval=interval, cron=cron, rrule=rrule, schedule=schedule\n    )\n\n    job_variables = job_variables or {}\n\n    deployment = cls(\n        name=Path(name).stem,\n        flow_name=flow.name,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        tags=tags or [],\n        triggers=triggers or [],\n        parameters=parameters or {},\n        description=description,\n        version=version,\n        enforce_parameter_schema=enforce_parameter_schema,\n        work_pool_name=work_pool_name,\n        work_queue_name=work_queue_name,\n        job_variables=job_variables,\n    )\n\n    if not deployment.entrypoint:\n        no_file_location_error = (\n            \"Flows defined interactively cannot be deployed. Check out the\"\n            \" quickstart guide for help getting started:\"\n            \" https://docs.prefect.io/latest/getting-started/quickstart\"\n        )\n        ## first see if an entrypoint can be determined\n        flow_file = getattr(flow, \"__globals__\", {}).get(\"__file__\")\n        mod_name = getattr(flow, \"__module__\", None)\n        if not flow_file:\n            if not mod_name:\n                raise ValueError(no_file_location_error)\n            try:\n                module = importlib.import_module(mod_name)\n                flow_file = getattr(module, \"__file__\", None)\n            except ModuleNotFoundError as exc:\n                if \"__prefect_loader__\" in str(exc):\n                    raise ValueError(\n                        \"Cannot create a RunnerDeployment from a flow that has been\"\n                        \" loaded from an entrypoint. To deploy a flow via\"\n                        \" entrypoint, use RunnerDeployment.from_entrypoint instead.\"\n                    )\n                raise ValueError(no_file_location_error)\n            if not flow_file:\n                raise ValueError(no_file_location_error)\n\n        # set entrypoint\n        entry_path = Path(flow_file).absolute().relative_to(Path.cwd().absolute())\n        deployment.entrypoint = f\"{entry_path}:{flow.fn.__name__}\"\n\n    if not deployment._path:\n        deployment._path = \".\"\n\n    cls._set_defaults_from_flow(deployment, flow)\n\n    return deployment\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.from_storage","title":"from_storage async classmethod","text":"

    Create a RunnerDeployment from a flow located at a given entrypoint and stored in a local storage location.

    Parameters:

    Name Type Description Default entrypoint str

    The path to a file containing a flow and the name of the flow function in the format ./path/to/file.py:flow_func_name.

    required name str

    A name for the deployment

    required storage RunnerStorage

    A storage object to use for retrieving flow code. If not provided, a URL must be provided.

    required interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the current flow. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this flow.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object of when to execute runs of this flow. Used for advanced scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that should kick of a run of this flow.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this flow.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None enforce_parameter_schema bool

    Whether or not the Prefect API should enforce the parameter schema for this deployment.

    False work_pool_name Optional[str]

    The name of the work pool to use for this deployment.

    None work_queue_name Optional[str]

    The name of the work queue to use for this deployment's scheduled runs. If not provided the default work queue for the work pool will be used.

    None job_variables Optional[Dict[str, Any]]

    Settings used to override the values specified default base job template of the chosen work pool. Refer to the base job template of the chosen work pool for available settings.

    None Source code in prefect/deployments/runner.py
    @classmethod\n@sync_compatible\nasync def from_storage(\n    cls,\n    storage: RunnerStorage,\n    entrypoint: str,\n    name: str,\n    interval: Optional[Union[int, float, timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n    work_pool_name: Optional[str] = None,\n    work_queue_name: Optional[str] = None,\n    job_variables: Optional[Dict[str, Any]] = None,\n):\n    \"\"\"\n    Create a RunnerDeployment from a flow located at a given entrypoint and stored in a\n    local storage location.\n\n    Args:\n        entrypoint:  The path to a file containing a flow and the name of the flow function in\n            the format `./path/to/file.py:flow_func_name`.\n        name: A name for the deployment\n        storage: A storage object to use for retrieving flow code. If not provided, a\n            URL must be provided.\n        interval: An interval on which to execute the current flow. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this flow.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        schedule: A schedule object of when to execute runs of this flow. Used for\n            advanced scheduling options like timezone.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        triggers: A list of triggers that should kick of a run of this flow.\n        parameters: A dictionary of default parameter values to pass to runs of this flow.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n        enforce_parameter_schema: Whether or not the Prefect API should enforce the\n            parameter schema for this deployment.\n        work_pool_name: The name of the work pool to use for this deployment.\n        work_queue_name: The name of the work queue to use for this deployment's scheduled runs.\n            If not provided the default work queue for the work pool will be used.\n        job_variables: Settings used to override the values specified default base job template\n            of the chosen work pool. Refer to the base job template of the chosen work pool for\n            available settings.\n    \"\"\"\n    from prefect.flows import load_flow_from_entrypoint\n\n    schedule = cls._construct_schedule(\n        interval=interval, cron=cron, rrule=rrule, schedule=schedule\n    )\n    job_variables = job_variables or {}\n\n    with tempfile.TemporaryDirectory() as tmpdir:\n        storage.set_base_path(Path(tmpdir))\n        await storage.pull_code()\n\n        full_entrypoint = str(storage.destination / entrypoint)\n        flow = await from_async.wait_for_call_in_new_thread(\n            create_call(load_flow_from_entrypoint, full_entrypoint)\n        )\n\n    deployment = cls(\n        name=Path(name).stem,\n        flow_name=flow.name,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        tags=tags or [],\n        triggers=triggers or [],\n        parameters=parameters or {},\n        description=description,\n        version=version,\n        entrypoint=entrypoint,\n        enforce_parameter_schema=enforce_parameter_schema,\n        storage=storage,\n        work_pool_name=work_pool_name,\n        work_queue_name=work_queue_name,\n        job_variables=job_variables,\n    )\n    deployment._path = str(storage.destination).replace(\n        tmpdir, \"$STORAGE_BASE_PATH\"\n    )\n\n    cls._set_defaults_from_flow(deployment, flow)\n\n    return deployment\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.RunnerDeployment.validate_automation_names","title":"validate_automation_names","text":"

    Ensure that each trigger has a name for its automation if none is provided.

    Source code in prefect/deployments/runner.py
    @validator(\"triggers\", allow_reuse=True)\ndef validate_automation_names(cls, field_value, values, field, config):\n    \"\"\"Ensure that each trigger has a name for its automation if none is provided.\"\"\"\n    for i, trigger in enumerate(field_value, start=1):\n        if trigger.name is None:\n            trigger.name = f\"{values['name']}__automation_{i}\"\n\n    return field_value\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/runner/#prefect.deployments.runner.deploy","title":"deploy async","text":"

    Deploy the provided list of deployments to dynamic infrastructure via a work pool.

    By default, calling this function will build a Docker image for the deployments, push it to a registry, and create each deployment via the Prefect API that will run the corresponding flow on the given schedule.

    If you want to use an existing image, you can pass build=False to skip building and pushing an image.

    Parameters:

    Name Type Description Default *deployments RunnerDeployment

    A list of deployments to deploy.

    () work_pool_name Optional[str]

    The name of the work pool to use for these deployments. Defaults to the value of PREFECT_DEFAULT_WORK_POOL_NAME.

    None image Optional[Union[str, DeploymentImage]]

    The name of the Docker image to build, including the registry and repository. Pass a DeploymentImage instance to customize the Dockerfile used and build arguments.

    None build bool

    Whether or not to build a new image for the flow. If False, the provided image will be used as-is and pulled at runtime.

    True push bool

    Whether or not to skip pushing the built image to a registry.

    True print_next_steps_message bool

    Whether or not to print a message with next steps after deploying the deployments.

    True

    Returns:

    Type Description List[UUID]

    A list of deployment IDs for the created/updated deployments.

    Examples:

    Deploy a group of flows to a work pool:

    from prefect import deploy, flow\n\n@flow(log_prints=True)\ndef local_flow():\n    print(\"I'm a locally defined flow!\")\n\nif __name__ == \"__main__\":\n    deploy(\n        local_flow.to_deployment(name=\"example-deploy-local-flow\"),\n        flow.from_source(\n            source=\"https://github.com/org/repo.git\",\n            entrypoint=\"flows.py:my_flow\",\n        ).to_deployment(\n            name=\"example-deploy-remote-flow\",\n        ),\n        work_pool_name=\"my-work-pool\",\n        image=\"my-registry/my-image:dev\",\n    )\n
    Source code in prefect/deployments/runner.py
    @sync_compatible\nasync def deploy(\n    *deployments: RunnerDeployment,\n    work_pool_name: Optional[str] = None,\n    image: Optional[Union[str, DeploymentImage]] = None,\n    build: bool = True,\n    push: bool = True,\n    print_next_steps_message: bool = True,\n) -> List[UUID]:\n    \"\"\"\n    Deploy the provided list of deployments to dynamic infrastructure via a\n    work pool.\n\n    By default, calling this function will build a Docker image for the deployments, push it to a\n    registry, and create each deployment via the Prefect API that will run the corresponding\n    flow on the given schedule.\n\n    If you want to use an existing image, you can pass `build=False` to skip building and pushing\n    an image.\n\n    Args:\n        *deployments: A list of deployments to deploy.\n        work_pool_name: The name of the work pool to use for these deployments. Defaults to\n            the value of `PREFECT_DEFAULT_WORK_POOL_NAME`.\n        image: The name of the Docker image to build, including the registry and\n            repository. Pass a DeploymentImage instance to customize the Dockerfile used\n            and build arguments.\n        build: Whether or not to build a new image for the flow. If False, the provided\n            image will be used as-is and pulled at runtime.\n        push: Whether or not to skip pushing the built image to a registry.\n        print_next_steps_message: Whether or not to print a message with next steps\n            after deploying the deployments.\n\n    Returns:\n        A list of deployment IDs for the created/updated deployments.\n\n    Examples:\n        Deploy a group of flows to a work pool:\n\n        ```python\n        from prefect import deploy, flow\n\n        @flow(log_prints=True)\n        def local_flow():\n            print(\"I'm a locally defined flow!\")\n\n        if __name__ == \"__main__\":\n            deploy(\n                local_flow.to_deployment(name=\"example-deploy-local-flow\"),\n                flow.from_source(\n                    source=\"https://github.com/org/repo.git\",\n                    entrypoint=\"flows.py:my_flow\",\n                ).to_deployment(\n                    name=\"example-deploy-remote-flow\",\n                ),\n                work_pool_name=\"my-work-pool\",\n                image=\"my-registry/my-image:dev\",\n            )\n        ```\n    \"\"\"\n    work_pool_name = work_pool_name or PREFECT_DEFAULT_WORK_POOL_NAME.value()\n\n    if not image and not all(d.storage for d in deployments):\n        raise ValueError(\n            \"Either an image or remote storage location must be provided when deploying\"\n            \" a deployment.\"\n        )\n\n    if not work_pool_name:\n        raise ValueError(\n            \"A work pool name must be provided when deploying a deployment. Either\"\n            \" provide a work pool name when calling `deploy` or set\"\n            \" `PREFECT_DEFAULT_WORK_POOL_NAME` in your profile.\"\n        )\n\n    if image and isinstance(image, str):\n        image_name, image_tag = parse_image_tag(image)\n        image = DeploymentImage(name=image_name, tag=image_tag)\n\n    try:\n        async with get_client() as client:\n            work_pool = await client.read_work_pool(work_pool_name)\n    except ObjectNotFound as exc:\n        raise ValueError(\n            f\"Could not find work pool {work_pool_name!r}. Please create it before\"\n            \" deploying this flow.\"\n        ) from exc\n\n    is_docker_based_work_pool = get_from_dict(\n        work_pool.base_job_template, \"variables.properties.image\", False\n    )\n    is_block_based_work_pool = get_from_dict(\n        work_pool.base_job_template, \"variables.properties.block\", False\n    )\n    # carve out an exception for block based work pools that only have a block in their base job template\n    if not is_docker_based_work_pool and not is_block_based_work_pool:\n        raise ValueError(\n            f\"Work pool {work_pool_name!r} does not support custom Docker images. \"\n            \"Please use a work pool with an `image` variable in its base job template.\"\n        )\n\n    is_managed_pool = work_pool.is_managed_pool\n    if is_managed_pool:\n        build = False\n        push = False\n\n    console = Console()\n    if image and build:\n        with Progress(\n            SpinnerColumn(),\n            TextColumn(f\"Building image {image.reference}...\"),\n            transient=True,\n            console=console,\n        ) as progress:\n            docker_build_task = progress.add_task(\"docker_build\", total=1)\n            image.build()\n\n            progress.update(docker_build_task, completed=1)\n            console.print(\n                f\"Successfully built image {image.reference!r}\", style=\"green\"\n            )\n\n    if image and build and push:\n        with Progress(\n            SpinnerColumn(),\n            TextColumn(\"Pushing image...\"),\n            transient=True,\n            console=console,\n        ) as progress:\n            docker_push_task = progress.add_task(\"docker_push\", total=1)\n\n            image.push()\n\n            progress.update(docker_push_task, completed=1)\n\n        console.print(f\"Successfully pushed image {image.reference!r}\", style=\"green\")\n\n    deployment_exceptions = []\n    deployment_ids = []\n    image_ref = image.reference if image else None\n    for deployment in track(\n        deployments,\n        description=\"Creating/updating deployments...\",\n        console=console,\n        transient=True,\n    ):\n        try:\n            deployment_ids.append(\n                await deployment.apply(image=image_ref, work_pool_name=work_pool_name)\n            )\n        except Exception as exc:\n            if len(deployments) == 1:\n                raise\n            deployment_exceptions.append({\"deployment\": deployment, \"exc\": exc})\n\n    if deployment_exceptions:\n        console.print(\n            \"Encountered errors while creating/updating deployments:\\n\",\n            style=\"orange_red1\",\n        )\n    else:\n        console.print(\"Successfully created/updated all deployments!\\n\", style=\"green\")\n\n    complete_failure = len(deployment_exceptions) == len(deployments)\n\n    table = Table(\n        title=\"Deployments\",\n        show_lines=True,\n    )\n\n    table.add_column(header=\"Name\", style=\"blue\", no_wrap=True)\n    table.add_column(header=\"Status\", style=\"blue\", no_wrap=True)\n    table.add_column(header=\"Details\", style=\"blue\")\n\n    for deployment in deployments:\n        errored_deployment = next(\n            (d for d in deployment_exceptions if d[\"deployment\"] == deployment),\n            None,\n        )\n        if errored_deployment:\n            table.add_row(\n                f\"{deployment.flow_name}/{deployment.name}\",\n                \"failed\",\n                str(errored_deployment[\"exc\"]),\n                style=\"red\",\n            )\n        else:\n            table.add_row(f\"{deployment.flow_name}/{deployment.name}\", \"applied\")\n    console.print(table)\n\n    if print_next_steps_message and not complete_failure:\n        if not work_pool.is_push_pool and not work_pool.is_managed_pool:\n            console.print(\n                \"\\nTo execute flow runs from these deployments, start a worker in a\"\n                \" separate terminal that pulls work from the\"\n                f\" {work_pool_name!r} work pool:\"\n            )\n            console.print(\n                f\"\\n\\t$ prefect worker start --pool {work_pool_name!r}\",\n                style=\"blue\",\n            )\n        console.print(\n            \"\\nTo trigger any of these deployments, use the\"\n            \" following command:\\n[blue]\\n\\t$ prefect deployment run\"\n            \" [DEPLOYMENT_NAME]\\n[/]\"\n        )\n\n        if PREFECT_UI_URL:\n            console.print(\n                \"\\nYou can also trigger your deployments via the Prefect UI:\"\n                f\" [blue]{PREFECT_UI_URL.value()}/deployments[/]\\n\"\n            )\n\n    return deployment_ids\n
    ","tags":["Python API","flow runs","deployments","runners"]},{"location":"api-ref/prefect/deployments/steps/core/","title":"steps.core","text":"","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/steps/core/#prefect.deployments.steps.core","title":"prefect.deployments.steps.core","text":"

    Core primitives for running Prefect project steps.

    Project steps are YAML representations of Python functions along with their inputs.

    Whenever a step is run, the following actions are taken:

    • The step's inputs and block / variable references are resolved (see the projects concepts documentation for more details)
    • The step's function is imported; if it cannot be found, the requires keyword is used to install the necessary packages
    • The step's function is called with the resolved inputs
    • The step's output is returned and used to resolve inputs for subsequent steps
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/steps/core/#prefect.deployments.steps.core.StepExecutionError","title":"StepExecutionError","text":"

    Bases: Exception

    Raised when a step fails to execute.

    Source code in prefect/deployments/steps/core.py
    class StepExecutionError(Exception):\n    \"\"\"\n    Raised when a step fails to execute.\n    \"\"\"\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/steps/core/#prefect.deployments.steps.core.run_step","title":"run_step async","text":"

    Runs a step, returns the step's output.

    Steps are assumed to be in the format {\"importable.func.name\": {\"kwarg1\": \"value1\", ...}}.

    The 'id and 'requires' keywords are reserved for specific purposes and will be removed from the inputs before passing to the step function:

    This keyword is used to specify packages that should be installed before running the step.

    Source code in prefect/deployments/steps/core.py
    async def run_step(step: Dict, upstream_outputs: Optional[Dict] = None) -> Dict:\n    \"\"\"\n    Runs a step, returns the step's output.\n\n    Steps are assumed to be in the format `{\"importable.func.name\": {\"kwarg1\": \"value1\", ...}}`.\n\n    The 'id and 'requires' keywords are reserved for specific purposes and will be removed from the\n    inputs before passing to the step function:\n\n    This keyword is used to specify packages that should be installed before running the step.\n    \"\"\"\n    fqn, inputs = _get_step_fully_qualified_name_and_inputs(step)\n    upstream_outputs = upstream_outputs or {}\n\n    if len(step.keys()) > 1:\n        raise ValueError(\n            f\"Step has unexpected additional keys: {', '.join(list(step.keys())[1:])}\"\n        )\n\n    keywords = {\n        keyword: inputs.pop(keyword)\n        for keyword in RESERVED_KEYWORDS\n        if keyword in inputs\n    }\n\n    inputs = apply_values(inputs, upstream_outputs)\n    inputs = await resolve_block_document_references(inputs)\n    inputs = await resolve_variables(inputs)\n    inputs = apply_values(inputs, os.environ)\n    step_func = _get_function_for_step(fqn, requires=keywords.get(\"requires\"))\n    result = await from_async.call_soon_in_new_thread(\n        Call.new(step_func, **inputs)\n    ).aresult()\n    return result\n
    ","tags":["Python API","projects","deployments","steps"]},{"location":"api-ref/prefect/deployments/steps/pull/","title":"steps.pull","text":"","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull","title":"prefect.deployments.steps.pull","text":"

    Core set of steps for specifying a Prefect project pull step.

    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.git_clone","title":"git_clone async","text":"

    Clones a git repository into the current working directory.

    Parameters:

    Name Type Description Default repository str

    the URL of the repository to clone

    required branch Optional[str]

    the branch to clone; if not provided, the default branch will be used

    None include_submodules bool

    whether to include git submodules when cloning the repository

    False access_token Optional[str]

    an access token to use for cloning the repository; if not provided the repository will be cloned using the default git credentials

    None credentials Optional[Block]

    a GitHubCredentials, GitLabCredentials, or BitBucketCredentials block can be used to specify the credentials to use for cloning the repository.

    None

    Returns:

    Name Type Description dict dict

    a dictionary containing a directory key of the new directory that was created

    Raises:

    Type Description CalledProcessError

    if the git clone command fails for any reason

    Examples:

    Clone a public repository:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/PrefectHQ/prefect.git\n

    Clone a branch of a public repository:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/PrefectHQ/prefect.git\n        branch: my-branch\n

    Clone a private repository using a GitHubCredentials block:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/org/repo.git\n        credentials: \"{{ prefect.blocks.github-credentials.my-github-credentials-block }}\"\n

    Clone a private repository using an access token:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/org/repo.git\n        access_token: \"{{ prefect.blocks.secret.github-access-token }}\" # Requires creation of a Secret block\n
    Note that you will need to create a Secret block to store the value of your git credentials. You can also store a username/password combo or token prefix (e.g. x-token-auth) in your secret block. Refer to your git providers documentation for the correct authentication schema.

    Clone a repository with submodules:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/org/repo.git\n        include_submodules: true\n

    Clone a repository with an SSH key (note that the SSH key must be added to the worker before executing flows):

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: git@github.com:org/repo.git\n

    Source code in prefect/deployments/steps/pull.py
    @sync_compatible\nasync def git_clone(\n    repository: str,\n    branch: Optional[str] = None,\n    include_submodules: bool = False,\n    access_token: Optional[str] = None,\n    credentials: Optional[Block] = None,\n) -> dict:\n    \"\"\"\n    Clones a git repository into the current working directory.\n\n    Args:\n        repository: the URL of the repository to clone\n        branch: the branch to clone; if not provided, the default branch will be used\n        include_submodules (bool): whether to include git submodules when cloning the repository\n        access_token: an access token to use for cloning the repository; if not provided\n            the repository will be cloned using the default git credentials\n        credentials: a GitHubCredentials, GitLabCredentials, or BitBucketCredentials block can be used to specify the\n            credentials to use for cloning the repository.\n\n    Returns:\n        dict: a dictionary containing a `directory` key of the new directory that was created\n\n    Raises:\n        subprocess.CalledProcessError: if the git clone command fails for any reason\n\n    Examples:\n        Clone a public repository:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/PrefectHQ/prefect.git\n        ```\n\n        Clone a branch of a public repository:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/PrefectHQ/prefect.git\n                branch: my-branch\n        ```\n\n        Clone a private repository using a GitHubCredentials block:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/org/repo.git\n                credentials: \"{{ prefect.blocks.github-credentials.my-github-credentials-block }}\"\n        ```\n\n        Clone a private repository using an access token:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/org/repo.git\n                access_token: \"{{ prefect.blocks.secret.github-access-token }}\" # Requires creation of a Secret block\n        ```\n        Note that you will need to [create a Secret block](/concepts/blocks/#using-existing-block-types) to store the\n        value of your git credentials. You can also store a username/password combo or token prefix (e.g. `x-token-auth`)\n        in your secret block. Refer to your git providers documentation for the correct authentication schema.\n\n        Clone a repository with submodules:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: https://github.com/org/repo.git\n                include_submodules: true\n        ```\n\n        Clone a repository with an SSH key (note that the SSH key must be added to the worker\n        before executing flows):\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                repository: git@github.com:org/repo.git\n        ```\n    \"\"\"\n    if access_token and credentials:\n        raise ValueError(\n            \"Please provide either an access token or credentials but not both.\"\n        )\n\n    credentials = {\"access_token\": access_token} if access_token else credentials\n\n    storage = GitRepository(\n        url=repository,\n        credentials=credentials,\n        branch=branch,\n        include_submodules=include_submodules,\n    )\n\n    await storage.pull_code()\n\n    directory = str(storage.destination.relative_to(Path.cwd()))\n    deployment_logger.info(f\"Cloned repository {repository!r} into {directory!r}\")\n    return {\"directory\": directory}\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.git_clone_project","title":"git_clone_project async","text":"

    Deprecated. Use git_clone instead.

    Source code in prefect/deployments/steps/pull.py
    @deprecated_callable(start_date=\"Jun 2023\", help=\"Use 'git clone' instead.\")\n@sync_compatible\nasync def git_clone_project(\n    repository: str,\n    branch: Optional[str] = None,\n    include_submodules: bool = False,\n    access_token: Optional[str] = None,\n) -> dict:\n    \"\"\"Deprecated. Use `git_clone` instead.\"\"\"\n    return await git_clone(\n        repository=repository,\n        branch=branch,\n        include_submodules=include_submodules,\n        access_token=access_token,\n    )\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.pull_from_remote_storage","title":"pull_from_remote_storage async","text":"

    Pulls code from a remote storage location into the current working directory.

    Works with protocols supported by fsspec.

    Parameters:

    Name Type Description Default url str

    the URL of the remote storage location. Should be a valid fsspec URL. Some protocols may require an additional fsspec dependency to be installed. Refer to the fsspec docs for more details.

    required **settings Any

    any additional settings to pass the fsspec filesystem class.

    {}

    Returns:

    Name Type Description dict

    a dictionary containing a directory key of the new directory that was created

    Examples:

    Pull code from a remote storage location:

    pull:\n    - prefect.deployments.steps.pull_from_remote_storage:\n        url: s3://my-bucket/my-folder\n

    Pull code from a remote storage location with additional settings:

    pull:\n    - prefect.deployments.steps.pull_from_remote_storage:\n        url: s3://my-bucket/my-folder\n        key: {{ prefect.blocks.secret.my-aws-access-key }}}\n        secret: {{ prefect.blocks.secret.my-aws-secret-key }}}\n

    Source code in prefect/deployments/steps/pull.py
    async def pull_from_remote_storage(url: str, **settings: Any):\n    \"\"\"\n    Pulls code from a remote storage location into the current working directory.\n\n    Works with protocols supported by `fsspec`.\n\n    Args:\n        url (str): the URL of the remote storage location. Should be a valid `fsspec` URL.\n            Some protocols may require an additional `fsspec` dependency to be installed.\n            Refer to the [`fsspec` docs](https://filesystem-spec.readthedocs.io/en/latest/api.html#other-known-implementations)\n            for more details.\n        **settings (Any): any additional settings to pass the `fsspec` filesystem class.\n\n    Returns:\n        dict: a dictionary containing a `directory` key of the new directory that was created\n\n    Examples:\n        Pull code from a remote storage location:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.pull_from_remote_storage:\n                url: s3://my-bucket/my-folder\n        ```\n\n        Pull code from a remote storage location with additional settings:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.pull_from_remote_storage:\n                url: s3://my-bucket/my-folder\n                key: {{ prefect.blocks.secret.my-aws-access-key }}}\n                secret: {{ prefect.blocks.secret.my-aws-secret-key }}}\n        ```\n    \"\"\"\n    storage = RemoteStorage(url, **settings)\n\n    await storage.pull_code()\n\n    directory = str(storage.destination.relative_to(Path.cwd()))\n    deployment_logger.info(f\"Pulled code from {url!r} into {directory!r}\")\n    return {\"directory\": directory}\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.pull_with_block","title":"pull_with_block async","text":"

    Pulls code using a block.

    Parameters:

    Name Type Description Default block_document_name str

    The name of the block document to use

    required block_type_slug str

    The slug of the type of block to use

    required Source code in prefect/deployments/steps/pull.py
    async def pull_with_block(block_document_name: str, block_type_slug: str):\n    \"\"\"\n    Pulls code using a block.\n\n    Args:\n        block_document_name: The name of the block document to use\n        block_type_slug: The slug of the type of block to use\n    \"\"\"\n    full_slug = f\"{block_type_slug}/{block_document_name}\"\n    try:\n        block = await Block.load(full_slug)\n    except Exception:\n        deployment_logger.exception(\"Unable to load block '%s'\", full_slug)\n        raise\n\n    try:\n        storage = BlockStorageAdapter(block)\n    except Exception:\n        deployment_logger.exception(\n            \"Unable to create storage adapter for block '%s'\", full_slug\n        )\n        raise\n\n    await storage.pull_code()\n\n    directory = str(storage.destination.relative_to(Path.cwd()))\n    deployment_logger.info(\n        \"Pulled code using block '%s' into '%s'\", full_slug, directory\n    )\n    return {\"directory\": directory}\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/pull/#prefect.deployments.steps.pull.set_working_directory","title":"set_working_directory","text":"

    Sets the working directory; works with both absolute and relative paths.

    Parameters:

    Name Type Description Default directory str

    the directory to set as the working directory

    required

    Returns:

    Name Type Description dict dict

    a dictionary containing a directory key of the directory that was set

    Source code in prefect/deployments/steps/pull.py
    def set_working_directory(directory: str) -> dict:\n    \"\"\"\n    Sets the working directory; works with both absolute and relative paths.\n\n    Args:\n        directory (str): the directory to set as the working directory\n\n    Returns:\n        dict: a dictionary containing a `directory` key of the\n            directory that was set\n    \"\"\"\n    os.chdir(directory)\n    return dict(directory=directory)\n
    ","tags":["Python API","projects","deployments","steps","pull step"]},{"location":"api-ref/prefect/deployments/steps/utility/","title":"steps.utility","text":"","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/deployments/steps/utility/#prefect.deployments.steps.utility","title":"prefect.deployments.steps.utility","text":"

    Utility project steps that are useful for managing a project's deployment lifecycle.

    Steps within this module can be used within a build, push, or pull deployment action.

    Example

    Use the run_shell_script setp to retrieve the short Git commit hash of the current repository and use it as a Docker image tag:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        id: get-commit-hash\n        script: git rev-parse --short HEAD\n        stream_output: false\n    - prefect_docker.deployments.steps.build_docker_image:\n        requires: prefect-docker\n        image_name: my-image\n        image_tag: \"{{ get-commit-hash.stdout }}\"\n        dockerfile: auto\n

    ","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/deployments/steps/utility/#prefect.deployments.steps.utility.RunShellScriptResult","title":"RunShellScriptResult","text":"

    Bases: TypedDict

    The result of a run_shell_script step.

    Attributes:

    Name Type Description stdout str

    The captured standard output of the script.

    stderr str

    The captured standard error of the script.

    Source code in prefect/deployments/steps/utility.py
    class RunShellScriptResult(TypedDict):\n    \"\"\"\n    The result of a `run_shell_script` step.\n\n    Attributes:\n        stdout: The captured standard output of the script.\n        stderr: The captured standard error of the script.\n    \"\"\"\n\n    stdout: str\n    stderr: str\n
    ","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/deployments/steps/utility/#prefect.deployments.steps.utility.pip_install_requirements","title":"pip_install_requirements async","text":"

    Installs dependencies from a requirements.txt file.

    Parameters:

    Name Type Description Default requirements_file str

    The requirements.txt to use for installation.

    'requirements.txt' directory Optional[str]

    The directory the requirements.txt file is in. Defaults to the current working directory.

    None stream_output bool

    Whether to stream the output from pip install should be streamed to the console

    True

    Returns:

    Type Description

    A dictionary with the keys stdout and stderr containing the output the pip install command

    Raises:

    Type Description CalledProcessError

    if the pip install command fails for any reason

    Example
    pull:\n    - prefect.deployments.steps.git_clone:\n        id: clone-step\n        repository: https://github.com/org/repo.git\n    - prefect.deployments.steps.pip_install_requirements:\n        directory: {{ clone-step.directory }}\n        requirements_file: requirements.txt\n        stream_output: False\n
    Source code in prefect/deployments/steps/utility.py
    async def pip_install_requirements(\n    directory: Optional[str] = None,\n    requirements_file: str = \"requirements.txt\",\n    stream_output: bool = True,\n):\n    \"\"\"\n    Installs dependencies from a requirements.txt file.\n\n    Args:\n        requirements_file: The requirements.txt to use for installation.\n        directory: The directory the requirements.txt file is in. Defaults to\n            the current working directory.\n        stream_output: Whether to stream the output from pip install should be\n            streamed to the console\n\n    Returns:\n        A dictionary with the keys `stdout` and `stderr` containing the output\n            the `pip install` command\n\n    Raises:\n        subprocess.CalledProcessError: if the pip install command fails for any reason\n\n    Example:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.git_clone:\n                id: clone-step\n                repository: https://github.com/org/repo.git\n            - prefect.deployments.steps.pip_install_requirements:\n                directory: {{ clone-step.directory }}\n                requirements_file: requirements.txt\n                stream_output: False\n        ```\n    \"\"\"\n    stdout_sink = io.StringIO()\n    stderr_sink = io.StringIO()\n\n    async with open_process(\n        [get_sys_executable(), \"-m\", \"pip\", \"install\", \"-r\", requirements_file],\n        stdout=subprocess.PIPE,\n        stderr=subprocess.PIPE,\n        cwd=directory,\n    ) as process:\n        await _stream_capture_process_output(\n            process,\n            stdout_sink=stdout_sink,\n            stderr_sink=stderr_sink,\n            stream_output=stream_output,\n        )\n        await process.wait()\n\n        if process.returncode != 0:\n            raise RuntimeError(\n                f\"pip_install_requirements failed with error code {process.returncode}:\"\n                f\" {stderr_sink.getvalue()}\"\n            )\n\n    return {\n        \"stdout\": stdout_sink.getvalue().strip(),\n        \"stderr\": stderr_sink.getvalue().strip(),\n    }\n
    ","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/deployments/steps/utility/#prefect.deployments.steps.utility.run_shell_script","title":"run_shell_script async","text":"

    Runs one or more shell commands in a subprocess. Returns the standard output and standard error of the script.

    Parameters:

    Name Type Description Default script str

    The script to run

    required directory Optional[str]

    The directory to run the script in. Defaults to the current working directory.

    None env Optional[Dict[str, str]]

    A dictionary of environment variables to set for the script

    None stream_output bool

    Whether to stream the output of the script to stdout/stderr

    True expand_env_vars bool

    Whether to expand environment variables in the script before running it

    False

    Returns:

    Type Description RunShellScriptResult

    A dictionary with the keys stdout and stderr containing the output of the script

    Examples:

    Retrieve the short Git commit hash of the current repository to use as a Docker image tag:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        id: get-commit-hash\n        script: git rev-parse --short HEAD\n        stream_output: false\n    - prefect_docker.deployments.steps.build_docker_image:\n        requires: prefect-docker\n        image_name: my-image\n        image_tag: \"{{ get-commit-hash.stdout }}\"\n        dockerfile: auto\n

    Run a multi-line shell script:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        script: |\n            echo \"Hello\"\n            echo \"World\"\n

    Run a shell script with environment variables:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        script: echo \"Hello $NAME\"\n        env:\n            NAME: World\n

    Run a shell script with environment variables expanded from the current environment:

    pull:\n    - prefect.deployments.steps.run_shell_script:\n        script: |\n            echo \"User: $USER\"\n            echo \"Home Directory: $HOME\"\n        stream_output: true\n        expand_env_vars: true\n

    Run a shell script in a specific directory:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        script: echo \"Hello\"\n        directory: /path/to/directory\n

    Run a script stored in a file:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        script: \"bash path/to/script.sh\"\n

    Source code in prefect/deployments/steps/utility.py
    async def run_shell_script(\n    script: str,\n    directory: Optional[str] = None,\n    env: Optional[Dict[str, str]] = None,\n    stream_output: bool = True,\n    expand_env_vars: bool = False,\n) -> RunShellScriptResult:\n    \"\"\"\n    Runs one or more shell commands in a subprocess. Returns the standard\n    output and standard error of the script.\n\n    Args:\n        script: The script to run\n        directory: The directory to run the script in. Defaults to the current\n            working directory.\n        env: A dictionary of environment variables to set for the script\n        stream_output: Whether to stream the output of the script to\n            stdout/stderr\n        expand_env_vars: Whether to expand environment variables in the script\n            before running it\n\n    Returns:\n        A dictionary with the keys `stdout` and `stderr` containing the output\n            of the script\n\n    Examples:\n        Retrieve the short Git commit hash of the current repository to use as\n            a Docker image tag:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                id: get-commit-hash\n                script: git rev-parse --short HEAD\n                stream_output: false\n            - prefect_docker.deployments.steps.build_docker_image:\n                requires: prefect-docker\n                image_name: my-image\n                image_tag: \"{{ get-commit-hash.stdout }}\"\n                dockerfile: auto\n        ```\n\n        Run a multi-line shell script:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                script: |\n                    echo \"Hello\"\n                    echo \"World\"\n        ```\n\n        Run a shell script with environment variables:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                script: echo \"Hello $NAME\"\n                env:\n                    NAME: World\n        ```\n\n        Run a shell script with environment variables expanded\n            from the current environment:\n        ```yaml\n        pull:\n            - prefect.deployments.steps.run_shell_script:\n                script: |\n                    echo \"User: $USER\"\n                    echo \"Home Directory: $HOME\"\n                stream_output: true\n                expand_env_vars: true\n        ```\n\n        Run a shell script in a specific directory:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                script: echo \"Hello\"\n                directory: /path/to/directory\n        ```\n\n        Run a script stored in a file:\n        ```yaml\n        build:\n            - prefect.deployments.steps.run_shell_script:\n                script: \"bash path/to/script.sh\"\n        ```\n    \"\"\"\n    current_env = os.environ.copy()\n    current_env.update(env or {})\n\n    commands = script.splitlines()\n    stdout_sink = io.StringIO()\n    stderr_sink = io.StringIO()\n\n    for command in commands:\n        if expand_env_vars:\n            # Expand environment variables in command and provided environment\n            command = string.Template(command).safe_substitute(current_env)\n        split_command = shlex.split(command, posix=sys.platform != \"win32\")\n        if not split_command:\n            continue\n        async with open_process(\n            split_command,\n            stdout=subprocess.PIPE,\n            stderr=subprocess.PIPE,\n            cwd=directory,\n            env=current_env,\n        ) as process:\n            await _stream_capture_process_output(\n                process,\n                stdout_sink=stdout_sink,\n                stderr_sink=stderr_sink,\n                stream_output=stream_output,\n            )\n\n            await process.wait()\n\n    return {\n        \"stdout\": stdout_sink.getvalue().strip(),\n        \"stderr\": stderr_sink.getvalue().strip(),\n    }\n
    ","tags":["Python API","projects","deployments","steps","shell","pip install","requirements.txt"]},{"location":"api-ref/prefect/input/actions/","title":"actions","text":"","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/actions/#prefect.input.actions","title":"prefect.input.actions","text":"","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/actions/#prefect.input.actions.create_flow_run_input","title":"create_flow_run_input async","text":"

    Create a new flow run input. The given value will be serialized to JSON and stored as a flow run input value.

    Parameters:

    Name Type Description Default - key (str

    the flow run input key

    required - value (Any

    the flow run input value

    required - flow_run_id (UUID

    the, optional, flow run ID. If not given will default to pulling the flow run ID from the current context.

    required Source code in prefect/input/actions.py
    @sync_compatible\n@inject_client\nasync def create_flow_run_input(\n    key: str,\n    value: Any,\n    flow_run_id: Optional[UUID] = None,\n    sender: Optional[str] = None,\n    client: \"PrefectClient\" = None,\n):\n    \"\"\"\n    Create a new flow run input. The given `value` will be serialized to JSON\n    and stored as a flow run input value.\n\n    Args:\n        - key (str): the flow run input key\n        - value (Any): the flow run input value\n        - flow_run_id (UUID): the, optional, flow run ID. If not given will\n          default to pulling the flow run ID from the current context.\n    \"\"\"\n    flow_run_id = ensure_flow_run_id(flow_run_id)\n\n    await client.create_flow_run_input(\n        flow_run_id=flow_run_id,\n        key=key,\n        sender=sender,\n        value=orjson.dumps(value).decode(),\n    )\n
    ","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/actions/#prefect.input.actions.delete_flow_run_input","title":"delete_flow_run_input async","text":"

    Delete a flow run input.

    Parameters:

    Name Type Description Default - flow_run_id (UUID

    the flow run ID

    required - key (str

    the flow run input key

    required Source code in prefect/input/actions.py
    @sync_compatible\n@inject_client\nasync def delete_flow_run_input(\n    key: str, flow_run_id: Optional[UUID] = None, client: \"PrefectClient\" = None\n):\n    \"\"\"Delete a flow run input.\n\n    Args:\n        - flow_run_id (UUID): the flow run ID\n        - key (str): the flow run input key\n    \"\"\"\n\n    flow_run_id = ensure_flow_run_id(flow_run_id)\n\n    await client.delete_flow_run_input(flow_run_id=flow_run_id, key=key)\n
    ","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/actions/#prefect.input.actions.read_flow_run_input","title":"read_flow_run_input async","text":"

    Read a flow run input.

    Parameters:

    Name Type Description Default - key (str

    the flow run input key

    required - flow_run_id (UUID

    the flow run ID

    required Source code in prefect/input/actions.py
    @sync_compatible\n@inject_client\nasync def read_flow_run_input(\n    key: str, flow_run_id: Optional[UUID] = None, client: \"PrefectClient\" = None\n) -> Any:\n    \"\"\"Read a flow run input.\n\n    Args:\n        - key (str): the flow run input key\n        - flow_run_id (UUID): the flow run ID\n    \"\"\"\n    flow_run_id = ensure_flow_run_id(flow_run_id)\n\n    try:\n        value = await client.read_flow_run_input(flow_run_id=flow_run_id, key=key)\n    except PrefectHTTPStatusError as exc:\n        if exc.response.status_code == 404:\n            return None\n        raise\n    else:\n        return orjson.loads(value)\n
    ","tags":["flow run","pause","suspend","input"]},{"location":"api-ref/prefect/input/run_input/","title":"run_input","text":"","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input","title":"prefect.input.run_input","text":"

    This module contains functions that allow sending type-checked RunInput data to flows at runtime. Flows can send back responses, establishing two-way channels with senders. These functions are particularly useful for systems that require ongoing data transfer or need to react to input quickly. real-time interaction and efficient data handling. It's designed to facilitate dynamic communication within distributed or microservices-oriented systems, making it ideal for scenarios requiring continuous data synchronization and processing. It's particularly useful for systems that require ongoing data input and output.

    The following is an example of two flows. One sends a random number to the other and waits for a response. The other receives the number, squares it, and sends the result back. The sender flow then prints the result.

    Sender flow:

    import random\nfrom uuid import UUID\nfrom prefect import flow, get_run_logger\nfrom prefect.input import RunInput\n\nclass NumberData(RunInput):\n    number: int\n\n\n@flow\nasync def sender_flow(receiver_flow_run_id: UUID):\n    logger = get_run_logger()\n\n    the_number = random.randint(1, 100)\n\n    await NumberData(number=the_number).send_to(receiver_flow_run_id)\n\n    receiver = NumberData.receive(flow_run_id=receiver_flow_run_id)\n    squared = await receiver.next()\n\n    logger.info(f\"{the_number} squared is {squared.number}\")\n

    Receiver flow:

    import random\nfrom uuid import UUID\nfrom prefect import flow, get_run_logger\nfrom prefect.input import RunInput\n\nclass NumberData(RunInput):\n    number: int\n\n\n@flow\nasync def receiver_flow():\n    async for data in NumberData.receive():\n        squared = data.number ** 2\n        data.respond(NumberData(number=squared))\n

    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput","title":"RunInput","text":"

    Bases: BaseModel

    Source code in prefect/input/run_input.py
    class RunInput(pydantic.BaseModel):\n    class Config:\n        extra = \"forbid\"\n\n    _metadata: RunInputMetadata = pydantic.PrivateAttr()\n\n    @property\n    def metadata(self) -> RunInputMetadata:\n        return self._metadata\n\n    @classmethod\n    def keyset_from_type(cls) -> Keyset:\n        return keyset_from_base_key(cls.__name__.lower())\n\n    @classmethod\n    @sync_compatible\n    async def save(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):\n        \"\"\"\n        Save the run input response to the given key.\n\n        Args:\n            - keyset (Keyset): the keyset to save the input for\n            - flow_run_id (UUID, optional): the flow run ID to save the input for\n        \"\"\"\n\n        if HAS_PYDANTIC_V2:\n            schema = create_v2_schema(cls.__name__, model_base=cls)\n        else:\n            schema = cls.schema(by_alias=True)\n\n        await create_flow_run_input(\n            key=keyset[\"schema\"], value=schema, flow_run_id=flow_run_id\n        )\n\n    @classmethod\n    @sync_compatible\n    async def load(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):\n        \"\"\"\n        Load the run input response from the given key.\n\n        Args:\n            - keyset (Keyset): the keyset to load the input for\n            - flow_run_id (UUID, optional): the flow run ID to load the input for\n        \"\"\"\n        flow_run_id = ensure_flow_run_id(flow_run_id)\n        value = await read_flow_run_input(keyset[\"response\"], flow_run_id=flow_run_id)\n        instance = cls(**value)\n        instance._metadata = RunInputMetadata(\n            key=keyset[\"response\"], sender=None, receiver=flow_run_id\n        )\n        return instance\n\n    @classmethod\n    def load_from_flow_run_input(cls, flow_run_input: \"FlowRunInput\"):\n        \"\"\"\n        Load the run input from a FlowRunInput object.\n\n        Args:\n            - flow_run_input (FlowRunInput): the flow run input to load the input for\n        \"\"\"\n        instance = cls(**flow_run_input.decoded_value)\n        instance._metadata = RunInputMetadata(\n            key=flow_run_input.key,\n            sender=flow_run_input.sender,\n            receiver=flow_run_input.flow_run_id,\n        )\n        return instance\n\n    @classmethod\n    def with_initial_data(cls: Type[T], **kwargs: Any) -> Type[T]:\n        \"\"\"\n        Create a new `RunInput` subclass with the given initial data as field\n        defaults.\n\n        Args:\n            - kwargs (Any): the initial data\n        \"\"\"\n        fields = {}\n        for key, value in kwargs.items():\n            fields[key] = (type(value), value)\n        return pydantic.create_model(cls.__name__, **fields, __base__=cls)\n\n    @sync_compatible\n    async def respond(\n        self,\n        run_input: \"RunInput\",\n        sender: Optional[str] = None,\n        key_prefix: Optional[str] = None,\n    ):\n        flow_run_id = None\n        if self.metadata.sender and self.metadata.sender.startswith(\"prefect.flow-run\"):\n            _, _, id = self.metadata.sender.rpartition(\".\")\n            flow_run_id = UUID(id)\n\n        if not flow_run_id:\n            raise RuntimeError(\n                \"Cannot respond to an input that was not sent by a flow run.\"\n            )\n\n        await _send_input(\n            flow_run_id=flow_run_id,\n            run_input=run_input,\n            sender=sender,\n            key_prefix=key_prefix,\n        )\n\n    @sync_compatible\n    async def send_to(\n        self,\n        flow_run_id: UUID,\n        sender: Optional[str] = None,\n        key_prefix: Optional[str] = None,\n    ):\n        await _send_input(\n            flow_run_id=flow_run_id,\n            run_input=self,\n            sender=sender,\n            key_prefix=key_prefix,\n        )\n\n    @classmethod\n    def receive(\n        cls,\n        timeout: Optional[float] = 3600,\n        poll_interval: float = 10,\n        raise_timeout_error: bool = False,\n        exclude_keys: Optional[Set[str]] = None,\n        key_prefix: Optional[str] = None,\n        flow_run_id: Optional[UUID] = None,\n    ):\n        if key_prefix is None:\n            key_prefix = f\"{cls.__name__.lower()}-auto\"\n\n        return GetInputHandler(\n            run_input_cls=cls,\n            key_prefix=key_prefix,\n            timeout=timeout,\n            poll_interval=poll_interval,\n            raise_timeout_error=raise_timeout_error,\n            exclude_keys=exclude_keys,\n            flow_run_id=flow_run_id,\n        )\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput.load","title":"load async classmethod","text":"

    Load the run input response from the given key.

    Parameters:

    Name Type Description Default - keyset (Keyset

    the keyset to load the input for

    required - flow_run_id (UUID

    the flow run ID to load the input for

    required Source code in prefect/input/run_input.py
    @classmethod\n@sync_compatible\nasync def load(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):\n    \"\"\"\n    Load the run input response from the given key.\n\n    Args:\n        - keyset (Keyset): the keyset to load the input for\n        - flow_run_id (UUID, optional): the flow run ID to load the input for\n    \"\"\"\n    flow_run_id = ensure_flow_run_id(flow_run_id)\n    value = await read_flow_run_input(keyset[\"response\"], flow_run_id=flow_run_id)\n    instance = cls(**value)\n    instance._metadata = RunInputMetadata(\n        key=keyset[\"response\"], sender=None, receiver=flow_run_id\n    )\n    return instance\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput.load_from_flow_run_input","title":"load_from_flow_run_input classmethod","text":"

    Load the run input from a FlowRunInput object.

    Parameters:

    Name Type Description Default - flow_run_input (FlowRunInput

    the flow run input to load the input for

    required Source code in prefect/input/run_input.py
    @classmethod\ndef load_from_flow_run_input(cls, flow_run_input: \"FlowRunInput\"):\n    \"\"\"\n    Load the run input from a FlowRunInput object.\n\n    Args:\n        - flow_run_input (FlowRunInput): the flow run input to load the input for\n    \"\"\"\n    instance = cls(**flow_run_input.decoded_value)\n    instance._metadata = RunInputMetadata(\n        key=flow_run_input.key,\n        sender=flow_run_input.sender,\n        receiver=flow_run_input.flow_run_id,\n    )\n    return instance\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput.save","title":"save async classmethod","text":"

    Save the run input response to the given key.

    Parameters:

    Name Type Description Default - keyset (Keyset

    the keyset to save the input for

    required - flow_run_id (UUID

    the flow run ID to save the input for

    required Source code in prefect/input/run_input.py
    @classmethod\n@sync_compatible\nasync def save(cls, keyset: Keyset, flow_run_id: Optional[UUID] = None):\n    \"\"\"\n    Save the run input response to the given key.\n\n    Args:\n        - keyset (Keyset): the keyset to save the input for\n        - flow_run_id (UUID, optional): the flow run ID to save the input for\n    \"\"\"\n\n    if HAS_PYDANTIC_V2:\n        schema = create_v2_schema(cls.__name__, model_base=cls)\n    else:\n        schema = cls.schema(by_alias=True)\n\n    await create_flow_run_input(\n        key=keyset[\"schema\"], value=schema, flow_run_id=flow_run_id\n    )\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.RunInput.with_initial_data","title":"with_initial_data classmethod","text":"

    Create a new RunInput subclass with the given initial data as field defaults.

    Parameters:

    Name Type Description Default - kwargs (Any

    the initial data

    required Source code in prefect/input/run_input.py
    @classmethod\ndef with_initial_data(cls: Type[T], **kwargs: Any) -> Type[T]:\n    \"\"\"\n    Create a new `RunInput` subclass with the given initial data as field\n    defaults.\n\n    Args:\n        - kwargs (Any): the initial data\n    \"\"\"\n    fields = {}\n    for key, value in kwargs.items():\n        fields[key] = (type(value), value)\n    return pydantic.create_model(cls.__name__, **fields, __base__=cls)\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.keyset_from_base_key","title":"keyset_from_base_key","text":"

    Get the keyset for the given base key.

    Parameters:

    Name Type Description Default - base_key (str

    the base key to get the keyset for

    required

    Returns:

    Type Description Keyset
    • Dict[str, str]: the keyset
    Source code in prefect/input/run_input.py
    def keyset_from_base_key(base_key: str) -> Keyset:\n    \"\"\"\n    Get the keyset for the given base key.\n\n    Args:\n        - base_key (str): the base key to get the keyset for\n\n    Returns:\n        - Dict[str, str]: the keyset\n    \"\"\"\n    return {\n        \"response\": f\"{base_key}-response\",\n        \"schema\": f\"{base_key}-schema\",\n    }\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/input/run_input/#prefect.input.run_input.keyset_from_paused_state","title":"keyset_from_paused_state","text":"

    Get the keyset for the given Paused state.

    Parameters:

    Name Type Description Default - state (State

    the state to get the keyset for

    required Source code in prefect/input/run_input.py
    def keyset_from_paused_state(state: \"State\") -> Keyset:\n    \"\"\"\n    Get the keyset for the given Paused state.\n\n    Args:\n        - state (State): the state to get the keyset for\n    \"\"\"\n\n    if not state.is_paused():\n        raise RuntimeError(f\"{state.type.value!r} is unsupported.\")\n\n    base_key = f\"{state.name.lower()}-{str(state.state_details.pause_key)}\"\n    return keyset_from_base_key(base_key)\n
    ","tags":["flow run","pause","suspend","input","run input"]},{"location":"api-ref/prefect/logging/configuration/","title":"configuration","text":"

    \"\"\"

    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/configuration/#prefect.logging.configuration","title":"prefect.logging.configuration","text":"","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/configuration/#prefect.logging.configuration.load_logging_config","title":"load_logging_config","text":"

    Loads logging configuration from a path allowing override from the environment

    Source code in prefect/logging/configuration.py
    def load_logging_config(path: Path) -> dict:\n    \"\"\"\n    Loads logging configuration from a path allowing override from the environment\n    \"\"\"\n    template = string.Template(path.read_text())\n    with warnings.catch_warnings():\n        warnings.filterwarnings(\"ignore\", category=DeprecationWarning)\n        config = yaml.safe_load(\n            # Substitute settings into the template in format $SETTING / ${SETTING}\n            template.substitute(\n                {\n                    setting.name: str(setting.value())\n                    for setting in SETTING_VARIABLES.values()\n                    if setting.value() is not None\n                }\n            )\n        )\n\n    # Load overrides from the environment\n    flat_config = dict_to_flatdict(config)\n\n    for key_tup, val in flat_config.items():\n        env_val = os.environ.get(\n            # Generate a valid environment variable with nesting indicated with '_'\n            to_envvar(\"PREFECT_LOGGING_\" + \"_\".join(key_tup)).upper()\n        )\n        if env_val:\n            val = env_val\n\n        # reassign the updated value\n        flat_config[key_tup] = val\n\n    return flatdict_to_dict(flat_config)\n
    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/configuration/#prefect.logging.configuration.setup_logging","title":"setup_logging","text":"

    Sets up logging.

    Returns the config used.

    Source code in prefect/logging/configuration.py
    def setup_logging(incremental: Optional[bool] = None) -> dict:\n    \"\"\"\n    Sets up logging.\n\n    Returns the config used.\n    \"\"\"\n    global PROCESS_LOGGING_CONFIG\n\n    # If the user has specified a logging path and it exists we will ignore the\n    # default entirely rather than dealing with complex merging\n    config = load_logging_config(\n        (\n            PREFECT_LOGGING_SETTINGS_PATH.value()\n            if PREFECT_LOGGING_SETTINGS_PATH.value().exists()\n            else DEFAULT_LOGGING_SETTINGS_PATH\n        )\n    )\n\n    incremental = (\n        incremental if incremental is not None else bool(PROCESS_LOGGING_CONFIG)\n    )\n\n    # Perform an incremental update if setup has already been run\n    config.setdefault(\"incremental\", incremental)\n\n    try:\n        logging.config.dictConfig(config)\n    except ValueError:\n        if incremental:\n            setup_logging(incremental=False)\n\n    # Copy configuration of the 'prefect.extra' logger to the extra loggers\n    extra_config = logging.getLogger(\"prefect.extra\")\n\n    for logger_name in PREFECT_LOGGING_EXTRA_LOGGERS.value():\n        logger = logging.getLogger(logger_name)\n        for handler in extra_config.handlers:\n            if not config[\"incremental\"]:\n                logger.addHandler(handler)\n            if logger.level == logging.NOTSET:\n                logger.setLevel(extra_config.level)\n            logger.propagate = extra_config.propagate\n\n    PROCESS_LOGGING_CONFIG = config\n\n    return config\n
    ","tags":["Python API","logging"]},{"location":"api-ref/prefect/logging/formatters/","title":"formatters","text":"

    \"\"\"

    ","tags":["Python API","logging","formatters"]},{"location":"api-ref/prefect/logging/formatters/#prefect.logging.formatters","title":"prefect.logging.formatters","text":"","tags":["Python API","logging","formatters"]},{"location":"api-ref/prefect/logging/formatters/#prefect.logging.formatters.JsonFormatter","title":"JsonFormatter","text":"

    Bases: Formatter

    Formats log records as a JSON string.

    The format may be specified as \"pretty\" to format the JSON with indents and newlines.

    Source code in prefect/logging/formatters.py
    class JsonFormatter(logging.Formatter):\n    \"\"\"\n    Formats log records as a JSON string.\n\n    The format may be specified as \"pretty\" to format the JSON with indents and\n    newlines.\n    \"\"\"\n\n    def __init__(self, fmt, dmft, style) -> None:  # noqa\n        super().__init__()\n\n        if fmt not in [\"pretty\", \"default\"]:\n            raise ValueError(\"Format must be either 'pretty' or 'default'.\")\n\n        self.serializer = JSONSerializer(\n            jsonlib=\"orjson\",\n            object_encoder=\"pydantic.json.pydantic_encoder\",\n            dumps_kwargs={\"option\": orjson.OPT_INDENT_2} if fmt == \"pretty\" else {},\n        )\n\n    def format(self, record: logging.LogRecord) -> str:\n        record_dict = record.__dict__.copy()\n\n        # GCP severity detection compatibility\n        record_dict.setdefault(\"severity\", record.levelname)\n\n        # replace any exception tuples returned by `sys.exc_info()`\n        # with a JSON-serializable `dict`.\n        if record.exc_info:\n            record_dict[\"exc_info\"] = format_exception_info(record.exc_info)\n\n        log_json_bytes = self.serializer.dumps(record_dict)\n\n        # JSONSerializer returns bytes; decode to string to conform to\n        # the `logging.Formatter.format` interface\n        return log_json_bytes.decode()\n
    ","tags":["Python API","logging","formatters"]},{"location":"api-ref/prefect/logging/formatters/#prefect.logging.formatters.PrefectFormatter","title":"PrefectFormatter","text":"

    Bases: Formatter

    Source code in prefect/logging/formatters.py
    class PrefectFormatter(logging.Formatter):\n    def __init__(\n        self,\n        format=None,\n        datefmt=None,\n        style=\"%\",\n        validate=True,\n        *,\n        defaults=None,\n        task_run_fmt: str = None,\n        flow_run_fmt: str = None\n    ) -> None:\n        \"\"\"\n        Implementation of the standard Python formatter with support for multiple\n        message formats.\n\n        \"\"\"\n        # See https://github.com/python/cpython/blob/c8c6113398ee9a7867fe9b08bc539cceb61e2aaa/Lib/logging/__init__.py#L546\n        # for implementation details\n\n        init_kwargs = {}\n        style_kwargs = {}\n\n        # defaults added in 3.10\n        if sys.version_info >= (3, 10):\n            init_kwargs[\"defaults\"] = defaults\n            style_kwargs[\"defaults\"] = defaults\n\n        # validate added in 3.8\n        if sys.version_info >= (3, 8):\n            init_kwargs[\"validate\"] = validate\n        else:\n            validate = False\n\n        super().__init__(format, datefmt, style, **init_kwargs)\n\n        self.flow_run_fmt = flow_run_fmt\n        self.task_run_fmt = task_run_fmt\n\n        # Retrieve the style class from the base class to avoid importing private\n        # `_STYLES` mapping\n        style_class = type(self._style)\n\n        self._flow_run_style = (\n            style_class(flow_run_fmt, **style_kwargs) if flow_run_fmt else self._style\n        )\n        self._task_run_style = (\n            style_class(task_run_fmt, **style_kwargs) if task_run_fmt else self._style\n        )\n        if validate:\n            self._flow_run_style.validate()\n            self._task_run_style.validate()\n\n    def formatMessage(self, record: logging.LogRecord):\n        if record.name == \"prefect.flow_runs\":\n            style = self._flow_run_style\n        elif record.name == \"prefect.task_runs\":\n            style = self._task_run_style\n        else:\n            style = self._style\n\n        return style.format(record)\n
    ","tags":["Python API","logging","formatters"]},{"location":"api-ref/prefect/logging/handlers/","title":"handlers","text":"

    \"\"\"

    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers","title":"prefect.logging.handlers","text":"","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler","title":"APILogHandler","text":"

    Bases: Handler

    A logging handler that sends logs to the Prefect API.

    Sends log records to the APILogWorker which manages sending batches of logs in the background.

    Source code in prefect/logging/handlers.py
    class APILogHandler(logging.Handler):\n    \"\"\"\n    A logging handler that sends logs to the Prefect API.\n\n    Sends log records to the `APILogWorker` which manages sending batches of logs in\n    the background.\n    \"\"\"\n\n    @classmethod\n    def flush(cls):\n        \"\"\"\n        Tell the `APILogWorker` to send any currently enqueued logs and block until\n        completion.\n\n        Use `aflush` from async contexts instead.\n        \"\"\"\n        loop = get_running_loop()\n        if loop:\n            if in_global_loop():  # Guard against internal misuse\n                raise RuntimeError(\n                    \"Cannot call `APILogWorker.flush` from the global event loop; it\"\n                    \" would block the event loop and cause a deadlock. Use\"\n                    \" `APILogWorker.aflush` instead.\"\n                )\n\n            # Not ideal, but this method is called by the stdlib and cannot return a\n            # coroutine so we just schedule the drain in a new thread and continue\n            from_sync.call_soon_in_new_thread(create_call(APILogWorker.drain_all))\n            return None\n        else:\n            # We set a timeout of 5s because we don't want to block forever if the worker\n            # is stuck. This can occur when the handler is being shutdown and the\n            # `logging._lock` is held but the worker is attempting to emit logs resulting\n            # in a deadlock.\n            return APILogWorker.drain_all(timeout=5)\n\n    @classmethod\n    def aflush(cls):\n        \"\"\"\n        Tell the `APILogWorker` to send any currently enqueued logs and block until\n        completion.\n\n        If called in a synchronous context, will only block up to 5s before returning.\n        \"\"\"\n\n        if not get_running_loop():\n            raise RuntimeError(\n                \"`aflush` cannot be used from a synchronous context; use `flush`\"\n                \" instead.\"\n            )\n\n        return APILogWorker.drain_all()\n\n    def emit(self, record: logging.LogRecord):\n        \"\"\"\n        Send a log to the `APILogWorker`\n        \"\"\"\n        try:\n            profile = prefect.context.get_settings_context()\n\n            if not PREFECT_LOGGING_TO_API_ENABLED.value_from(profile.settings):\n                return  # Respect the global settings toggle\n            if not getattr(record, \"send_to_api\", True):\n                return  # Do not send records that have opted out\n            if not getattr(record, \"send_to_orion\", True):\n                return  # Backwards compatibility\n\n            log = self.prepare(record)\n            APILogWorker.instance().send(log)\n\n        except Exception:\n            self.handleError(record)\n\n    def handleError(self, record: logging.LogRecord) -> None:\n        _, exc, _ = sys.exc_info()\n\n        if isinstance(exc, MissingContextError):\n            log_handling_when_missing_flow = (\n                PREFECT_LOGGING_TO_API_WHEN_MISSING_FLOW.value()\n            )\n            if log_handling_when_missing_flow == \"warn\":\n                # Warn when a logger is used outside of a run context, the stack level here\n                # gets us to the user logging call\n                warnings.warn(str(exc), stacklevel=8)\n                return\n            elif log_handling_when_missing_flow == \"ignore\":\n                return\n            else:\n                raise exc\n\n        # Display a longer traceback for other errors\n        return super().handleError(record)\n\n    def prepare(self, record: logging.LogRecord) -> Dict[str, Any]:\n        \"\"\"\n        Convert a `logging.LogRecord` to the API `LogCreate` schema and serialize.\n\n        This infers the linked flow or task run from the log record or the current\n        run context.\n\n        If a flow run id cannot be found, the log will be dropped.\n\n        Logs exceeding the maximum size will be dropped.\n        \"\"\"\n        flow_run_id = getattr(record, \"flow_run_id\", None)\n        task_run_id = getattr(record, \"task_run_id\", None)\n\n        if not flow_run_id:\n            try:\n                context = prefect.context.get_run_context()\n            except MissingContextError:\n                raise MissingContextError(\n                    f\"Logger {record.name!r} attempted to send logs to the API without\"\n                    \" a flow run id. The API log handler can only send logs within\"\n                    \" flow run contexts unless the flow run id is manually provided.\"\n                ) from None\n\n            if hasattr(context, \"flow_run\"):\n                flow_run_id = context.flow_run.id\n            elif hasattr(context, \"task_run\"):\n                flow_run_id = context.task_run.flow_run_id\n                task_run_id = task_run_id or context.task_run.id\n            else:\n                raise ValueError(\n                    \"Encountered malformed run context. Does not contain flow or task \"\n                    \"run information.\"\n                )\n\n        # Parsing to a `LogCreate` object here gives us nice parsing error messages\n        # from the standard lib `handleError` method if something goes wrong and\n        # prevents malformed logs from entering the queue\n        try:\n            is_uuid_like = isinstance(flow_run_id, uuid.UUID) or (\n                isinstance(flow_run_id, str) and uuid.UUID(flow_run_id)\n            )\n        except ValueError:\n            is_uuid_like = False\n\n        log = LogCreate(\n            flow_run_id=flow_run_id if is_uuid_like else None,\n            task_run_id=task_run_id,\n            name=record.name,\n            level=record.levelno,\n            timestamp=pendulum.from_timestamp(\n                getattr(record, \"created\", None) or time.time()\n            ),\n            message=self.format(record),\n        ).dict(json_compatible=True)\n\n        log_size = log[\"__payload_size__\"] = self._get_payload_size(log)\n        if log_size > PREFECT_LOGGING_TO_API_MAX_LOG_SIZE.value():\n            raise ValueError(\n                f\"Log of size {log_size} is greater than the max size of \"\n                f\"{PREFECT_LOGGING_TO_API_MAX_LOG_SIZE.value()}\"\n            )\n\n        return log\n\n    def _get_payload_size(self, log: Dict[str, Any]) -> int:\n        return len(json.dumps(log).encode())\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler.aflush","title":"aflush classmethod","text":"

    Tell the APILogWorker to send any currently enqueued logs and block until completion.

    If called in a synchronous context, will only block up to 5s before returning.

    Source code in prefect/logging/handlers.py
    @classmethod\ndef aflush(cls):\n    \"\"\"\n    Tell the `APILogWorker` to send any currently enqueued logs and block until\n    completion.\n\n    If called in a synchronous context, will only block up to 5s before returning.\n    \"\"\"\n\n    if not get_running_loop():\n        raise RuntimeError(\n            \"`aflush` cannot be used from a synchronous context; use `flush`\"\n            \" instead.\"\n        )\n\n    return APILogWorker.drain_all()\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler.emit","title":"emit","text":"

    Send a log to the APILogWorker

    Source code in prefect/logging/handlers.py
    def emit(self, record: logging.LogRecord):\n    \"\"\"\n    Send a log to the `APILogWorker`\n    \"\"\"\n    try:\n        profile = prefect.context.get_settings_context()\n\n        if not PREFECT_LOGGING_TO_API_ENABLED.value_from(profile.settings):\n            return  # Respect the global settings toggle\n        if not getattr(record, \"send_to_api\", True):\n            return  # Do not send records that have opted out\n        if not getattr(record, \"send_to_orion\", True):\n            return  # Backwards compatibility\n\n        log = self.prepare(record)\n        APILogWorker.instance().send(log)\n\n    except Exception:\n        self.handleError(record)\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler.flush","title":"flush classmethod","text":"

    Tell the APILogWorker to send any currently enqueued logs and block until completion.

    Use aflush from async contexts instead.

    Source code in prefect/logging/handlers.py
    @classmethod\ndef flush(cls):\n    \"\"\"\n    Tell the `APILogWorker` to send any currently enqueued logs and block until\n    completion.\n\n    Use `aflush` from async contexts instead.\n    \"\"\"\n    loop = get_running_loop()\n    if loop:\n        if in_global_loop():  # Guard against internal misuse\n            raise RuntimeError(\n                \"Cannot call `APILogWorker.flush` from the global event loop; it\"\n                \" would block the event loop and cause a deadlock. Use\"\n                \" `APILogWorker.aflush` instead.\"\n            )\n\n        # Not ideal, but this method is called by the stdlib and cannot return a\n        # coroutine so we just schedule the drain in a new thread and continue\n        from_sync.call_soon_in_new_thread(create_call(APILogWorker.drain_all))\n        return None\n    else:\n        # We set a timeout of 5s because we don't want to block forever if the worker\n        # is stuck. This can occur when the handler is being shutdown and the\n        # `logging._lock` is held but the worker is attempting to emit logs resulting\n        # in a deadlock.\n        return APILogWorker.drain_all(timeout=5)\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.APILogHandler.prepare","title":"prepare","text":"

    Convert a logging.LogRecord to the API LogCreate schema and serialize.

    This infers the linked flow or task run from the log record or the current run context.

    If a flow run id cannot be found, the log will be dropped.

    Logs exceeding the maximum size will be dropped.

    Source code in prefect/logging/handlers.py
    def prepare(self, record: logging.LogRecord) -> Dict[str, Any]:\n    \"\"\"\n    Convert a `logging.LogRecord` to the API `LogCreate` schema and serialize.\n\n    This infers the linked flow or task run from the log record or the current\n    run context.\n\n    If a flow run id cannot be found, the log will be dropped.\n\n    Logs exceeding the maximum size will be dropped.\n    \"\"\"\n    flow_run_id = getattr(record, \"flow_run_id\", None)\n    task_run_id = getattr(record, \"task_run_id\", None)\n\n    if not flow_run_id:\n        try:\n            context = prefect.context.get_run_context()\n        except MissingContextError:\n            raise MissingContextError(\n                f\"Logger {record.name!r} attempted to send logs to the API without\"\n                \" a flow run id. The API log handler can only send logs within\"\n                \" flow run contexts unless the flow run id is manually provided.\"\n            ) from None\n\n        if hasattr(context, \"flow_run\"):\n            flow_run_id = context.flow_run.id\n        elif hasattr(context, \"task_run\"):\n            flow_run_id = context.task_run.flow_run_id\n            task_run_id = task_run_id or context.task_run.id\n        else:\n            raise ValueError(\n                \"Encountered malformed run context. Does not contain flow or task \"\n                \"run information.\"\n            )\n\n    # Parsing to a `LogCreate` object here gives us nice parsing error messages\n    # from the standard lib `handleError` method if something goes wrong and\n    # prevents malformed logs from entering the queue\n    try:\n        is_uuid_like = isinstance(flow_run_id, uuid.UUID) or (\n            isinstance(flow_run_id, str) and uuid.UUID(flow_run_id)\n        )\n    except ValueError:\n        is_uuid_like = False\n\n    log = LogCreate(\n        flow_run_id=flow_run_id if is_uuid_like else None,\n        task_run_id=task_run_id,\n        name=record.name,\n        level=record.levelno,\n        timestamp=pendulum.from_timestamp(\n            getattr(record, \"created\", None) or time.time()\n        ),\n        message=self.format(record),\n    ).dict(json_compatible=True)\n\n    log_size = log[\"__payload_size__\"] = self._get_payload_size(log)\n    if log_size > PREFECT_LOGGING_TO_API_MAX_LOG_SIZE.value():\n        raise ValueError(\n            f\"Log of size {log_size} is greater than the max size of \"\n            f\"{PREFECT_LOGGING_TO_API_MAX_LOG_SIZE.value()}\"\n        )\n\n    return log\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/handlers/#prefect.logging.handlers.PrefectConsoleHandler","title":"PrefectConsoleHandler","text":"

    Bases: StreamHandler

    Source code in prefect/logging/handlers.py
    class PrefectConsoleHandler(logging.StreamHandler):\n    def __init__(\n        self,\n        stream=None,\n        highlighter: Highlighter = PrefectConsoleHighlighter,\n        styles: Dict[str, str] = None,\n        level: Union[int, str] = logging.NOTSET,\n    ):\n        \"\"\"\n        The default console handler for Prefect, which highlights log levels,\n        web and file URLs, flow and task (run) names, and state types in the\n        local console (terminal).\n\n        Highlighting can be toggled on/off with the PREFECT_LOGGING_COLORS setting.\n        For finer control, use logging.yml to add or remove styles, and/or\n        adjust colors.\n        \"\"\"\n        super().__init__(stream=stream)\n\n        styled_console = PREFECT_LOGGING_COLORS.value()\n        markup_console = PREFECT_LOGGING_MARKUP.value()\n        if styled_console:\n            highlighter = highlighter()\n            theme = Theme(styles, inherit=False)\n        else:\n            highlighter = NullHighlighter()\n            theme = Theme(inherit=False)\n\n        self.level = level\n        self.console = Console(\n            highlighter=highlighter,\n            theme=theme,\n            file=self.stream,\n            markup=markup_console,\n        )\n\n    def emit(self, record: logging.LogRecord):\n        try:\n            message = self.format(record)\n            self.console.print(message, soft_wrap=True)\n        except RecursionError:\n            # This was copied over from logging.StreamHandler().emit()\n            # https://bugs.python.org/issue36272\n            raise\n        except Exception:\n            self.handleError(record)\n
    ","tags":["Python API","logging","handlers"]},{"location":"api-ref/prefect/logging/highlighters/","title":"highlighters","text":"

    \"\"\"

    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers","title":"prefect.logging.loggers","text":"","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.PrefectLogAdapter","title":"PrefectLogAdapter","text":"

    Bases: LoggerAdapter

    Adapter that ensures extra kwargs are passed through correctly; without this the extra fields set on the adapter would overshadow any provided on a log-by-log basis.

    See https://bugs.python.org/issue32732 \u2014 the Python team has declared that this is not a bug in the LoggingAdapter and subclassing is the intended workaround.

    Source code in prefect/logging/loggers.py
    class PrefectLogAdapter(logging.LoggerAdapter):\n    \"\"\"\n    Adapter that ensures extra kwargs are passed through correctly; without this\n    the `extra` fields set on the adapter would overshadow any provided on a\n    log-by-log basis.\n\n    See https://bugs.python.org/issue32732 \u2014 the Python team has declared that this is\n    not a bug in the LoggingAdapter and subclassing is the intended workaround.\n    \"\"\"\n\n    def process(self, msg, kwargs):\n        kwargs[\"extra\"] = {**(self.extra or {}), **(kwargs.get(\"extra\") or {})}\n\n        from prefect._internal.compatibility.deprecated import (\n            PrefectDeprecationWarning,\n            generate_deprecation_message,\n        )\n\n        if \"send_to_orion\" in kwargs[\"extra\"]:\n            warnings.warn(\n                generate_deprecation_message(\n                    'The \"send_to_orion\" option',\n                    start_date=\"May 2023\",\n                    help='Use \"send_to_api\" instead.',\n                ),\n                PrefectDeprecationWarning,\n                stacklevel=4,\n            )\n\n        return (msg, kwargs)\n\n    def getChild(\n        self, suffix: str, extra: Optional[Dict[str, str]] = None\n    ) -> \"PrefectLogAdapter\":\n        if extra is None:\n            extra = {}\n\n        return PrefectLogAdapter(\n            self.logger.getChild(suffix),\n            extra={\n                **self.extra,\n                **extra,\n            },\n        )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.disable_logger","title":"disable_logger","text":"

    Get a logger by name and disables it within the context manager. Upon exiting the context manager, the logger is returned to its original state.

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef disable_logger(name: str):\n    \"\"\"\n    Get a logger by name and disables it within the context manager.\n    Upon exiting the context manager, the logger is returned to its\n    original state.\n    \"\"\"\n    logger = logging.getLogger(name=name)\n\n    # determine if it's already disabled\n    base_state = logger.disabled\n    try:\n        # disable the logger\n        logger.disabled = True\n        yield\n    finally:\n        # return to base state\n        logger.disabled = base_state\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.disable_run_logger","title":"disable_run_logger","text":"

    Gets both prefect.flow_run and prefect.task_run and disables them within the context manager. Upon exiting the context manager, both loggers are returned to its original state.

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef disable_run_logger():\n    \"\"\"\n    Gets both `prefect.flow_run` and `prefect.task_run` and disables them\n    within the context manager. Upon exiting the context manager, both loggers\n    are returned to its original state.\n    \"\"\"\n    with disable_logger(\"prefect.flow_run\"), disable_logger(\"prefect.task_run\"):\n        yield\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.flow_run_logger","title":"flow_run_logger","text":"

    Create a flow run logger with the run's metadata attached.

    Additional keyword arguments can be provided to attach custom data to the log records.

    If the flow run context is available, see get_run_logger instead.

    Source code in prefect/logging/loggers.py
    def flow_run_logger(\n    flow_run: Union[\"FlowRun\", \"ClientFlowRun\"],\n    flow: Optional[\"Flow\"] = None,\n    **kwargs: str,\n):\n    \"\"\"\n    Create a flow run logger with the run's metadata attached.\n\n    Additional keyword arguments can be provided to attach custom data to the log\n    records.\n\n    If the flow run context is available, see `get_run_logger` instead.\n    \"\"\"\n    return PrefectLogAdapter(\n        get_logger(\"prefect.flow_runs\"),\n        extra={\n            **{\n                \"flow_run_name\": flow_run.name if flow_run else \"<unknown>\",\n                \"flow_run_id\": str(flow_run.id) if flow_run else \"<unknown>\",\n                \"flow_name\": flow.name if flow else \"<unknown>\",\n            },\n            **kwargs,\n        },\n    )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.get_logger","title":"get_logger cached","text":"

    Get a prefect logger. These loggers are intended for internal use within the prefect package.

    See get_run_logger for retrieving loggers for use within task or flow runs. By default, only run-related loggers are connected to the APILogHandler.

    Source code in prefect/logging/loggers.py
    @lru_cache()\ndef get_logger(name: str = None) -> logging.Logger:\n    \"\"\"\n    Get a `prefect` logger. These loggers are intended for internal use within the\n    `prefect` package.\n\n    See `get_run_logger` for retrieving loggers for use within task or flow runs.\n    By default, only run-related loggers are connected to the `APILogHandler`.\n    \"\"\"\n\n    parent_logger = logging.getLogger(\"prefect\")\n\n    if name:\n        # Append the name if given but allow explicit full names e.g. \"prefect.test\"\n        # should not become \"prefect.prefect.test\"\n        if not name.startswith(parent_logger.name + \".\"):\n            logger = parent_logger.getChild(name)\n        else:\n            logger = logging.getLogger(name)\n    else:\n        logger = parent_logger\n\n    return logger\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.get_run_logger","title":"get_run_logger","text":"

    Get a Prefect logger for the current task run or flow run.

    The logger will be named either prefect.task_runs or prefect.flow_runs. Contextual data about the run will be attached to the log records.

    These loggers are connected to the APILogHandler by default to send log records to the API.

    Parameters:

    Name Type Description Default context RunContext

    A specific context may be provided as an override. By default, the context is inferred from global state and this should not be needed.

    None **kwargs str

    Additional keyword arguments will be attached to the log records in addition to the run metadata

    {}

    Raises:

    Type Description RuntimeError

    If no context can be found

    Source code in prefect/logging/loggers.py
    def get_run_logger(\n    context: \"RunContext\" = None, **kwargs: str\n) -> Union[logging.Logger, logging.LoggerAdapter]:\n    \"\"\"\n    Get a Prefect logger for the current task run or flow run.\n\n    The logger will be named either `prefect.task_runs` or `prefect.flow_runs`.\n    Contextual data about the run will be attached to the log records.\n\n    These loggers are connected to the `APILogHandler` by default to send log records to\n    the API.\n\n    Arguments:\n        context: A specific context may be provided as an override. By default, the\n            context is inferred from global state and this should not be needed.\n        **kwargs: Additional keyword arguments will be attached to the log records in\n            addition to the run metadata\n\n    Raises:\n        RuntimeError: If no context can be found\n    \"\"\"\n    # Check for existing contexts\n    task_run_context = prefect.context.TaskRunContext.get()\n    flow_run_context = prefect.context.FlowRunContext.get()\n\n    # Apply the context override\n    if context:\n        if isinstance(context, prefect.context.FlowRunContext):\n            flow_run_context = context\n        elif isinstance(context, prefect.context.TaskRunContext):\n            task_run_context = context\n        else:\n            raise TypeError(\n                f\"Received unexpected type {type(context).__name__!r} for context. \"\n                \"Expected one of 'None', 'FlowRunContext', or 'TaskRunContext'.\"\n            )\n\n    # Determine if this is a task or flow run logger\n    if task_run_context:\n        logger = task_run_logger(\n            task_run=task_run_context.task_run,\n            task=task_run_context.task,\n            flow_run=flow_run_context.flow_run if flow_run_context else None,\n            flow=flow_run_context.flow if flow_run_context else None,\n            **kwargs,\n        )\n    elif flow_run_context:\n        logger = flow_run_logger(\n            flow_run=flow_run_context.flow_run, flow=flow_run_context.flow, **kwargs\n        )\n    elif (\n        get_logger(\"prefect.flow_run\").disabled\n        and get_logger(\"prefect.task_run\").disabled\n    ):\n        logger = logging.getLogger(\"null\")\n    else:\n        raise MissingContextError(\"There is no active flow or task run context.\")\n\n    return logger\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.patch_print","title":"patch_print","text":"

    Patches the Python builtin print method to use print_as_log

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef patch_print():\n    \"\"\"\n    Patches the Python builtin `print` method to use `print_as_log`\n    \"\"\"\n    import builtins\n\n    original = builtins.print\n\n    try:\n        builtins.print = print_as_log\n        yield\n    finally:\n        builtins.print = original\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.print_as_log","title":"print_as_log","text":"

    A patch for print to send printed messages to the Prefect run logger.

    If no run is active, print will behave as if it were not patched.

    If print sends data to a file other than sys.stdout or sys.stderr, it will not be forwarded to the Prefect logger either.

    Source code in prefect/logging/loggers.py
    def print_as_log(*args, **kwargs):\n    \"\"\"\n    A patch for `print` to send printed messages to the Prefect run logger.\n\n    If no run is active, `print` will behave as if it were not patched.\n\n    If `print` sends data to a file other than `sys.stdout` or `sys.stderr`, it will\n    not be forwarded to the Prefect logger either.\n    \"\"\"\n    from prefect.context import FlowRunContext, TaskRunContext\n\n    context = TaskRunContext.get() or FlowRunContext.get()\n    if (\n        not context\n        or not context.log_prints\n        or kwargs.get(\"file\") not in {None, sys.stdout, sys.stderr}\n    ):\n        return print(*args, **kwargs)\n\n    logger = get_run_logger()\n\n    # Print to an in-memory buffer; so we do not need to implement `print`\n    buffer = io.StringIO()\n    kwargs[\"file\"] = buffer\n    print(*args, **kwargs)\n\n    # Remove trailing whitespace to prevent duplicates\n    logger.info(buffer.getvalue().rstrip())\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/highlighters/#prefect.logging.loggers.task_run_logger","title":"task_run_logger","text":"

    Create a task run logger with the run's metadata attached.

    Additional keyword arguments can be provided to attach custom data to the log records.

    If the task run context is available, see get_run_logger instead.

    If only the flow run context is available, it will be used for default values of flow_run and flow.

    Source code in prefect/logging/loggers.py
    def task_run_logger(\n    task_run: \"TaskRun\",\n    task: \"Task\" = None,\n    flow_run: \"FlowRun\" = None,\n    flow: \"Flow\" = None,\n    **kwargs: str,\n):\n    \"\"\"\n    Create a task run logger with the run's metadata attached.\n\n    Additional keyword arguments can be provided to attach custom data to the log\n    records.\n\n    If the task run context is available, see `get_run_logger` instead.\n\n    If only the flow run context is available, it will be used for default values\n    of `flow_run` and `flow`.\n    \"\"\"\n    if not flow_run or not flow:\n        flow_run_context = prefect.context.FlowRunContext.get()\n        if flow_run_context:\n            flow_run = flow_run or flow_run_context.flow_run\n            flow = flow or flow_run_context.flow\n\n    return PrefectLogAdapter(\n        get_logger(\"prefect.task_runs\"),\n        extra={\n            **{\n                \"task_run_id\": str(task_run.id),\n                \"flow_run_id\": str(task_run.flow_run_id),\n                \"task_run_name\": task_run.name,\n                \"task_name\": task.name if task else \"<unknown>\",\n                \"flow_run_name\": flow_run.name if flow_run else \"<unknown>\",\n                \"flow_name\": flow.name if flow else \"<unknown>\",\n            },\n            **kwargs,\n        },\n    )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/","title":"loggers","text":"

    \"\"\"

    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers","title":"prefect.logging.loggers","text":"","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.PrefectLogAdapter","title":"PrefectLogAdapter","text":"

    Bases: LoggerAdapter

    Adapter that ensures extra kwargs are passed through correctly; without this the extra fields set on the adapter would overshadow any provided on a log-by-log basis.

    See https://bugs.python.org/issue32732 \u2014 the Python team has declared that this is not a bug in the LoggingAdapter and subclassing is the intended workaround.

    Source code in prefect/logging/loggers.py
    class PrefectLogAdapter(logging.LoggerAdapter):\n    \"\"\"\n    Adapter that ensures extra kwargs are passed through correctly; without this\n    the `extra` fields set on the adapter would overshadow any provided on a\n    log-by-log basis.\n\n    See https://bugs.python.org/issue32732 \u2014 the Python team has declared that this is\n    not a bug in the LoggingAdapter and subclassing is the intended workaround.\n    \"\"\"\n\n    def process(self, msg, kwargs):\n        kwargs[\"extra\"] = {**(self.extra or {}), **(kwargs.get(\"extra\") or {})}\n\n        from prefect._internal.compatibility.deprecated import (\n            PrefectDeprecationWarning,\n            generate_deprecation_message,\n        )\n\n        if \"send_to_orion\" in kwargs[\"extra\"]:\n            warnings.warn(\n                generate_deprecation_message(\n                    'The \"send_to_orion\" option',\n                    start_date=\"May 2023\",\n                    help='Use \"send_to_api\" instead.',\n                ),\n                PrefectDeprecationWarning,\n                stacklevel=4,\n            )\n\n        return (msg, kwargs)\n\n    def getChild(\n        self, suffix: str, extra: Optional[Dict[str, str]] = None\n    ) -> \"PrefectLogAdapter\":\n        if extra is None:\n            extra = {}\n\n        return PrefectLogAdapter(\n            self.logger.getChild(suffix),\n            extra={\n                **self.extra,\n                **extra,\n            },\n        )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.disable_logger","title":"disable_logger","text":"

    Get a logger by name and disables it within the context manager. Upon exiting the context manager, the logger is returned to its original state.

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef disable_logger(name: str):\n    \"\"\"\n    Get a logger by name and disables it within the context manager.\n    Upon exiting the context manager, the logger is returned to its\n    original state.\n    \"\"\"\n    logger = logging.getLogger(name=name)\n\n    # determine if it's already disabled\n    base_state = logger.disabled\n    try:\n        # disable the logger\n        logger.disabled = True\n        yield\n    finally:\n        # return to base state\n        logger.disabled = base_state\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.disable_run_logger","title":"disable_run_logger","text":"

    Gets both prefect.flow_run and prefect.task_run and disables them within the context manager. Upon exiting the context manager, both loggers are returned to its original state.

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef disable_run_logger():\n    \"\"\"\n    Gets both `prefect.flow_run` and `prefect.task_run` and disables them\n    within the context manager. Upon exiting the context manager, both loggers\n    are returned to its original state.\n    \"\"\"\n    with disable_logger(\"prefect.flow_run\"), disable_logger(\"prefect.task_run\"):\n        yield\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.flow_run_logger","title":"flow_run_logger","text":"

    Create a flow run logger with the run's metadata attached.

    Additional keyword arguments can be provided to attach custom data to the log records.

    If the flow run context is available, see get_run_logger instead.

    Source code in prefect/logging/loggers.py
    def flow_run_logger(\n    flow_run: Union[\"FlowRun\", \"ClientFlowRun\"],\n    flow: Optional[\"Flow\"] = None,\n    **kwargs: str,\n):\n    \"\"\"\n    Create a flow run logger with the run's metadata attached.\n\n    Additional keyword arguments can be provided to attach custom data to the log\n    records.\n\n    If the flow run context is available, see `get_run_logger` instead.\n    \"\"\"\n    return PrefectLogAdapter(\n        get_logger(\"prefect.flow_runs\"),\n        extra={\n            **{\n                \"flow_run_name\": flow_run.name if flow_run else \"<unknown>\",\n                \"flow_run_id\": str(flow_run.id) if flow_run else \"<unknown>\",\n                \"flow_name\": flow.name if flow else \"<unknown>\",\n            },\n            **kwargs,\n        },\n    )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.get_logger","title":"get_logger cached","text":"

    Get a prefect logger. These loggers are intended for internal use within the prefect package.

    See get_run_logger for retrieving loggers for use within task or flow runs. By default, only run-related loggers are connected to the APILogHandler.

    Source code in prefect/logging/loggers.py
    @lru_cache()\ndef get_logger(name: str = None) -> logging.Logger:\n    \"\"\"\n    Get a `prefect` logger. These loggers are intended for internal use within the\n    `prefect` package.\n\n    See `get_run_logger` for retrieving loggers for use within task or flow runs.\n    By default, only run-related loggers are connected to the `APILogHandler`.\n    \"\"\"\n\n    parent_logger = logging.getLogger(\"prefect\")\n\n    if name:\n        # Append the name if given but allow explicit full names e.g. \"prefect.test\"\n        # should not become \"prefect.prefect.test\"\n        if not name.startswith(parent_logger.name + \".\"):\n            logger = parent_logger.getChild(name)\n        else:\n            logger = logging.getLogger(name)\n    else:\n        logger = parent_logger\n\n    return logger\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.get_run_logger","title":"get_run_logger","text":"

    Get a Prefect logger for the current task run or flow run.

    The logger will be named either prefect.task_runs or prefect.flow_runs. Contextual data about the run will be attached to the log records.

    These loggers are connected to the APILogHandler by default to send log records to the API.

    Parameters:

    Name Type Description Default context RunContext

    A specific context may be provided as an override. By default, the context is inferred from global state and this should not be needed.

    None **kwargs str

    Additional keyword arguments will be attached to the log records in addition to the run metadata

    {}

    Raises:

    Type Description RuntimeError

    If no context can be found

    Source code in prefect/logging/loggers.py
    def get_run_logger(\n    context: \"RunContext\" = None, **kwargs: str\n) -> Union[logging.Logger, logging.LoggerAdapter]:\n    \"\"\"\n    Get a Prefect logger for the current task run or flow run.\n\n    The logger will be named either `prefect.task_runs` or `prefect.flow_runs`.\n    Contextual data about the run will be attached to the log records.\n\n    These loggers are connected to the `APILogHandler` by default to send log records to\n    the API.\n\n    Arguments:\n        context: A specific context may be provided as an override. By default, the\n            context is inferred from global state and this should not be needed.\n        **kwargs: Additional keyword arguments will be attached to the log records in\n            addition to the run metadata\n\n    Raises:\n        RuntimeError: If no context can be found\n    \"\"\"\n    # Check for existing contexts\n    task_run_context = prefect.context.TaskRunContext.get()\n    flow_run_context = prefect.context.FlowRunContext.get()\n\n    # Apply the context override\n    if context:\n        if isinstance(context, prefect.context.FlowRunContext):\n            flow_run_context = context\n        elif isinstance(context, prefect.context.TaskRunContext):\n            task_run_context = context\n        else:\n            raise TypeError(\n                f\"Received unexpected type {type(context).__name__!r} for context. \"\n                \"Expected one of 'None', 'FlowRunContext', or 'TaskRunContext'.\"\n            )\n\n    # Determine if this is a task or flow run logger\n    if task_run_context:\n        logger = task_run_logger(\n            task_run=task_run_context.task_run,\n            task=task_run_context.task,\n            flow_run=flow_run_context.flow_run if flow_run_context else None,\n            flow=flow_run_context.flow if flow_run_context else None,\n            **kwargs,\n        )\n    elif flow_run_context:\n        logger = flow_run_logger(\n            flow_run=flow_run_context.flow_run, flow=flow_run_context.flow, **kwargs\n        )\n    elif (\n        get_logger(\"prefect.flow_run\").disabled\n        and get_logger(\"prefect.task_run\").disabled\n    ):\n        logger = logging.getLogger(\"null\")\n    else:\n        raise MissingContextError(\"There is no active flow or task run context.\")\n\n    return logger\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.patch_print","title":"patch_print","text":"

    Patches the Python builtin print method to use print_as_log

    Source code in prefect/logging/loggers.py
    @contextmanager\ndef patch_print():\n    \"\"\"\n    Patches the Python builtin `print` method to use `print_as_log`\n    \"\"\"\n    import builtins\n\n    original = builtins.print\n\n    try:\n        builtins.print = print_as_log\n        yield\n    finally:\n        builtins.print = original\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.print_as_log","title":"print_as_log","text":"

    A patch for print to send printed messages to the Prefect run logger.

    If no run is active, print will behave as if it were not patched.

    If print sends data to a file other than sys.stdout or sys.stderr, it will not be forwarded to the Prefect logger either.

    Source code in prefect/logging/loggers.py
    def print_as_log(*args, **kwargs):\n    \"\"\"\n    A patch for `print` to send printed messages to the Prefect run logger.\n\n    If no run is active, `print` will behave as if it were not patched.\n\n    If `print` sends data to a file other than `sys.stdout` or `sys.stderr`, it will\n    not be forwarded to the Prefect logger either.\n    \"\"\"\n    from prefect.context import FlowRunContext, TaskRunContext\n\n    context = TaskRunContext.get() or FlowRunContext.get()\n    if (\n        not context\n        or not context.log_prints\n        or kwargs.get(\"file\") not in {None, sys.stdout, sys.stderr}\n    ):\n        return print(*args, **kwargs)\n\n    logger = get_run_logger()\n\n    # Print to an in-memory buffer; so we do not need to implement `print`\n    buffer = io.StringIO()\n    kwargs[\"file\"] = buffer\n    print(*args, **kwargs)\n\n    # Remove trailing whitespace to prevent duplicates\n    logger.info(buffer.getvalue().rstrip())\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/logging/loggers/#prefect.logging.loggers.task_run_logger","title":"task_run_logger","text":"

    Create a task run logger with the run's metadata attached.

    Additional keyword arguments can be provided to attach custom data to the log records.

    If the task run context is available, see get_run_logger instead.

    If only the flow run context is available, it will be used for default values of flow_run and flow.

    Source code in prefect/logging/loggers.py
    def task_run_logger(\n    task_run: \"TaskRun\",\n    task: \"Task\" = None,\n    flow_run: \"FlowRun\" = None,\n    flow: \"Flow\" = None,\n    **kwargs: str,\n):\n    \"\"\"\n    Create a task run logger with the run's metadata attached.\n\n    Additional keyword arguments can be provided to attach custom data to the log\n    records.\n\n    If the task run context is available, see `get_run_logger` instead.\n\n    If only the flow run context is available, it will be used for default values\n    of `flow_run` and `flow`.\n    \"\"\"\n    if not flow_run or not flow:\n        flow_run_context = prefect.context.FlowRunContext.get()\n        if flow_run_context:\n            flow_run = flow_run or flow_run_context.flow_run\n            flow = flow or flow_run_context.flow\n\n    return PrefectLogAdapter(\n        get_logger(\"prefect.task_runs\"),\n        extra={\n            **{\n                \"task_run_id\": str(task_run.id),\n                \"flow_run_id\": str(task_run.flow_run_id),\n                \"task_run_name\": task_run.name,\n                \"task_name\": task.name if task else \"<unknown>\",\n                \"flow_run_name\": flow_run.name if flow_run else \"<unknown>\",\n                \"flow_name\": flow.name if flow else \"<unknown>\",\n            },\n            **kwargs,\n        },\n    )\n
    ","tags":["Python API","logging","loggers"]},{"location":"api-ref/prefect/packaging/base/","title":"base","text":"","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base","title":"prefect.packaging.base","text":"","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.PackageManifest","title":"PackageManifest","text":"

    Bases: BaseModel, ABC

    Describes a package.

    Source code in prefect/packaging/base.py
    @add_type_dispatch\nclass PackageManifest(BaseModel, abc.ABC):\n    \"\"\"\n    Describes a package.\n    \"\"\"\n\n    type: str\n    flow_name: str\n    flow_parameter_schema: ParameterSchema\n\n    @abc.abstractmethod\n    async def unpackage(self) -> Flow:\n        \"\"\"\n        Retrieve a flow from the package.\n        \"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.PackageManifest.unpackage","title":"unpackage abstractmethod async","text":"

    Retrieve a flow from the package.

    Source code in prefect/packaging/base.py
    @abc.abstractmethod\nasync def unpackage(self) -> Flow:\n    \"\"\"\n    Retrieve a flow from the package.\n    \"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Packager","title":"Packager","text":"

    Bases: BaseModel, ABC

    Creates a package for a flow.

    A package contains the flow and is typically stored outside of Prefect. To facilitate interaction with the package, a manifest is returned that describes how to access and use the package.

    Source code in prefect/packaging/base.py
    @add_type_dispatch\nclass Packager(BaseModel, abc.ABC):\n    \"\"\"\n    Creates a package for a flow.\n\n    A package contains the flow and is typically stored outside of Prefect. To\n    facilitate interaction with the package, a manifest is returned that describes how\n    to access and use the package.\n    \"\"\"\n\n    type: str\n\n    def base_manifest(self, flow: Flow) -> PartialModel[PackageManifest]:\n        manifest_cls = lookup_type(PackageManifest, self.type)\n        return PartialModel(\n            manifest_cls,\n            type=self.type,\n            flow_name=flow.name,\n            flow_parameter_schema=parameter_schema(flow.fn),\n        )\n\n    @abc.abstractmethod\n    async def package(self, flow: Flow) -> \"PackageManifest\":\n        \"\"\"\n        Package a flow and return a manifest describing the created package.\n        \"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Packager.package","title":"package abstractmethod async","text":"

    Package a flow and return a manifest describing the created package.

    Source code in prefect/packaging/base.py
    @abc.abstractmethod\nasync def package(self, flow: Flow) -> \"PackageManifest\":\n    \"\"\"\n    Package a flow and return a manifest describing the created package.\n    \"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Serializer","title":"Serializer","text":"

    Bases: BaseModel, Generic[D], ABC

    A serializer that can encode objects of type 'D' into bytes.

    Source code in prefect/packaging/base.py
    @add_type_dispatch\nclass Serializer(BaseModel, Generic[D], abc.ABC):\n    \"\"\"\n    A serializer that can encode objects of type 'D' into bytes.\n    \"\"\"\n\n    type: str\n\n    def dumps(self, obj: D) -> bytes:\n        \"\"\"Encode the object into a blob of bytes.\"\"\"\n\n    def loads(self, blob: bytes) -> D:\n        \"\"\"Decode the blob of bytes into an object.\"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Serializer.dumps","title":"dumps","text":"

    Encode the object into a blob of bytes.

    Source code in prefect/packaging/base.py
    def dumps(self, obj: D) -> bytes:\n    \"\"\"Encode the object into a blob of bytes.\"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/base/#prefect.packaging.base.Serializer.loads","title":"loads","text":"

    Decode the blob of bytes into an object.

    Source code in prefect/packaging/base.py
    def loads(self, blob: bytes) -> D:\n    \"\"\"Decode the blob of bytes into an object.\"\"\"\n
    ","tags":["Python API","packaging"]},{"location":"api-ref/prefect/packaging/docker/","title":"docker","text":"","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/docker/#prefect.packaging.docker","title":"prefect.packaging.docker","text":"","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/docker/#prefect.packaging.docker.DockerPackageManifest","title":"DockerPackageManifest","text":"

    Bases: PackageManifest

    Represents a flow packaged in a Docker image

    Source code in prefect/packaging/docker.py
    class DockerPackageManifest(PackageManifest):\n    \"\"\"\n    Represents a flow packaged in a Docker image\n    \"\"\"\n\n    type: Literal[\"docker\"] = \"docker\"\n\n    image: str\n    image_flow_location: str\n\n    async def unpackage(self) -> Flow:\n        return load_flow_from_script(self.image_flow_location, self.flow_name)\n
    ","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/docker/#prefect.packaging.docker.DockerPackager","title":"DockerPackager","text":"

    Bases: Packager

    This packager builds a Docker image containing the flow and the runtime environment necessary to run the flow. The resulting image is optionally pushed to a container registry, given by registry_url.

    Source code in prefect/packaging/docker.py
    class DockerPackager(Packager):\n    \"\"\"\n    This packager builds a Docker image containing the flow and the runtime environment\n    necessary to run the flow.  The resulting image is optionally pushed to a container\n    registry, given by `registry_url`.\n    \"\"\"\n\n    type: Literal[\"docker\"] = \"docker\"\n\n    base_image: Optional[str] = None\n    python_environment: Optional[Union[PythonEnvironment, CondaEnvironment]] = None\n    dockerfile: Optional[Path] = None\n    platform: Optional[str] = (None,)\n    image_flow_location: str = \"/flow.py\"\n    registry_url: Optional[AnyHttpUrl] = None\n\n    @root_validator\n    def set_default_base_image(cls, values):\n        if not values.get(\"base_image\") and not values.get(\"dockerfile\"):\n            values[\"base_image\"] = get_prefect_image_name(\n                flavor=(\n                    \"conda\"\n                    if isinstance(values.get(\"python_environment\"), CondaEnvironment)\n                    else None\n                )\n            )\n        return values\n\n    @root_validator\n    def base_image_and_dockerfile_exclusive(cls, values: Mapping[str, Any]):\n        if values.get(\"base_image\") and values.get(\"dockerfile\"):\n            raise ValueError(\n                \"Either `base_image` or `dockerfile` should be provided, but not both\"\n            )\n        return values\n\n    @root_validator\n    def default_python_environment(cls, values: Mapping[str, Any]):\n        if values.get(\"base_image\") and not values.get(\"python_environment\"):\n            values[\"python_environment\"] = PythonEnvironment.from_environment()\n        return values\n\n    @validator(\"registry_url\", pre=True)\n    def ensure_registry_url_is_prefixed(cls, value):\n        if isinstance(value, str):\n            if \"://\" not in value:\n                return \"https://\" + value\n        return value\n\n    async def package(self, flow: Flow) -> DockerPackageManifest:\n        \"\"\"\n        Package a flow as a Docker image and, optionally, push it to a registry\n        \"\"\"\n        image_reference = await self._build_image(flow)\n\n        if self.registry_url:\n            image_name = f\"{slugify(flow.name)}\"\n            image_reference = await run_sync_in_worker_thread(\n                push_image, image_reference, self.registry_url, image_name\n            )\n\n        return self.base_manifest(flow).finalize(\n            image=image_reference, image_flow_location=self.image_flow_location\n        )\n\n    async def _build_image(self, flow: Flow) -> str:\n        if self.dockerfile:\n            return await self._build_from_dockerfile()\n        return await self._build_from_base_image(flow)\n\n    async def _build_from_dockerfile(self) -> str:\n        context = self.dockerfile.resolve().parent\n        dockerfile = self.dockerfile.relative_to(context)\n        return await run_sync_in_worker_thread(\n            build_image,\n            platform=self.platform,\n            context=context,\n            dockerfile=str(dockerfile),\n        )\n\n    async def _build_from_base_image(self, flow: Flow) -> str:\n        with ImageBuilder(\n            base_image=self.base_image, platform=self.platform\n        ) as builder:\n            for command in self.python_environment.install_commands():\n                builder.add_line(to_run_command(command))\n\n            source_info = json.loads(SourceSerializer().dumps(flow))\n\n            builder.write_text(source_info[\"source\"], self.image_flow_location)\n\n            return await run_sync_in_worker_thread(\n                builder.build, stream_progress_to=sys.stdout\n            )\n
    ","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/docker/#prefect.packaging.docker.DockerPackager.package","title":"package async","text":"

    Package a flow as a Docker image and, optionally, push it to a registry

    Source code in prefect/packaging/docker.py
    async def package(self, flow: Flow) -> DockerPackageManifest:\n    \"\"\"\n    Package a flow as a Docker image and, optionally, push it to a registry\n    \"\"\"\n    image_reference = await self._build_image(flow)\n\n    if self.registry_url:\n        image_name = f\"{slugify(flow.name)}\"\n        image_reference = await run_sync_in_worker_thread(\n            push_image, image_reference, self.registry_url, image_name\n        )\n\n    return self.base_manifest(flow).finalize(\n        image=image_reference, image_flow_location=self.image_flow_location\n    )\n
    ","tags":["Python API","packaging","Docker"]},{"location":"api-ref/prefect/packaging/file/","title":"file","text":"","tags":["Python API","packaging","file"]},{"location":"api-ref/prefect/packaging/file/#prefect.packaging.file","title":"prefect.packaging.file","text":"","tags":["Python API","packaging","file"]},{"location":"api-ref/prefect/packaging/file/#prefect.packaging.file.FilePackager","title":"FilePackager","text":"

    Bases: Packager

    This packager stores the flow as a single file.

    By default, the file is the source code of the module the flow is defined in. Alternative serialization modes are available in prefect.packaging.serializers.

    Source code in prefect/packaging/file.py
    class FilePackager(Packager):\n    \"\"\"\n    This packager stores the flow as a single file.\n\n    By default, the file is the source code of the module the flow is defined in.\n    Alternative serialization modes are available in `prefect.packaging.serializers`.\n    \"\"\"\n\n    type: Literal[\"file\"] = \"file\"\n    serializer: Serializer = Field(default_factory=SourceSerializer)\n    filesystem: WritableFileSystem = Field(\n        default_factory=lambda: LocalFileSystem(\n            basepath=PREFECT_HOME.value() / \"storage\"\n        )\n    )\n\n    @inject_client\n    async def package(self, flow: Flow, client: \"PrefectClient\") -> FilePackageManifest:\n        content = self.serializer.dumps(flow)\n        key = stable_hash(content)\n\n        await self.filesystem.write_path(key, content)\n\n        filesystem_id = (\n            self.filesystem._block_document_id\n            or await self.filesystem._save(is_anonymous=True)\n        )\n\n        return self.base_manifest(flow).finalize(\n            serializer=self.serializer,\n            filesystem_id=filesystem_id,\n            key=key,\n        )\n
    ","tags":["Python API","packaging","file"]},{"location":"api-ref/prefect/packaging/serializers/","title":"serializers","text":"","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers","title":"prefect.packaging.serializers","text":"","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.ImportSerializer","title":"ImportSerializer","text":"

    Bases: Serializer

    Serializes objects by storing their importable path.

    Source code in prefect/packaging/serializers.py
    class ImportSerializer(Serializer):\n    \"\"\"\n    Serializes objects by storing their importable path.\n    \"\"\"\n\n    type: Literal[\"import\"] = \"import\"\n\n    def dumps(self, obj: Any) -> bytes:\n        return to_qualified_name(obj).encode()\n\n    def loads(self, blob: bytes) -> Any:\n        return from_qualified_name(blob.decode())\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.PickleSerializer","title":"PickleSerializer","text":"

    Bases: Serializer

    Serializes objects using the pickle protocol.

    If using cloudpickle, you may specify a list of 'pickle_modules'. These modules will be serialized by value instead of by reference, which means they do not have to be installed in the runtime location. This is especially useful for serializing objects that rely on local packages.

    Wraps pickles in base64 for safe transmission.

    Source code in prefect/packaging/serializers.py
    class PickleSerializer(Serializer):\n    \"\"\"\n    Serializes objects using the pickle protocol.\n\n    If using cloudpickle, you may specify a list of 'pickle_modules'. These modules will\n    be serialized by value instead of by reference, which means they do not have to be\n    installed in the runtime location. This is especially useful for serializing objects\n    that rely on local packages.\n\n    Wraps pickles in base64 for safe transmission.\n    \"\"\"\n\n    type: Literal[\"pickle\"] = \"pickle\"\n\n    picklelib: str = \"cloudpickle\"\n    picklelib_version: str = None\n    pickle_modules: List[str] = pydantic.Field(default_factory=list)\n\n    @pydantic.validator(\"picklelib\")\n    def check_picklelib(cls, value):\n        \"\"\"\n        Check that the given pickle library is importable and has dumps/loads methods.\n        \"\"\"\n        try:\n            pickler = from_qualified_name(value)\n        except (ImportError, AttributeError) as exc:\n            raise ValueError(\n                f\"Failed to import requested pickle library: {value!r}.\"\n            ) from exc\n\n        if not callable(getattr(pickler, \"dumps\", None)):\n            raise ValueError(\n                f\"Pickle library at {value!r} does not have a 'dumps' method.\"\n            )\n\n        if not callable(getattr(pickler, \"loads\", None)):\n            raise ValueError(\n                f\"Pickle library at {value!r} does not have a 'loads' method.\"\n            )\n\n        return value\n\n    @pydantic.root_validator\n    def check_picklelib_version(cls, values):\n        \"\"\"\n        Infers a default value for `picklelib_version` if null or ensures it matches\n        the version retrieved from the `pickelib`.\n        \"\"\"\n        picklelib = values.get(\"picklelib\")\n        picklelib_version = values.get(\"picklelib_version\")\n\n        if not picklelib:\n            raise ValueError(\"Unable to check version of unrecognized picklelib module\")\n\n        pickler = from_qualified_name(picklelib)\n        pickler_version = getattr(pickler, \"__version__\", None)\n\n        if not picklelib_version:\n            values[\"picklelib_version\"] = pickler_version\n        elif picklelib_version != pickler_version:\n            warnings.warn(\n                (\n                    f\"Mismatched {picklelib!r} versions. Found {pickler_version} in the\"\n                    f\" environment but {picklelib_version} was requested. This may\"\n                    \" cause the serializer to fail.\"\n                ),\n                RuntimeWarning,\n                stacklevel=3,\n            )\n\n        return values\n\n    @pydantic.root_validator\n    def check_picklelib_and_modules(cls, values):\n        \"\"\"\n        Prevents modules from being specified if picklelib is not cloudpickle\n        \"\"\"\n        if values.get(\"picklelib\") != \"cloudpickle\" and values.get(\"pickle_modules\"):\n            raise ValueError(\n                \"`pickle_modules` cannot be used without 'cloudpickle'. Got\"\n                f\" {values.get('picklelib')!r}.\"\n            )\n        return values\n\n    def dumps(self, obj: Any) -> bytes:\n        pickler = from_qualified_name(self.picklelib)\n\n        for module in self.pickle_modules:\n            pickler.register_pickle_by_value(from_qualified_name(module))\n\n        blob = pickler.dumps(obj)\n\n        for module in self.pickle_modules:\n            # Restore the pickler settings\n            pickler.unregister_pickle_by_value(from_qualified_name(module))\n\n        return base64.encodebytes(blob)\n\n    def loads(self, blob: bytes) -> Any:\n        pickler = from_qualified_name(self.picklelib)\n        return pickler.loads(base64.decodebytes(blob))\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.PickleSerializer.check_picklelib","title":"check_picklelib","text":"

    Check that the given pickle library is importable and has dumps/loads methods.

    Source code in prefect/packaging/serializers.py
    @pydantic.validator(\"picklelib\")\ndef check_picklelib(cls, value):\n    \"\"\"\n    Check that the given pickle library is importable and has dumps/loads methods.\n    \"\"\"\n    try:\n        pickler = from_qualified_name(value)\n    except (ImportError, AttributeError) as exc:\n        raise ValueError(\n            f\"Failed to import requested pickle library: {value!r}.\"\n        ) from exc\n\n    if not callable(getattr(pickler, \"dumps\", None)):\n        raise ValueError(\n            f\"Pickle library at {value!r} does not have a 'dumps' method.\"\n        )\n\n    if not callable(getattr(pickler, \"loads\", None)):\n        raise ValueError(\n            f\"Pickle library at {value!r} does not have a 'loads' method.\"\n        )\n\n    return value\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.PickleSerializer.check_picklelib_and_modules","title":"check_picklelib_and_modules","text":"

    Prevents modules from being specified if picklelib is not cloudpickle

    Source code in prefect/packaging/serializers.py
    @pydantic.root_validator\ndef check_picklelib_and_modules(cls, values):\n    \"\"\"\n    Prevents modules from being specified if picklelib is not cloudpickle\n    \"\"\"\n    if values.get(\"picklelib\") != \"cloudpickle\" and values.get(\"pickle_modules\"):\n        raise ValueError(\n            \"`pickle_modules` cannot be used without 'cloudpickle'. Got\"\n            f\" {values.get('picklelib')!r}.\"\n        )\n    return values\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.PickleSerializer.check_picklelib_version","title":"check_picklelib_version","text":"

    Infers a default value for picklelib_version if null or ensures it matches the version retrieved from the pickelib.

    Source code in prefect/packaging/serializers.py
    @pydantic.root_validator\ndef check_picklelib_version(cls, values):\n    \"\"\"\n    Infers a default value for `picklelib_version` if null or ensures it matches\n    the version retrieved from the `pickelib`.\n    \"\"\"\n    picklelib = values.get(\"picklelib\")\n    picklelib_version = values.get(\"picklelib_version\")\n\n    if not picklelib:\n        raise ValueError(\"Unable to check version of unrecognized picklelib module\")\n\n    pickler = from_qualified_name(picklelib)\n    pickler_version = getattr(pickler, \"__version__\", None)\n\n    if not picklelib_version:\n        values[\"picklelib_version\"] = pickler_version\n    elif picklelib_version != pickler_version:\n        warnings.warn(\n            (\n                f\"Mismatched {picklelib!r} versions. Found {pickler_version} in the\"\n                f\" environment but {picklelib_version} was requested. This may\"\n                \" cause the serializer to fail.\"\n            ),\n            RuntimeWarning,\n            stacklevel=3,\n        )\n\n    return values\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/packaging/serializers/#prefect.packaging.serializers.SourceSerializer","title":"SourceSerializer","text":"

    Bases: Serializer

    Serializes objects by retrieving the source code of the module they are defined in.

    Creates a JSON blob with keys

    Deserialization requires the code to run with exec.

    Source code in prefect/packaging/serializers.py
    class SourceSerializer(Serializer):\n    \"\"\"\n    Serializes objects by retrieving the source code of the module they are defined in.\n\n    Creates a JSON blob with keys:\n        source: The source code\n        file_name: The name of the file the source was in\n        symbol_name: The name of the object to extract from the source code\n\n    Deserialization requires the code to run with `exec`.\n    \"\"\"\n\n    type: Literal[\"source\"] = \"source\"\n\n    def dumps(self, obj: Any) -> bytes:\n        module = inspect.getmodule(obj)\n\n        if module is None:\n            raise ValueError(f\"Cannot determine source module for object: {obj!r}.\")\n\n        if not getattr(module, \"__file__\", None):\n            raise ValueError(\n                f\"Found module {module!r} without source code file while serializing \"\n                f\"object: {obj!r}.\"\n            )\n\n        source = inspect.getsource(module)\n\n        return json.dumps(\n            {\n                \"source\": source,\n                \"file_name\": os.path.basename(module.__file__),\n                \"symbol_name\": obj.__name__,\n            }\n        ).encode()\n\n    def loads(self, blob: bytes) -> Any:\n        document = json.loads(blob)\n        if not isinstance(document, dict) or set(document.keys()) != {\n            \"source\",\n            \"file_name\",\n            \"symbol_name\",\n        }:\n            raise ValueError(\n                \"Invalid serialized data. \"\n                \"Expected dictionary with keys 'source', 'file_name', and \"\n                \"'symbol_name'. \"\n                f\"Got: {document}\"\n            )\n\n        with TemporaryDirectory() as tmpdir:\n            temp_script = Path(tmpdir) / document[\"file_name\"]\n            temp_script.write_text(document[\"source\"])\n            module = load_script_as_module(str(temp_script))\n\n        return getattr(module, document[\"symbol_name\"])\n
    ","tags":["Python API","packaging","serializers"]},{"location":"api-ref/prefect/runner/runner/","title":"runner","text":"","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner","title":"prefect.runner.runner","text":"

    Runners are responsible for managing the execution of deployments created and managed by either flow.serve or the serve utility.

    Example
    import time\nfrom prefect import flow, serve\n\n\n@flow\ndef slow_flow(sleep: int = 60):\n    \"Sleepy flow - sleeps the provided amount of time (in seconds).\"\n    time.sleep(sleep)\n\n\n@flow\ndef fast_flow():\n    \"Fastest flow this side of the Mississippi.\"\n    return\n\n\nif __name__ == \"__main__\":\n    slow_deploy = slow_flow.to_deployment(name=\"sleeper\", interval=45)\n    fast_deploy = fast_flow.to_deployment(name=\"fast\")\n\n    # serve generates a Runner instance\n    serve(slow_deploy, fast_deploy)\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner","title":"Runner","text":"Source code in prefect/runner/runner.py
    class Runner:\n    def __init__(\n        self,\n        name: Optional[str] = None,\n        query_seconds: Optional[float] = None,\n        prefetch_seconds: float = 10,\n        limit: Optional[int] = None,\n        pause_on_shutdown: bool = True,\n        webserver: bool = False,\n    ):\n        \"\"\"\n        Responsible for managing the execution of remotely initiated flow runs.\n\n        Args:\n            name: The name of the runner. If not provided, a random one\n                will be generated. If provided, it cannot contain '/' or '%'.\n            query_seconds: The number of seconds to wait between querying for\n                scheduled flow runs; defaults to `PREFECT_RUNNER_POLL_FREQUENCY`\n            prefetch_seconds: The number of seconds to prefetch flow runs for.\n            limit: The maximum number of flow runs this runner should be running at\n            pause_on_shutdown: A boolean for whether or not to automatically pause\n                deployment schedules on shutdown; defaults to `True`\n            webserver: a boolean flag for whether to start a webserver for this runner\n\n        Examples:\n            Set up a Runner to manage the execute of scheduled flow runs for two flows:\n                ```python\n                from prefect import flow, Runner\n\n                @flow\n                def hello_flow(name):\n                    print(f\"hello {name}\")\n\n                @flow\n                def goodbye_flow(name):\n                    print(f\"goodbye {name}\")\n\n                if __name__ == \"__main__\"\n                    runner = Runner(name=\"my-runner\")\n\n                    # Will be runnable via the API\n                    runner.add_flow(hello_flow)\n\n                    # Run on a cron schedule\n                    runner.add_flow(goodbye_flow, schedule={\"cron\": \"0 * * * *\"})\n\n                    runner.start()\n                ```\n        \"\"\"\n        if name and (\"/\" in name or \"%\" in name):\n            raise ValueError(\"Runner name cannot contain '/' or '%'\")\n        self.name = Path(name).stem if name is not None else f\"runner-{uuid4()}\"\n        self._logger = get_logger(\"runner\")\n\n        self.started = False\n        self.stopping = False\n        self.pause_on_shutdown = pause_on_shutdown\n        self.limit = limit or PREFECT_RUNNER_PROCESS_LIMIT.value()\n        self.webserver = webserver\n\n        self.query_seconds = query_seconds or PREFECT_RUNNER_POLL_FREQUENCY.value()\n        self._prefetch_seconds = prefetch_seconds\n\n        self._runs_task_group: anyio.abc.TaskGroup = anyio.create_task_group()\n        self._loops_task_group: anyio.abc.TaskGroup = anyio.create_task_group()\n\n        self._limiter: Optional[anyio.CapacityLimiter] = anyio.CapacityLimiter(\n            self.limit\n        )\n        self._client = get_client()\n        self._submitting_flow_run_ids = set()\n        self._cancelling_flow_run_ids = set()\n        self._scheduled_task_scopes = set()\n        self._deployment_ids: Set[UUID] = set()\n        self._flow_run_process_map = dict()\n\n        self._tmp_dir: Path = (\n            Path(tempfile.gettempdir()) / \"runner_storage\" / str(uuid4())\n        )\n        self._storage_objs: List[RunnerStorage] = []\n        self._deployment_storage_map: Dict[UUID, RunnerStorage] = {}\n\n    @sync_compatible\n    async def add_deployment(\n        self,\n        deployment: RunnerDeployment,\n    ) -> UUID:\n        \"\"\"\n        Registers the deployment with the Prefect API and will monitor for work once\n        the runner is started.\n\n        Args:\n            deployment: A deployment for the runner to register.\n        \"\"\"\n        deployment_id = await deployment.apply()\n        storage = deployment.storage\n        if storage is not None:\n            storage = await self._add_storage(storage)\n            self._deployment_storage_map[deployment_id] = storage\n        self._deployment_ids.add(deployment_id)\n\n        return deployment_id\n\n    @sync_compatible\n    async def add_flow(\n        self,\n        flow: Flow,\n        name: str = None,\n        interval: Optional[Union[int, float, datetime.timedelta]] = None,\n        cron: Optional[str] = None,\n        rrule: Optional[str] = None,\n        schedule: Optional[SCHEDULE_TYPES] = None,\n        is_schedule_active: Optional[bool] = None,\n        parameters: Optional[dict] = None,\n        triggers: Optional[List[DeploymentTrigger]] = None,\n        description: Optional[str] = None,\n        tags: Optional[List[str]] = None,\n        version: Optional[str] = None,\n        enforce_parameter_schema: bool = False,\n    ) -> UUID:\n        \"\"\"\n        Provides a flow to the runner to be run based on the provided configuration.\n\n        Will create a deployment for the provided flow and register the deployment\n        with the runner.\n\n        Args:\n            flow: A flow for the runner to run.\n            name: The name to give the created deployment. Will default to the name\n                of the runner.\n            interval: An interval on which to execute the current flow. Accepts either a number\n                or a timedelta object. If a number is given, it will be interpreted as seconds.\n            cron: A cron schedule of when to execute runs of this flow.\n            rrule: An rrule schedule of when to execute runs of this flow.\n            schedule: A schedule object of when to execute runs of this flow. Used for\n                advanced scheduling options like timezone.\n            is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n                not provided when creating a deployment, the schedule will be set as active. If not\n                provided when updating a deployment, the schedule's activation will not be changed.\n            triggers: A list of triggers that should kick of a run of this flow.\n            parameters: A dictionary of default parameter values to pass to runs of this flow.\n            description: A description for the created deployment. Defaults to the flow's\n                description if not provided.\n            tags: A list of tags to associate with the created deployment for organizational\n                purposes.\n            version: A version for the created deployment. Defaults to the flow's version.\n        \"\"\"\n        api = PREFECT_API_URL.value()\n        if any([interval, cron, rrule]) and not api:\n            self._logger.warning(\n                \"Cannot schedule flows on an ephemeral server; run `prefect server\"\n                \" start` to start the scheduler.\"\n            )\n        name = self.name if name is None else name\n\n        deployment = await flow.to_deployment(\n            name=name,\n            interval=interval,\n            cron=cron,\n            rrule=rrule,\n            schedule=schedule,\n            is_schedule_active=is_schedule_active,\n            triggers=triggers,\n            parameters=parameters,\n            description=description,\n            tags=tags,\n            version=version,\n            enforce_parameter_schema=enforce_parameter_schema,\n        )\n        return await self.add_deployment(deployment)\n\n    @sync_compatible\n    async def _add_storage(self, storage: RunnerStorage) -> RunnerStorage:\n        \"\"\"\n        Adds a storage object to the runner. The storage object will be used to pull\n        code to the runner's working directory before the runner starts.\n\n        Args:\n            storage: The storage object to add to the runner.\n        Returns:\n            The updated storage object that was added to the runner.\n        \"\"\"\n        if storage not in self._storage_objs:\n            storage_copy = deepcopy(storage)\n            storage_copy.set_base_path(self._tmp_dir)\n\n            self._logger.debug(\n                f\"Adding storage {storage_copy!r} to runner at\"\n                f\" {str(storage_copy.destination)!r}\"\n            )\n            self._storage_objs.append(storage_copy)\n\n            return storage_copy\n        else:\n            return next(s for s in self._storage_objs if s == storage)\n\n    def handle_sigterm(self, signum, frame):\n        \"\"\"\n        Gracefully shuts down the runner when a SIGTERM is received.\n        \"\"\"\n        self._logger.info(\"SIGTERM received, initiating graceful shutdown...\")\n        from_sync.call_in_loop_thread(create_call(self.stop))\n\n        sys.exit(0)\n\n    @sync_compatible\n    async def start(\n        self, run_once: bool = False, webserver: Optional[bool] = None\n    ) -> None:\n        \"\"\"\n        Starts a runner.\n\n        The runner will begin monitoring for and executing any scheduled work for all added flows.\n\n        Args:\n            run_once: If True, the runner will through one query loop and then exit.\n            webserver: a boolean for whether to start a webserver for this runner. If provided,\n                overrides the default on the runner\n\n        Examples:\n            Initialize a Runner, add two flows, and serve them by starting the Runner:\n\n            ```python\n            from prefect import flow, Runner\n\n            @flow\n            def hello_flow(name):\n                print(f\"hello {name}\")\n\n            @flow\n            def goodbye_flow(name):\n                print(f\"goodbye {name}\")\n\n            if __name__ == \"__main__\"\n                runner = Runner(name=\"my-runner\")\n\n                # Will be runnable via the API\n                runner.add_flow(hello_flow)\n\n                # Run on a cron schedule\n                runner.add_flow(goodbye_flow, schedule={\"cron\": \"0 * * * *\"})\n\n                runner.start()\n            ```\n        \"\"\"\n        _register_signal(signal.SIGTERM, self.handle_sigterm)\n\n        webserver = webserver if webserver is not None else self.webserver\n\n        if webserver:\n            # we'll start the ASGI server in a separate thread so that\n            # uvicorn does not block the main thread\n            server_thread = threading.Thread(\n                name=\"runner-server-thread\",\n                target=partial(\n                    start_webserver,\n                    runner=self,\n                ),\n                daemon=True,\n            )\n            server_thread.start()\n\n        async with self as runner:\n            async with self._loops_task_group as tg:\n                for storage in self._storage_objs:\n                    if storage.pull_interval:\n                        tg.start_soon(\n                            partial(\n                                critical_service_loop,\n                                workload=storage.pull_code,\n                                interval=storage.pull_interval,\n                                run_once=run_once,\n                                jitter_range=0.3,\n                            )\n                        )\n                    else:\n                        tg.start_soon(storage.pull_code)\n                tg.start_soon(\n                    partial(\n                        critical_service_loop,\n                        workload=runner._get_and_submit_flow_runs,\n                        interval=self.query_seconds,\n                        run_once=run_once,\n                        jitter_range=0.3,\n                    )\n                )\n                tg.start_soon(\n                    partial(\n                        critical_service_loop,\n                        workload=runner._check_for_cancelled_flow_runs,\n                        interval=self.query_seconds * 2,\n                        run_once=run_once,\n                        jitter_range=0.3,\n                    )\n                )\n\n    async def cancel_all(self):\n        runs_to_cancel = []\n\n        # done to avoid dictionary size changing during iteration\n        for flow_run_id, info in self._flow_run_process_map.items():\n            runs_to_cancel.append(info[\"flow_run\"])\n        if runs_to_cancel:\n            for run in runs_to_cancel:\n                try:\n                    await self._cancel_run(run, state_msg=\"Runner is shutting down.\")\n                except Exception:\n                    self._logger.exception(\n                        f\"Exception encountered while cancelling {run.id}\",\n                        exc_info=True,\n                    )\n\n    @sync_compatible\n    async def stop(self):\n        \"\"\"Stops the runner's polling cycle.\"\"\"\n        if not self.started:\n            raise RuntimeError(\n                \"Runner has not yet started. Please start the runner by calling\"\n                \" .start()\"\n            )\n\n        self.started = False\n        self.stopping = True\n        await self.cancel_all()\n        try:\n            self._loops_task_group.cancel_scope.cancel()\n        except Exception:\n            self._logger.exception(\n                \"Exception encountered while shutting down\", exc_info=True\n            )\n\n    async def execute_flow_run(self, flow_run_id: UUID):\n        \"\"\"\n        Executes a single flow run with the given ID.\n\n        Execution will wait to monitor for cancellation requests. Exits once\n        the flow run process has exited.\n        \"\"\"\n        self.pause_on_shutdown = False\n        async with self:\n            if not self._acquire_limit_slot(flow_run_id):\n                return\n\n            async with anyio.create_task_group() as tg:\n                with anyio.CancelScope():\n                    self._submitting_flow_run_ids.add(flow_run_id)\n                    flow_run = await self._client.read_flow_run(flow_run_id)\n\n                    pid = await self._runs_task_group.start(\n                        self._submit_run_and_capture_errors, flow_run\n                    )\n\n                    self._flow_run_process_map[flow_run.id] = dict(\n                        pid=pid, flow_run=flow_run\n                    )\n\n                    # We want this loop to stop when the flow run process exits\n                    # so we'll check if the flow run process is still alive on\n                    # each iteration and cancel the task group if it is not.\n                    workload = partial(\n                        self._check_for_cancelled_flow_runs,\n                        should_stop=lambda: not self._flow_run_process_map,\n                        on_stop=tg.cancel_scope.cancel,\n                    )\n\n                    tg.start_soon(\n                        partial(\n                            critical_service_loop,\n                            workload=workload,\n                            interval=self.query_seconds,\n                            jitter_range=0.3,\n                        )\n                    )\n\n    def _get_flow_run_logger(self, flow_run: \"FlowRun\") -> PrefectLogAdapter:\n        return flow_run_logger(flow_run=flow_run).getChild(\n            \"runner\",\n            extra={\n                \"runner_name\": self.name,\n            },\n        )\n\n    async def _run_process(\n        self,\n        flow_run: \"FlowRun\",\n        task_status: Optional[anyio.abc.TaskStatus] = None,\n    ):\n        \"\"\"\n        Runs the given flow run in a subprocess.\n\n        Args:\n            flow_run: Flow run to execute via process. The ID of this flow run\n                is stored in the PREFECT__FLOW_RUN_ID environment variable to\n                allow the engine to retrieve the corresponding flow's code and\n                begin execution.\n            task_status: anyio task status used to send a message to the caller\n                than the flow run process has started.\n        \"\"\"\n        command = f\"{shlex.quote(sys.executable)} -m prefect.engine\"\n\n        flow_run_logger = self._get_flow_run_logger(flow_run)\n\n        # We must add creationflags to a dict so it is only passed as a function\n        # parameter on Windows, because the presence of creationflags causes\n        # errors on Unix even if set to None\n        kwargs: Dict[str, object] = {}\n        if sys.platform == \"win32\":\n            kwargs[\"creationflags\"] = subprocess.CREATE_NEW_PROCESS_GROUP\n\n        _use_threaded_child_watcher()\n        flow_run_logger.info(\"Opening process...\")\n\n        env = get_current_settings().to_environment_variables(exclude_unset=True)\n        env.update(\n            {\n                \"PREFECT__FLOW_RUN_ID\": str(flow_run.id),\n                \"PREFECT__STORAGE_BASE_PATH\": str(self._tmp_dir),\n                \"PREFECT__ENABLE_CANCELLATION_AND_CRASHED_HOOKS\": \"false\",\n            }\n        )\n        env.update(**os.environ)  # is this really necessary??\n\n        storage = self._deployment_storage_map.get(flow_run.deployment_id)\n        if storage and storage.pull_interval:\n            # perform an adhoc pull of code before running the flow if an\n            # adhoc pull hasn't been performed in the last pull_interval\n            # TODO: Explore integrating this behavior with global concurrency.\n            last_adhoc_pull = getattr(storage, \"last_adhoc_pull\", None)\n            if (\n                last_adhoc_pull is None\n                or last_adhoc_pull\n                < datetime.datetime.now()\n                - datetime.timedelta(seconds=storage.pull_interval)\n            ):\n                self._logger.debug(\n                    \"Performing adhoc pull of code for flow run %s with storage %r\",\n                    flow_run.id,\n                    storage,\n                )\n                await storage.pull_code()\n                setattr(storage, \"last_adhoc_pull\", datetime.datetime.now())\n\n        process = await run_process(\n            shlex.split(command),\n            stream_output=True,\n            task_status=task_status,\n            env=env,\n            **kwargs,\n            cwd=storage.destination if storage else None,\n        )\n\n        # Use the pid for display if no name was given\n\n        if process.returncode:\n            help_message = None\n            level = logging.ERROR\n            if process.returncode == -9:\n                level = logging.INFO\n                help_message = (\n                    \"This indicates that the process exited due to a SIGKILL signal. \"\n                    \"Typically, this is either caused by manual cancellation or \"\n                    \"high memory usage causing the operating system to \"\n                    \"terminate the process.\"\n                )\n            if process.returncode == -15:\n                level = logging.INFO\n                help_message = (\n                    \"This indicates that the process exited due to a SIGTERM signal. \"\n                    \"Typically, this is caused by manual cancellation.\"\n                )\n            elif process.returncode == 247:\n                help_message = (\n                    \"This indicates that the process was terminated due to high \"\n                    \"memory usage.\"\n                )\n            elif (\n                sys.platform == \"win32\" and process.returncode == STATUS_CONTROL_C_EXIT\n            ):\n                level = logging.INFO\n                help_message = (\n                    \"Process was terminated due to a Ctrl+C or Ctrl+Break signal. \"\n                    \"Typically, this is caused by manual cancellation.\"\n                )\n\n            flow_run_logger.log(\n                level,\n                f\"Process for flow run {flow_run.name!r} exited with status code:\"\n                f\" {process.returncode}\"\n                + (f\"; {help_message}\" if help_message else \"\"),\n            )\n        else:\n            flow_run_logger.info(\n                f\"Process for flow run {flow_run.name!r} exited cleanly.\"\n            )\n\n        return process.returncode\n\n    async def _kill_process(\n        self,\n        pid: int,\n        grace_seconds: int = 30,\n    ):\n        \"\"\"\n        Kills a given flow run process.\n\n        Args:\n            pid: ID of the process to kill\n            grace_seconds: Number of seconds to wait for the process to end.\n        \"\"\"\n        # In a non-windows environment first send a SIGTERM, then, after\n        # `grace_seconds` seconds have passed subsequent send SIGKILL. In\n        # Windows we use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        if sys.platform == \"win32\":\n            try:\n                os.kill(pid, signal.CTRL_BREAK_EVENT)\n            except (ProcessLookupError, WindowsError):\n                raise RuntimeError(\n                    f\"Unable to kill process {pid!r}: The process was not found.\"\n                )\n        else:\n            try:\n                os.kill(pid, signal.SIGTERM)\n            except ProcessLookupError:\n                raise RuntimeError(\n                    f\"Unable to kill process {pid!r}: The process was not found.\"\n                )\n\n            # Throttle how often we check if the process is still alive to keep\n            # from making too many system calls in a short period of time.\n            check_interval = max(grace_seconds / 10, 1)\n\n            with anyio.move_on_after(grace_seconds):\n                while True:\n                    await anyio.sleep(check_interval)\n\n                    # Detect if the process is still alive. If not do an early\n                    # return as the process respected the SIGTERM from above.\n                    try:\n                        os.kill(pid, 0)\n                    except ProcessLookupError:\n                        return\n\n            try:\n                os.kill(pid, signal.SIGKILL)\n            except OSError:\n                # We shouldn't ever end up here, but it's possible that the\n                # process ended right after the check above.\n                return\n\n    async def _pause_schedules(self):\n        \"\"\"\n        Pauses all deployment schedules.\n        \"\"\"\n        self._logger.info(\"Pausing schedules for all deployments...\")\n        for deployment_id in self._deployment_ids:\n            self._logger.debug(f\"Pausing schedule for deployment '{deployment_id}'\")\n            await self._client.update_schedule(deployment_id, active=False)\n        self._logger.info(\"All deployment schedules have been paused!\")\n\n    async def _get_and_submit_flow_runs(self):\n        if self.stopping:\n            return\n        runs_response = await self._get_scheduled_flow_runs()\n        self.last_polled = pendulum.now(\"UTC\")\n        return await self._submit_scheduled_flow_runs(flow_run_response=runs_response)\n\n    async def _check_for_cancelled_flow_runs(\n        self, should_stop: Callable = lambda: False, on_stop: Callable = lambda: None\n    ):\n        \"\"\"\n        Checks for flow runs with CANCELLING a cancelling state and attempts to\n        cancel them.\n\n        Args:\n            should_stop: A callable that returns a boolean indicating whether or not\n                the runner should stop checking for cancelled flow runs.\n            on_stop: A callable that is called when the runner should stop checking\n                for cancelled flow runs.\n        \"\"\"\n        if self.stopping:\n            return\n        if not self.started:\n            raise RuntimeError(\n                \"Runner is not set up. Please make sure you are running this runner \"\n                \"as an async context manager.\"\n            )\n\n        if should_stop():\n            self._logger.debug(\n                \"Runner has no active flow runs or deployments. Sending message to loop\"\n                \" service that no further cancellation checks are needed.\"\n            )\n            on_stop()\n\n        self._logger.debug(\"Checking for cancelled flow runs...\")\n\n        named_cancelling_flow_runs = await self._client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLED]),\n                    name=FlowRunFilterStateName(any_=[\"Cancelling\"]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(\n                    any_=list(\n                        self._flow_run_process_map.keys()\n                        - self._cancelling_flow_run_ids\n                    )\n                ),\n            ),\n        )\n\n        typed_cancelling_flow_runs = await self._client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLING]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(\n                    any_=list(\n                        self._flow_run_process_map.keys()\n                        - self._cancelling_flow_run_ids\n                    )\n                ),\n            ),\n        )\n\n        cancelling_flow_runs = named_cancelling_flow_runs + typed_cancelling_flow_runs\n\n        if cancelling_flow_runs:\n            self._logger.info(\n                f\"Found {len(cancelling_flow_runs)} flow runs awaiting cancellation.\"\n            )\n\n        for flow_run in cancelling_flow_runs:\n            self._cancelling_flow_run_ids.add(flow_run.id)\n            self._runs_task_group.start_soon(self._cancel_run, flow_run)\n\n        return cancelling_flow_runs\n\n    async def _cancel_run(self, flow_run: \"FlowRun\", state_msg: Optional[str] = None):\n        run_logger = self._get_flow_run_logger(flow_run)\n\n        pid = self._flow_run_process_map.get(flow_run.id, {}).get(\"pid\")\n        if not pid:\n            await self._run_on_cancellation_hooks(flow_run, flow_run.state)\n            await self._mark_flow_run_as_cancelled(\n                flow_run,\n                state_updates={\n                    \"message\": (\n                        \"Could not find process ID for flow run\"\n                        \" and cancellation cannot be guaranteed.\"\n                    )\n                },\n            )\n            return\n\n        try:\n            await self._kill_process(pid)\n        except RuntimeError as exc:\n            self._logger.warning(f\"{exc} Marking flow run as cancelled.\")\n            await self._run_on_cancellation_hooks(flow_run, flow_run.state)\n            await self._mark_flow_run_as_cancelled(flow_run)\n        except Exception:\n            run_logger.exception(\n                \"Encountered exception while killing process for flow run \"\n                f\"'{flow_run.id}'. Flow run may not be cancelled.\"\n            )\n            # We will try again on generic exceptions\n            self._cancelling_flow_run_ids.remove(flow_run.id)\n        else:\n            await self._run_on_cancellation_hooks(flow_run, flow_run.state)\n            await self._mark_flow_run_as_cancelled(\n                flow_run,\n                state_updates={\n                    \"message\": state_msg or \"Flow run was cancelled successfully.\"\n                },\n            )\n            run_logger.info(f\"Cancelled flow run '{flow_run.name}'!\")\n\n    async def _get_scheduled_flow_runs(\n        self,\n    ) -> List[\"FlowRun\"]:\n        \"\"\"\n        Retrieve scheduled flow runs for this runner.\n        \"\"\"\n        scheduled_before = pendulum.now(\"utc\").add(seconds=int(self._prefetch_seconds))\n        self._logger.debug(\n            f\"Querying for flow runs scheduled before {scheduled_before}\"\n        )\n\n        scheduled_flow_runs = (\n            await self._client.get_scheduled_flow_runs_for_deployments(\n                deployment_ids=list(self._deployment_ids),\n                scheduled_before=scheduled_before,\n            )\n        )\n        self._logger.debug(f\"Discovered {len(scheduled_flow_runs)} scheduled_flow_runs\")\n        return scheduled_flow_runs\n\n    def _acquire_limit_slot(self, flow_run_id: str) -> bool:\n        \"\"\"\n        Enforces flow run limit set on runner.\n\n        Returns:\n            - bool: True if a slot was acquired, False otherwise.\n        \"\"\"\n        try:\n            if self._limiter:\n                self._limiter.acquire_on_behalf_of_nowait(flow_run_id)\n                self._logger.debug(\"Limit slot acquired for flow run '%s'\", flow_run_id)\n            return True\n        except RuntimeError as exc:\n            if (\n                \"this borrower is already holding one of this CapacityLimiter's tokens\"\n                in str(exc)\n            ):\n                self._logger.warning(\n                    f\"Duplicate submission of flow run '{flow_run_id}' detected. Runner\"\n                    \" will not re-submit flow run.\"\n                )\n                return False\n            else:\n                raise\n        except anyio.WouldBlock:\n            self._logger.info(\n                f\"Flow run limit reached; {self._limiter.borrowed_tokens} flow runs\"\n                \" in progress. You can control this limit by adjusting the\"\n                \" PREFECT_RUNNER_PROCESS_LIMIT setting.\"\n            )\n            return False\n\n    def _release_limit_slot(self, flow_run_id: str) -> None:\n        \"\"\"\n        Frees up a slot taken by the given flow run id.\n        \"\"\"\n        if self._limiter:\n            self._limiter.release_on_behalf_of(flow_run_id)\n            self._logger.debug(\"Limit slot released for flow run '%s'\", flow_run_id)\n\n    async def _submit_scheduled_flow_runs(\n        self, flow_run_response: List[\"FlowRun\"]\n    ) -> List[\"FlowRun\"]:\n        \"\"\"\n        Takes a list of FlowRuns and submits the referenced flow runs\n        for execution by the runner.\n        \"\"\"\n        submittable_flow_runs = flow_run_response\n        submittable_flow_runs.sort(key=lambda run: run.next_scheduled_start_time)\n        for flow_run in submittable_flow_runs:\n            if flow_run.id in self._submitting_flow_run_ids:\n                continue\n\n            if self._acquire_limit_slot(flow_run.id):\n                run_logger = self._get_flow_run_logger(flow_run)\n                run_logger.info(\n                    f\"Runner '{self.name}' submitting flow run '{flow_run.id}'\"\n                )\n                self._submitting_flow_run_ids.add(flow_run.id)\n                self._runs_task_group.start_soon(\n                    self._submit_run,\n                    flow_run,\n                )\n            else:\n                break\n\n        return list(\n            filter(\n                lambda run: run.id in self._submitting_flow_run_ids,\n                submittable_flow_runs,\n            )\n        )\n\n    async def _submit_run(self, flow_run: \"FlowRun\") -> None:\n        \"\"\"\n        Submits a given flow run for execution by the runner.\n        \"\"\"\n        run_logger = self._get_flow_run_logger(flow_run)\n\n        ready_to_submit = await self._propose_pending_state(flow_run)\n\n        if ready_to_submit:\n            readiness_result = await self._runs_task_group.start(\n                self._submit_run_and_capture_errors, flow_run\n            )\n\n            if readiness_result and not isinstance(readiness_result, Exception):\n                self._flow_run_process_map[flow_run.id] = dict(\n                    pid=readiness_result, flow_run=flow_run\n                )\n\n            run_logger.info(f\"Completed submission of flow run '{flow_run.id}'\")\n        else:\n            # If the run is not ready to submit, release the concurrency slot\n            self._release_limit_slot(flow_run.id)\n\n        self._submitting_flow_run_ids.remove(flow_run.id)\n\n    async def _submit_run_and_capture_errors(\n        self, flow_run: \"FlowRun\", task_status: anyio.abc.TaskStatus = None\n    ) -> Union[Optional[int], Exception]:\n        run_logger = self._get_flow_run_logger(flow_run)\n\n        try:\n            status_code = await self._run_process(\n                flow_run=flow_run,\n                task_status=task_status,\n            )\n        except Exception as exc:\n            if not task_status._future.done():\n                # This flow run was being submitted and did not start successfully\n                run_logger.exception(\n                    f\"Failed to start process for flow run '{flow_run.id}'.\"\n                )\n                # Mark the task as started to prevent agent crash\n                task_status.started(exc)\n                await self._propose_crashed_state(\n                    flow_run, \"Flow run process could not be started\"\n                )\n            else:\n                run_logger.exception(\n                    f\"An error occurred while monitoring flow run '{flow_run.id}'. \"\n                    \"The flow run will not be marked as failed, but an issue may have \"\n                    \"occurred.\"\n                )\n            return exc\n        finally:\n            self._release_limit_slot(flow_run.id)\n            self._flow_run_process_map.pop(flow_run.id, None)\n\n        if status_code != 0:\n            await self._propose_crashed_state(\n                flow_run,\n                f\"Flow run process exited with non-zero status code {status_code}.\",\n            )\n\n        api_flow_run = await self._client.read_flow_run(flow_run_id=flow_run.id)\n        terminal_state = api_flow_run.state\n        if terminal_state.is_crashed():\n            await self._run_on_crashed_hooks(flow_run=flow_run, state=terminal_state)\n\n        return status_code\n\n    async def _propose_pending_state(self, flow_run: \"FlowRun\") -> bool:\n        run_logger = self._get_flow_run_logger(flow_run)\n        state = flow_run.state\n        try:\n            state = await propose_state(\n                self._client, Pending(), flow_run_id=flow_run.id\n            )\n        except Abort as exc:\n            run_logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}'. \"\n                    f\"Server sent an abort signal: {exc}\"\n                ),\n            )\n            return False\n        except Exception:\n            run_logger.exception(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n            )\n            return False\n\n        if not state.is_pending():\n            run_logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}': \"\n                    f\"Server returned a non-pending state {state.type.value!r}\"\n                ),\n            )\n            return False\n\n        return True\n\n    async def _propose_failed_state(self, flow_run: \"FlowRun\", exc: Exception) -> None:\n        run_logger = self._get_flow_run_logger(flow_run)\n        try:\n            await propose_state(\n                self._client,\n                await exception_to_failed_state(message=\"Submission failed.\", exc=exc),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # We've already failed, no need to note the abort but we don't want it to\n            # raise in the agent process\n            pass\n        except Exception:\n            run_logger.error(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n                exc_info=True,\n            )\n\n    async def _propose_crashed_state(self, flow_run: \"FlowRun\", message: str) -> None:\n        run_logger = self._get_flow_run_logger(flow_run)\n        try:\n            state = await propose_state(\n                self._client,\n                Crashed(message=message),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # Flow run already marked as failed\n            pass\n        except Exception:\n            run_logger.exception(f\"Failed to update state of flow run '{flow_run.id}'\")\n        else:\n            if state.is_crashed():\n                run_logger.info(\n                    f\"Reported flow run '{flow_run.id}' as crashed: {message}\"\n                )\n\n    async def _mark_flow_run_as_cancelled(\n        self, flow_run: \"FlowRun\", state_updates: Optional[dict] = None\n    ) -> None:\n        state_updates = state_updates or {}\n        state_updates.setdefault(\"name\", \"Cancelled\")\n        state_updates.setdefault(\"type\", StateType.CANCELLED)\n        state = flow_run.state.copy(update=state_updates)\n\n        await self._client.set_flow_run_state(flow_run.id, state, force=True)\n\n        # Do not remove the flow run from the cancelling set immediately because\n        # the API caches responses for the `read_flow_runs` and we do not want to\n        # duplicate cancellations.\n        await self._schedule_task(\n            60 * 10, self._cancelling_flow_run_ids.remove, flow_run.id\n        )\n\n    async def _schedule_task(self, __in_seconds: int, fn, *args, **kwargs):\n        \"\"\"\n        Schedule a background task to start after some time.\n\n        These tasks will be run immediately when the runner exits instead of waiting.\n\n        The function may be async or sync. Async functions will be awaited.\n        \"\"\"\n\n        async def wrapper(task_status):\n            # If we are shutting down, do not sleep; otherwise sleep until the scheduled\n            # time or shutdown\n            if self.started:\n                with anyio.CancelScope() as scope:\n                    self._scheduled_task_scopes.add(scope)\n                    task_status.started()\n                    await anyio.sleep(__in_seconds)\n\n                self._scheduled_task_scopes.remove(scope)\n            else:\n                task_status.started()\n\n            result = fn(*args, **kwargs)\n            if inspect.iscoroutine(result):\n                await result\n\n        await self._runs_task_group.start(wrapper)\n\n    async def _run_on_cancellation_hooks(\n        self,\n        flow_run: \"FlowRun\",\n        state: State,\n    ) -> None:\n        \"\"\"\n        Run the hooks for a flow.\n        \"\"\"\n        if state.is_cancelling():\n            flow = await load_flow_from_flow_run(\n                flow_run, client=self._client, storage_base_path=str(self._tmp_dir)\n            )\n            hooks = flow.on_cancellation or []\n\n            await _run_hooks(hooks, flow_run, flow, state)\n\n    async def _run_on_crashed_hooks(\n        self,\n        flow_run: \"FlowRun\",\n        state: State,\n    ) -> None:\n        \"\"\"\n        Run the hooks for a flow.\n        \"\"\"\n        if state.is_crashed():\n            flow = await load_flow_from_flow_run(\n                flow_run, client=self._client, storage_base_path=str(self._tmp_dir)\n            )\n            hooks = flow.on_crashed or []\n\n            await _run_hooks(hooks, flow_run, flow, state)\n\n    async def __aenter__(self):\n        self._logger.debug(\"Starting runner...\")\n        self._client = get_client()\n        self._tmp_dir.mkdir(parents=True)\n        await self._client.__aenter__()\n        await self._runs_task_group.__aenter__()\n\n        self.started = True\n        return self\n\n    async def __aexit__(self, *exc_info):\n        self._logger.debug(\"Stopping runner...\")\n        if self.pause_on_shutdown:\n            await self._pause_schedules()\n        self.started = False\n        for scope in self._scheduled_task_scopes:\n            scope.cancel()\n        if self._runs_task_group:\n            await self._runs_task_group.__aexit__(*exc_info)\n        if self._client:\n            await self._client.__aexit__(*exc_info)\n        shutil.rmtree(str(self._tmp_dir))\n\n    def __repr__(self):\n        return f\"Runner(name={self.name!r})\"\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.add_deployment","title":"add_deployment async","text":"

    Registers the deployment with the Prefect API and will monitor for work once the runner is started.

    Parameters:

    Name Type Description Default deployment RunnerDeployment

    A deployment for the runner to register.

    required Source code in prefect/runner/runner.py
    @sync_compatible\nasync def add_deployment(\n    self,\n    deployment: RunnerDeployment,\n) -> UUID:\n    \"\"\"\n    Registers the deployment with the Prefect API and will monitor for work once\n    the runner is started.\n\n    Args:\n        deployment: A deployment for the runner to register.\n    \"\"\"\n    deployment_id = await deployment.apply()\n    storage = deployment.storage\n    if storage is not None:\n        storage = await self._add_storage(storage)\n        self._deployment_storage_map[deployment_id] = storage\n    self._deployment_ids.add(deployment_id)\n\n    return deployment_id\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.add_flow","title":"add_flow async","text":"

    Provides a flow to the runner to be run based on the provided configuration.

    Will create a deployment for the provided flow and register the deployment with the runner.

    Parameters:

    Name Type Description Default flow Flow

    A flow for the runner to run.

    required name str

    The name to give the created deployment. Will default to the name of the runner.

    None interval Optional[Union[int, float, timedelta]]

    An interval on which to execute the current flow. Accepts either a number or a timedelta object. If a number is given, it will be interpreted as seconds.

    None cron Optional[str]

    A cron schedule of when to execute runs of this flow.

    None rrule Optional[str]

    An rrule schedule of when to execute runs of this flow.

    None schedule Optional[SCHEDULE_TYPES]

    A schedule object of when to execute runs of this flow. Used for advanced scheduling options like timezone.

    None is_schedule_active Optional[bool]

    Whether or not to set the schedule for this deployment as active. If not provided when creating a deployment, the schedule will be set as active. If not provided when updating a deployment, the schedule's activation will not be changed.

    None triggers Optional[List[DeploymentTrigger]]

    A list of triggers that should kick of a run of this flow.

    None parameters Optional[dict]

    A dictionary of default parameter values to pass to runs of this flow.

    None description Optional[str]

    A description for the created deployment. Defaults to the flow's description if not provided.

    None tags Optional[List[str]]

    A list of tags to associate with the created deployment for organizational purposes.

    None version Optional[str]

    A version for the created deployment. Defaults to the flow's version.

    None Source code in prefect/runner/runner.py
    @sync_compatible\nasync def add_flow(\n    self,\n    flow: Flow,\n    name: str = None,\n    interval: Optional[Union[int, float, datetime.timedelta]] = None,\n    cron: Optional[str] = None,\n    rrule: Optional[str] = None,\n    schedule: Optional[SCHEDULE_TYPES] = None,\n    is_schedule_active: Optional[bool] = None,\n    parameters: Optional[dict] = None,\n    triggers: Optional[List[DeploymentTrigger]] = None,\n    description: Optional[str] = None,\n    tags: Optional[List[str]] = None,\n    version: Optional[str] = None,\n    enforce_parameter_schema: bool = False,\n) -> UUID:\n    \"\"\"\n    Provides a flow to the runner to be run based on the provided configuration.\n\n    Will create a deployment for the provided flow and register the deployment\n    with the runner.\n\n    Args:\n        flow: A flow for the runner to run.\n        name: The name to give the created deployment. Will default to the name\n            of the runner.\n        interval: An interval on which to execute the current flow. Accepts either a number\n            or a timedelta object. If a number is given, it will be interpreted as seconds.\n        cron: A cron schedule of when to execute runs of this flow.\n        rrule: An rrule schedule of when to execute runs of this flow.\n        schedule: A schedule object of when to execute runs of this flow. Used for\n            advanced scheduling options like timezone.\n        is_schedule_active: Whether or not to set the schedule for this deployment as active. If\n            not provided when creating a deployment, the schedule will be set as active. If not\n            provided when updating a deployment, the schedule's activation will not be changed.\n        triggers: A list of triggers that should kick of a run of this flow.\n        parameters: A dictionary of default parameter values to pass to runs of this flow.\n        description: A description for the created deployment. Defaults to the flow's\n            description if not provided.\n        tags: A list of tags to associate with the created deployment for organizational\n            purposes.\n        version: A version for the created deployment. Defaults to the flow's version.\n    \"\"\"\n    api = PREFECT_API_URL.value()\n    if any([interval, cron, rrule]) and not api:\n        self._logger.warning(\n            \"Cannot schedule flows on an ephemeral server; run `prefect server\"\n            \" start` to start the scheduler.\"\n        )\n    name = self.name if name is None else name\n\n    deployment = await flow.to_deployment(\n        name=name,\n        interval=interval,\n        cron=cron,\n        rrule=rrule,\n        schedule=schedule,\n        is_schedule_active=is_schedule_active,\n        triggers=triggers,\n        parameters=parameters,\n        description=description,\n        tags=tags,\n        version=version,\n        enforce_parameter_schema=enforce_parameter_schema,\n    )\n    return await self.add_deployment(deployment)\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.execute_flow_run","title":"execute_flow_run async","text":"

    Executes a single flow run with the given ID.

    Execution will wait to monitor for cancellation requests. Exits once the flow run process has exited.

    Source code in prefect/runner/runner.py
    async def execute_flow_run(self, flow_run_id: UUID):\n    \"\"\"\n    Executes a single flow run with the given ID.\n\n    Execution will wait to monitor for cancellation requests. Exits once\n    the flow run process has exited.\n    \"\"\"\n    self.pause_on_shutdown = False\n    async with self:\n        if not self._acquire_limit_slot(flow_run_id):\n            return\n\n        async with anyio.create_task_group() as tg:\n            with anyio.CancelScope():\n                self._submitting_flow_run_ids.add(flow_run_id)\n                flow_run = await self._client.read_flow_run(flow_run_id)\n\n                pid = await self._runs_task_group.start(\n                    self._submit_run_and_capture_errors, flow_run\n                )\n\n                self._flow_run_process_map[flow_run.id] = dict(\n                    pid=pid, flow_run=flow_run\n                )\n\n                # We want this loop to stop when the flow run process exits\n                # so we'll check if the flow run process is still alive on\n                # each iteration and cancel the task group if it is not.\n                workload = partial(\n                    self._check_for_cancelled_flow_runs,\n                    should_stop=lambda: not self._flow_run_process_map,\n                    on_stop=tg.cancel_scope.cancel,\n                )\n\n                tg.start_soon(\n                    partial(\n                        critical_service_loop,\n                        workload=workload,\n                        interval=self.query_seconds,\n                        jitter_range=0.3,\n                    )\n                )\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.handle_sigterm","title":"handle_sigterm","text":"

    Gracefully shuts down the runner when a SIGTERM is received.

    Source code in prefect/runner/runner.py
    def handle_sigterm(self, signum, frame):\n    \"\"\"\n    Gracefully shuts down the runner when a SIGTERM is received.\n    \"\"\"\n    self._logger.info(\"SIGTERM received, initiating graceful shutdown...\")\n    from_sync.call_in_loop_thread(create_call(self.stop))\n\n    sys.exit(0)\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.start","title":"start async","text":"

    Starts a runner.

    The runner will begin monitoring for and executing any scheduled work for all added flows.

    Parameters:

    Name Type Description Default run_once bool

    If True, the runner will through one query loop and then exit.

    False webserver Optional[bool]

    a boolean for whether to start a webserver for this runner. If provided, overrides the default on the runner

    None

    Examples:

    Initialize a Runner, add two flows, and serve them by starting the Runner:

    from prefect import flow, Runner\n\n@flow\ndef hello_flow(name):\n    print(f\"hello {name}\")\n\n@flow\ndef goodbye_flow(name):\n    print(f\"goodbye {name}\")\n\nif __name__ == \"__main__\"\n    runner = Runner(name=\"my-runner\")\n\n    # Will be runnable via the API\n    runner.add_flow(hello_flow)\n\n    # Run on a cron schedule\n    runner.add_flow(goodbye_flow, schedule={\"cron\": \"0 * * * *\"})\n\n    runner.start()\n
    Source code in prefect/runner/runner.py
    @sync_compatible\nasync def start(\n    self, run_once: bool = False, webserver: Optional[bool] = None\n) -> None:\n    \"\"\"\n    Starts a runner.\n\n    The runner will begin monitoring for and executing any scheduled work for all added flows.\n\n    Args:\n        run_once: If True, the runner will through one query loop and then exit.\n        webserver: a boolean for whether to start a webserver for this runner. If provided,\n            overrides the default on the runner\n\n    Examples:\n        Initialize a Runner, add two flows, and serve them by starting the Runner:\n\n        ```python\n        from prefect import flow, Runner\n\n        @flow\n        def hello_flow(name):\n            print(f\"hello {name}\")\n\n        @flow\n        def goodbye_flow(name):\n            print(f\"goodbye {name}\")\n\n        if __name__ == \"__main__\"\n            runner = Runner(name=\"my-runner\")\n\n            # Will be runnable via the API\n            runner.add_flow(hello_flow)\n\n            # Run on a cron schedule\n            runner.add_flow(goodbye_flow, schedule={\"cron\": \"0 * * * *\"})\n\n            runner.start()\n        ```\n    \"\"\"\n    _register_signal(signal.SIGTERM, self.handle_sigterm)\n\n    webserver = webserver if webserver is not None else self.webserver\n\n    if webserver:\n        # we'll start the ASGI server in a separate thread so that\n        # uvicorn does not block the main thread\n        server_thread = threading.Thread(\n            name=\"runner-server-thread\",\n            target=partial(\n                start_webserver,\n                runner=self,\n            ),\n            daemon=True,\n        )\n        server_thread.start()\n\n    async with self as runner:\n        async with self._loops_task_group as tg:\n            for storage in self._storage_objs:\n                if storage.pull_interval:\n                    tg.start_soon(\n                        partial(\n                            critical_service_loop,\n                            workload=storage.pull_code,\n                            interval=storage.pull_interval,\n                            run_once=run_once,\n                            jitter_range=0.3,\n                        )\n                    )\n                else:\n                    tg.start_soon(storage.pull_code)\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=runner._get_and_submit_flow_runs,\n                    interval=self.query_seconds,\n                    run_once=run_once,\n                    jitter_range=0.3,\n                )\n            )\n            tg.start_soon(\n                partial(\n                    critical_service_loop,\n                    workload=runner._check_for_cancelled_flow_runs,\n                    interval=self.query_seconds * 2,\n                    run_once=run_once,\n                    jitter_range=0.3,\n                )\n            )\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.Runner.stop","title":"stop async","text":"

    Stops the runner's polling cycle.

    Source code in prefect/runner/runner.py
    @sync_compatible\nasync def stop(self):\n    \"\"\"Stops the runner's polling cycle.\"\"\"\n    if not self.started:\n        raise RuntimeError(\n            \"Runner has not yet started. Please start the runner by calling\"\n            \" .start()\"\n        )\n\n    self.started = False\n    self.stopping = True\n    await self.cancel_all()\n    try:\n        self._loops_task_group.cancel_scope.cancel()\n    except Exception:\n        self._logger.exception(\n            \"Exception encountered while shutting down\", exc_info=True\n        )\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/runner/#prefect.runner.runner.serve","title":"serve async","text":"

    Serve the provided list of deployments.

    Parameters:

    Name Type Description Default *args RunnerDeployment

    A list of deployments to serve.

    () pause_on_shutdown bool

    A boolean for whether or not to automatically pause deployment schedules on shutdown.

    True **kwargs

    Additional keyword arguments to pass to the runner.

    {}

    Examples:

    Prepare two deployments and serve them:

    import datetime\n\nfrom prefect import flow, serve\n\n@flow\ndef my_flow(name):\n    print(f\"hello {name}\")\n\n@flow\ndef my_other_flow(name):\n    print(f\"goodbye {name}\")\n\nif __name__ == \"__main__\":\n    # Run once a day\n    hello_deploy = my_flow.to_deployment(\n        \"hello\", tags=[\"dev\"], interval=datetime.timedelta(days=1)\n    )\n\n    # Run every Sunday at 4:00 AM\n    bye_deploy = my_other_flow.to_deployment(\n        \"goodbye\", tags=[\"dev\"], cron=\"0 4 * * sun\"\n    )\n\n    serve(hello_deploy, bye_deploy)\n
    Source code in prefect/runner/runner.py
    @sync_compatible\nasync def serve(\n    *args: RunnerDeployment,\n    pause_on_shutdown: bool = True,\n    print_starting_message: bool = True,\n    **kwargs,\n):\n    \"\"\"\n    Serve the provided list of deployments.\n\n    Args:\n        *args: A list of deployments to serve.\n        pause_on_shutdown: A boolean for whether or not to automatically pause\n            deployment schedules on shutdown.\n        **kwargs: Additional keyword arguments to pass to the runner.\n\n    Examples:\n        Prepare two deployments and serve them:\n\n        ```python\n        import datetime\n\n        from prefect import flow, serve\n\n        @flow\n        def my_flow(name):\n            print(f\"hello {name}\")\n\n        @flow\n        def my_other_flow(name):\n            print(f\"goodbye {name}\")\n\n        if __name__ == \"__main__\":\n            # Run once a day\n            hello_deploy = my_flow.to_deployment(\n                \"hello\", tags=[\"dev\"], interval=datetime.timedelta(days=1)\n            )\n\n            # Run every Sunday at 4:00 AM\n            bye_deploy = my_other_flow.to_deployment(\n                \"goodbye\", tags=[\"dev\"], cron=\"0 4 * * sun\"\n            )\n\n            serve(hello_deploy, bye_deploy)\n        ```\n    \"\"\"\n    runner = Runner(pause_on_shutdown=pause_on_shutdown, **kwargs)\n    for deployment in args:\n        await runner.add_deployment(deployment)\n\n    if print_starting_message:\n        help_message_top = (\n            \"[green]Your deployments are being served and polling for\"\n            \" scheduled runs!\\n[/]\"\n        )\n\n        table = Table(title=\"Deployments\", show_header=False)\n\n        table.add_column(style=\"blue\", no_wrap=True)\n\n        for deployment in args:\n            table.add_row(f\"{deployment.flow_name}/{deployment.name}\")\n\n        help_message_bottom = (\n            \"\\nTo trigger any of these deployments, use the\"\n            \" following command:\\n[blue]\\n\\t$ prefect deployment run\"\n            \" [DEPLOYMENT_NAME]\\n[/]\"\n        )\n        if PREFECT_UI_URL:\n            help_message_bottom += (\n                \"\\nYou can also trigger your deployments via the Prefect UI:\"\n                f\" [blue]{PREFECT_UI_URL.value()}/deployments[/]\\n\"\n            )\n\n        console = Console()\n        console.print(Panel(Group(help_message_top, table, help_message_bottom)))\n\n    await runner.start()\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/server/","title":"server","text":"","tags":["Python API","server"]},{"location":"api-ref/prefect/runner/server/#prefect.runner.server","title":"prefect.runner.server","text":"","tags":["Python API","server"]},{"location":"api-ref/prefect/runner/server/#prefect.runner.server.build_server","title":"build_server async","text":"

    Build a FastAPI server for a runner.

    Parameters:

    Name Type Description Default runner Runner

    the runner this server interacts with and monitors

    required log_level str

    the log level to use for the server

    required Source code in prefect/runner/server.py
    @sync_compatible\nasync def build_server(runner: \"Runner\") -> FastAPI:\n    \"\"\"\n    Build a FastAPI server for a runner.\n\n    Args:\n        runner (Runner): the runner this server interacts with and monitors\n        log_level (str): the log level to use for the server\n    \"\"\"\n    webserver = FastAPI()\n    router = APIRouter()\n\n    router.add_api_route(\n        \"/health\", perform_health_check(runner=runner), methods=[\"GET\"]\n    )\n    router.add_api_route(\"/run_count\", run_count(runner=runner), methods=[\"GET\"])\n    router.add_api_route(\"/shutdown\", shutdown(runner=runner), methods=[\"POST\"])\n    webserver.include_router(router)\n\n    if PREFECT_EXPERIMENTAL_ENABLE_EXTRA_RUNNER_ENDPOINTS.value():\n        deployments_router, deployment_schemas = await get_deployment_router(runner)\n        webserver.include_router(deployments_router)\n\n        def customize_openapi():\n            if webserver.openapi_schema:\n                return webserver.openapi_schema\n\n            openapi_schema = inject_schemas_into_openapi(webserver, deployment_schemas)\n            webserver.openapi_schema = openapi_schema\n            return webserver.openapi_schema\n\n        webserver.openapi = customize_openapi\n\n    return webserver\n
    ","tags":["Python API","server"]},{"location":"api-ref/prefect/runner/server/#prefect.runner.server.start_webserver","title":"start_webserver","text":"

    Run a FastAPI server for a runner.

    Parameters:

    Name Type Description Default runner Runner

    the runner this server interacts with and monitors

    required log_level str

    the log level to use for the server

    None Source code in prefect/runner/server.py
    def start_webserver(runner: \"Runner\", log_level: Optional[str] = None) -> None:\n    \"\"\"\n    Run a FastAPI server for a runner.\n\n    Args:\n        runner (Runner): the runner this server interacts with and monitors\n        log_level (str): the log level to use for the server\n    \"\"\"\n    host = PREFECT_RUNNER_SERVER_HOST.value()\n    port = PREFECT_RUNNER_SERVER_PORT.value()\n    log_level = log_level or PREFECT_RUNNER_SERVER_LOG_LEVEL.value()\n    webserver = build_server(runner)\n    uvicorn.run(webserver, host=host, port=port, log_level=log_level)\n
    ","tags":["Python API","server"]},{"location":"api-ref/prefect/runner/storage/","title":"storage","text":"","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage","title":"prefect.runner.storage","text":"","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.BlockStorageAdapter","title":"BlockStorageAdapter","text":"

    A storage adapter for a storage block object to allow it to be used as a runner storage object.

    Source code in prefect/runner/storage.py
    class BlockStorageAdapter:\n    \"\"\"\n    A storage adapter for a storage block object to allow it to be used as a\n    runner storage object.\n    \"\"\"\n\n    def __init__(\n        self,\n        block: Union[ReadableDeploymentStorage, WritableDeploymentStorage],\n        pull_interval: Optional[int] = 60,\n    ):\n        self._block = block\n        self._pull_interval = pull_interval\n        self._storage_base_path = Path.cwd()\n        if not isinstance(block, Block):\n            raise TypeError(\n                f\"Expected a block object. Received a {type(block).__name__!r} object.\"\n            )\n        if not hasattr(block, \"get_directory\"):\n            raise ValueError(\"Provided block must have a `get_directory` method.\")\n\n        self._name = (\n            f\"{block.get_block_type_slug()}-{block._block_document_name}\"\n            if block._block_document_name\n            else str(uuid4())\n        )\n\n    def set_base_path(self, path: Path):\n        self._storage_base_path = path\n\n    @property\n    def pull_interval(self) -> Optional[int]:\n        return self._pull_interval\n\n    @property\n    def destination(self) -> Path:\n        return self._storage_base_path / self._name\n\n    async def pull_code(self):\n        if not self.destination.exists():\n            self.destination.mkdir(parents=True, exist_ok=True)\n        await self._block.get_directory(local_path=str(self.destination))\n\n    def to_pull_step(self) -> dict:\n        # Give blocks the change to implement their own pull step\n        if hasattr(self._block, \"get_pull_step\"):\n            return self._block.get_pull_step()\n        else:\n            if not self._block._block_document_name:\n                raise BlockNotSavedError(\n                    \"Block must be saved with `.save()` before it can be converted to a\"\n                    \" pull step.\"\n                )\n            return {\n                \"prefect.deployments.steps.pull_with_block\": {\n                    \"block_type_slug\": self._block.get_block_type_slug(),\n                    \"block_document_name\": self._block._block_document_name,\n                }\n            }\n\n    def __eq__(self, __value) -> bool:\n        if isinstance(__value, BlockStorageAdapter):\n            return self._block == __value._block\n        return False\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.GitRepository","title":"GitRepository","text":"

    Pulls the contents of a git repository to the local filesystem.

    Parameters:

    Name Type Description Default url str

    The URL of the git repository to pull from

    required credentials Union[GitCredentials, Block, None]

    A dictionary of credentials to use when pulling from the repository. If a username is provided, an access token must also be provided.

    None name Optional[str]

    The name of the repository. If not provided, the name will be inferred from the repository URL.

    None branch Optional[str]

    The branch to pull from. Defaults to \"main\".

    None pull_interval Optional[int]

    The interval in seconds at which to pull contents from remote storage to local storage. If None, remote storage will perform a one-time sync.

    60

    Examples:

    Pull the contents of a private git repository to the local filesystem:

    from prefect.runner.storage import GitRepository\n\nstorage = GitRepository(\n    url=\"https://github.com/org/repo.git\",\n    credentials={\"username\": \"oauth2\", \"access_token\": \"my-access-token\"},\n)\n\nawait storage.pull_code()\n
    Source code in prefect/runner/storage.py
    class GitRepository:\n    \"\"\"\n    Pulls the contents of a git repository to the local filesystem.\n\n    Parameters:\n        url: The URL of the git repository to pull from\n        credentials: A dictionary of credentials to use when pulling from the\n            repository. If a username is provided, an access token must also be\n            provided.\n        name: The name of the repository. If not provided, the name will be\n            inferred from the repository URL.\n        branch: The branch to pull from. Defaults to \"main\".\n        pull_interval: The interval in seconds at which to pull contents from\n            remote storage to local storage. If None, remote storage will perform\n            a one-time sync.\n\n    Examples:\n        Pull the contents of a private git repository to the local filesystem:\n\n        ```python\n        from prefect.runner.storage import GitRepository\n\n        storage = GitRepository(\n            url=\"https://github.com/org/repo.git\",\n            credentials={\"username\": \"oauth2\", \"access_token\": \"my-access-token\"},\n        )\n\n        await storage.pull_code()\n        ```\n    \"\"\"\n\n    def __init__(\n        self,\n        url: str,\n        credentials: Union[GitCredentials, Block, None] = None,\n        name: Optional[str] = None,\n        branch: Optional[str] = None,\n        include_submodules: bool = False,\n        pull_interval: Optional[int] = 60,\n    ):\n        if credentials is None:\n            credentials = {}\n\n        if (\n            isinstance(credentials, dict)\n            and credentials.get(\"username\")\n            and not (credentials.get(\"access_token\") or credentials.get(\"password\"))\n        ):\n            raise ValueError(\n                \"If a username is provided, an access token or password must also be\"\n                \" provided.\"\n            )\n        self._url = url\n        self._branch = branch\n        self._credentials = credentials\n        self._include_submodules = include_submodules\n        repo_name = urlparse(url).path.split(\"/\")[-1].replace(\".git\", \"\")\n        default_name = f\"{repo_name}-{branch}\" if branch else repo_name\n        self._name = name or default_name\n        self._logger = get_logger(f\"runner.storage.git-repository.{self._name}\")\n        self._storage_base_path = Path.cwd()\n        self._pull_interval = pull_interval\n\n    @property\n    def destination(self) -> Path:\n        return self._storage_base_path / self._name\n\n    def set_base_path(self, path: Path):\n        self._storage_base_path = path\n\n    @property\n    def pull_interval(self) -> Optional[int]:\n        return self._pull_interval\n\n    @property\n    def _repository_url_with_credentials(self) -> str:\n        if not self._credentials:\n            return self._url\n\n        url_components = urlparse(self._url)\n\n        credentials = (\n            self._credentials.dict()\n            if isinstance(self._credentials, Block)\n            else deepcopy(self._credentials)\n        )\n\n        for k, v in credentials.items():\n            if isinstance(v, Secret):\n                credentials[k] = v.get()\n            elif isinstance(v, SecretStr):\n                credentials[k] = v.get_secret_value()\n\n        formatted_credentials = _format_token_from_credentials(\n            urlparse(self._url).netloc, credentials\n        )\n        if url_components.scheme == \"https\" and formatted_credentials is not None:\n            updated_components = url_components._replace(\n                netloc=f\"{formatted_credentials}@{url_components.netloc}\"\n            )\n            repository_url = urlunparse(updated_components)\n        else:\n            repository_url = self._url\n\n        return repository_url\n\n    async def pull_code(self):\n        \"\"\"\n        Pulls the contents of the configured repository to the local filesystem.\n        \"\"\"\n        self._logger.debug(\n            \"Pulling contents from repository '%s' to '%s'...\",\n            self._name,\n            self.destination,\n        )\n\n        git_dir = self.destination / \".git\"\n\n        if git_dir.exists():\n            # Check if the existing repository matches the configured repository\n            result = await run_process(\n                [\"git\", \"config\", \"--get\", \"remote.origin.url\"],\n                cwd=str(self.destination),\n            )\n            existing_repo_url = None\n            if result.stdout is not None:\n                existing_repo_url = _strip_auth_from_url(result.stdout.decode().strip())\n\n            if existing_repo_url != self._url:\n                raise ValueError(\n                    f\"The existing repository at {str(self.destination)} \"\n                    f\"does not match the configured repository {self._url}\"\n                )\n\n            self._logger.debug(\"Pulling latest changes from origin/%s\", self._branch)\n            # Update the existing repository\n            cmd = [\"git\", \"pull\", \"origin\"]\n            if self._branch:\n                cmd += [self._branch]\n            if self._include_submodules:\n                cmd += [\"--recurse-submodules\"]\n            cmd += [\"--depth\", \"1\"]\n            try:\n                await run_process(cmd, cwd=self.destination)\n                self._logger.debug(\"Successfully pulled latest changes\")\n            except subprocess.CalledProcessError as exc:\n                self._logger.error(\n                    f\"Failed to pull latest changes with exit code {exc}\"\n                )\n                shutil.rmtree(self.destination)\n                await self._clone_repo()\n\n        else:\n            await self._clone_repo()\n\n    async def _clone_repo(self):\n        \"\"\"\n        Clones the repository into the local destination.\n        \"\"\"\n        self._logger.debug(\"Cloning repository %s\", self._url)\n\n        repository_url = self._repository_url_with_credentials\n\n        cmd = [\n            \"git\",\n            \"clone\",\n            repository_url,\n        ]\n        if self._branch:\n            cmd += [\"--branch\", self._branch]\n        if self._include_submodules:\n            cmd += [\"--recurse-submodules\"]\n\n        # Limit git history and set path to clone to\n        cmd += [\"--depth\", \"1\", str(self.destination)]\n\n        try:\n            await run_process(cmd)\n        except subprocess.CalledProcessError as exc:\n            # Hide the command used to avoid leaking the access token\n            exc_chain = None if self._credentials else exc\n            raise RuntimeError(\n                f\"Failed to clone repository {self._url!r} with exit code\"\n                f\" {exc.returncode}.\"\n            ) from exc_chain\n\n    def __eq__(self, __value) -> bool:\n        if isinstance(__value, GitRepository):\n            return (\n                self._url == __value._url\n                and self._branch == __value._branch\n                and self._name == __value._name\n            )\n        return False\n\n    def __repr__(self) -> str:\n        return (\n            f\"GitRepository(name={self._name!r} repository={self._url!r},\"\n            f\" branch={self._branch!r})\"\n        )\n\n    def to_pull_step(self) -> Dict:\n        pull_step = {\n            \"prefect.deployments.steps.git_clone\": {\n                \"repository\": self._url,\n                \"branch\": self._branch,\n            }\n        }\n        if isinstance(self._credentials, Block):\n            pull_step[\"prefect.deployments.steps.git_clone\"][\n                \"credentials\"\n            ] = f\"{{{{ {self._credentials.get_block_placeholder()} }}}}\"\n        elif isinstance(self._credentials, dict):\n            if isinstance(self._credentials.get(\"access_token\"), Secret):\n                pull_step[\"prefect.deployments.steps.git_clone\"][\"credentials\"] = {\n                    **self._credentials,\n                    \"access_token\": (\n                        \"{{\"\n                        f\" {self._credentials['access_token'].get_block_placeholder()} }}}}\"\n                    ),\n                }\n            elif self._credentials.get(\"access_token\") is not None:\n                raise ValueError(\n                    \"Please save your access token as a Secret block before converting\"\n                    \" this storage object to a pull step.\"\n                )\n\n        return pull_step\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.GitRepository.pull_code","title":"pull_code async","text":"

    Pulls the contents of the configured repository to the local filesystem.

    Source code in prefect/runner/storage.py
    async def pull_code(self):\n    \"\"\"\n    Pulls the contents of the configured repository to the local filesystem.\n    \"\"\"\n    self._logger.debug(\n        \"Pulling contents from repository '%s' to '%s'...\",\n        self._name,\n        self.destination,\n    )\n\n    git_dir = self.destination / \".git\"\n\n    if git_dir.exists():\n        # Check if the existing repository matches the configured repository\n        result = await run_process(\n            [\"git\", \"config\", \"--get\", \"remote.origin.url\"],\n            cwd=str(self.destination),\n        )\n        existing_repo_url = None\n        if result.stdout is not None:\n            existing_repo_url = _strip_auth_from_url(result.stdout.decode().strip())\n\n        if existing_repo_url != self._url:\n            raise ValueError(\n                f\"The existing repository at {str(self.destination)} \"\n                f\"does not match the configured repository {self._url}\"\n            )\n\n        self._logger.debug(\"Pulling latest changes from origin/%s\", self._branch)\n        # Update the existing repository\n        cmd = [\"git\", \"pull\", \"origin\"]\n        if self._branch:\n            cmd += [self._branch]\n        if self._include_submodules:\n            cmd += [\"--recurse-submodules\"]\n        cmd += [\"--depth\", \"1\"]\n        try:\n            await run_process(cmd, cwd=self.destination)\n            self._logger.debug(\"Successfully pulled latest changes\")\n        except subprocess.CalledProcessError as exc:\n            self._logger.error(\n                f\"Failed to pull latest changes with exit code {exc}\"\n            )\n            shutil.rmtree(self.destination)\n            await self._clone_repo()\n\n    else:\n        await self._clone_repo()\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage","title":"RemoteStorage","text":"

    Pulls the contents of a remote storage location to the local filesystem.

    Parameters:

    Name Type Description Default url str

    The URL of the remote storage location to pull from. Supports fsspec URLs. Some protocols may require an additional fsspec dependency to be installed. Refer to the fsspec docs for more details.

    required pull_interval Optional[int]

    The interval in seconds at which to pull contents from remote storage to local storage. If None, remote storage will perform a one-time sync.

    60 **settings Any

    Any additional settings to pass the fsspec filesystem class.

    {}

    Examples:

    Pull the contents of a remote storage location to the local filesystem:

    from prefect.runner.storage import RemoteStorage\n\nstorage = RemoteStorage(url=\"s3://my-bucket/my-folder\")\n\nawait storage.pull_code()\n

    Pull the contents of a remote storage location to the local filesystem with additional settings:

    from prefect.runner.storage import RemoteStorage\nfrom prefect.blocks.system import Secret\n\nstorage = RemoteStorage(\n    url=\"s3://my-bucket/my-folder\",\n    # Use Secret blocks to keep credentials out of your code\n    key=Secret.load(\"my-aws-access-key\"),\n    secret=Secret.load(\"my-aws-secret-key\"),\n)\n\nawait storage.pull_code()\n
    Source code in prefect/runner/storage.py
    class RemoteStorage:\n    \"\"\"\n    Pulls the contents of a remote storage location to the local filesystem.\n\n    Parameters:\n        url: The URL of the remote storage location to pull from. Supports\n            `fsspec` URLs. Some protocols may require an additional `fsspec`\n            dependency to be installed. Refer to the\n            [`fsspec` docs](https://filesystem-spec.readthedocs.io/en/latest/api.html#other-known-implementations)\n            for more details.\n        pull_interval: The interval in seconds at which to pull contents from\n            remote storage to local storage. If None, remote storage will perform\n            a one-time sync.\n        **settings: Any additional settings to pass the `fsspec` filesystem class.\n\n    Examples:\n        Pull the contents of a remote storage location to the local filesystem:\n\n        ```python\n        from prefect.runner.storage import RemoteStorage\n\n        storage = RemoteStorage(url=\"s3://my-bucket/my-folder\")\n\n        await storage.pull_code()\n        ```\n\n        Pull the contents of a remote storage location to the local filesystem\n        with additional settings:\n\n        ```python\n        from prefect.runner.storage import RemoteStorage\n        from prefect.blocks.system import Secret\n\n        storage = RemoteStorage(\n            url=\"s3://my-bucket/my-folder\",\n            # Use Secret blocks to keep credentials out of your code\n            key=Secret.load(\"my-aws-access-key\"),\n            secret=Secret.load(\"my-aws-secret-key\"),\n        )\n\n        await storage.pull_code()\n        ```\n    \"\"\"\n\n    def __init__(\n        self,\n        url: str,\n        pull_interval: Optional[int] = 60,\n        **settings: Any,\n    ):\n        self._url = url\n        self._settings = settings\n        self._logger = get_logger(\"runner.storage.remote-storage\")\n        self._storage_base_path = Path.cwd()\n        self._pull_interval = pull_interval\n\n    @staticmethod\n    def _get_required_package_for_scheme(scheme: str) -> Optional[str]:\n        # attempt to discover the package name for the given scheme\n        # from fsspec's registry\n        known_implementation = fsspec.registry.get(scheme)\n        if known_implementation:\n            return known_implementation.__module__.split(\".\")[0]\n        # if we don't know the implementation, try to guess it for some\n        # common schemes\n        elif scheme == \"s3\":\n            return \"s3fs\"\n        elif scheme == \"gs\" or scheme == \"gcs\":\n            return \"gcsfs\"\n        elif scheme == \"abfs\" or scheme == \"az\":\n            return \"adlfs\"\n        else:\n            return None\n\n    @property\n    def _filesystem(self) -> fsspec.AbstractFileSystem:\n        scheme, _, _, _, _ = urlsplit(self._url)\n\n        def replace_blocks_with_values(obj: Any) -> Any:\n            if isinstance(obj, Block):\n                if hasattr(obj, \"get\"):\n                    return obj.get()\n                if hasattr(obj, \"value\"):\n                    return obj.value\n                else:\n                    return obj.dict()\n            return obj\n\n        settings_with_block_values = visit_collection(\n            self._settings, replace_blocks_with_values, return_data=True\n        )\n\n        return fsspec.filesystem(scheme, **settings_with_block_values)\n\n    def set_base_path(self, path: Path):\n        self._storage_base_path = path\n\n    @property\n    def pull_interval(self) -> Optional[int]:\n        \"\"\"\n        The interval at which contents from remote storage should be pulled to\n        local storage. If None, remote storage will perform a one-time sync.\n        \"\"\"\n        return self._pull_interval\n\n    @property\n    def destination(self) -> Path:\n        \"\"\"\n        The local file path to pull contents from remote storage to.\n        \"\"\"\n        return self._storage_base_path / self._remote_path\n\n    @property\n    def _remote_path(self) -> Path:\n        \"\"\"\n        The remote file path to pull contents from remote storage to.\n        \"\"\"\n        _, netloc, urlpath, _, _ = urlsplit(self._url)\n        return Path(netloc) / Path(urlpath.lstrip(\"/\"))\n\n    async def pull_code(self):\n        \"\"\"\n        Pulls contents from remote storage to the local filesystem.\n        \"\"\"\n        self._logger.debug(\n            \"Pulling contents from remote storage '%s' to '%s'...\",\n            self._url,\n            self.destination,\n        )\n\n        if not self.destination.exists():\n            self.destination.mkdir(parents=True, exist_ok=True)\n\n        remote_path = str(self._remote_path) + \"/\"\n\n        try:\n            await from_async.wait_for_call_in_new_thread(\n                create_call(\n                    self._filesystem.get,\n                    remote_path,\n                    str(self.destination),\n                    recursive=True,\n                )\n            )\n        except Exception as exc:\n            raise RuntimeError(\n                f\"Failed to pull contents from remote storage {self._url!r} to\"\n                f\" {self.destination!r}\"\n            ) from exc\n\n    def to_pull_step(self) -> dict:\n        \"\"\"\n        Returns a dictionary representation of the storage object that can be\n        used as a deployment pull step.\n        \"\"\"\n\n        def replace_block_with_placeholder(obj: Any) -> Any:\n            if isinstance(obj, Block):\n                return f\"{{{{ {obj.get_block_placeholder()} }}}}\"\n            return obj\n\n        settings_with_placeholders = visit_collection(\n            self._settings, replace_block_with_placeholder, return_data=True\n        )\n        required_package = self._get_required_package_for_scheme(\n            urlparse(self._url).scheme\n        )\n        step = {\n            \"prefect.deployments.steps.pull_from_remote_storage\": {\n                \"url\": self._url,\n                **settings_with_placeholders,\n            }\n        }\n        if required_package:\n            step[\"prefect.deployments.steps.pull_from_remote_storage\"][\n                \"requires\"\n            ] = required_package\n        return step\n\n    def __eq__(self, __value) -> bool:\n        \"\"\"\n        Equality check for runner storage objects.\n        \"\"\"\n        if isinstance(__value, RemoteStorage):\n            return self._url == __value._url and self._settings == __value._settings\n        return False\n\n    def __repr__(self) -> str:\n        return f\"RemoteStorage(url={self._url!r})\"\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage.destination","title":"destination: Path property","text":"

    The local file path to pull contents from remote storage to.

    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage.pull_interval","title":"pull_interval: Optional[int] property","text":"

    The interval at which contents from remote storage should be pulled to local storage. If None, remote storage will perform a one-time sync.

    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage.pull_code","title":"pull_code async","text":"

    Pulls contents from remote storage to the local filesystem.

    Source code in prefect/runner/storage.py
    async def pull_code(self):\n    \"\"\"\n    Pulls contents from remote storage to the local filesystem.\n    \"\"\"\n    self._logger.debug(\n        \"Pulling contents from remote storage '%s' to '%s'...\",\n        self._url,\n        self.destination,\n    )\n\n    if not self.destination.exists():\n        self.destination.mkdir(parents=True, exist_ok=True)\n\n    remote_path = str(self._remote_path) + \"/\"\n\n    try:\n        await from_async.wait_for_call_in_new_thread(\n            create_call(\n                self._filesystem.get,\n                remote_path,\n                str(self.destination),\n                recursive=True,\n            )\n        )\n    except Exception as exc:\n        raise RuntimeError(\n            f\"Failed to pull contents from remote storage {self._url!r} to\"\n            f\" {self.destination!r}\"\n        ) from exc\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RemoteStorage.to_pull_step","title":"to_pull_step","text":"

    Returns a dictionary representation of the storage object that can be used as a deployment pull step.

    Source code in prefect/runner/storage.py
    def to_pull_step(self) -> dict:\n    \"\"\"\n    Returns a dictionary representation of the storage object that can be\n    used as a deployment pull step.\n    \"\"\"\n\n    def replace_block_with_placeholder(obj: Any) -> Any:\n        if isinstance(obj, Block):\n            return f\"{{{{ {obj.get_block_placeholder()} }}}}\"\n        return obj\n\n    settings_with_placeholders = visit_collection(\n        self._settings, replace_block_with_placeholder, return_data=True\n    )\n    required_package = self._get_required_package_for_scheme(\n        urlparse(self._url).scheme\n    )\n    step = {\n        \"prefect.deployments.steps.pull_from_remote_storage\": {\n            \"url\": self._url,\n            **settings_with_placeholders,\n        }\n    }\n    if required_package:\n        step[\"prefect.deployments.steps.pull_from_remote_storage\"][\n            \"requires\"\n        ] = required_package\n    return step\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage","title":"RunnerStorage","text":"

    Bases: Protocol

    A storage interface for a runner to use to retrieve remotely stored flow code.

    Source code in prefect/runner/storage.py
    @runtime_checkable\nclass RunnerStorage(Protocol):\n    \"\"\"\n    A storage interface for a runner to use to retrieve\n    remotely stored flow code.\n    \"\"\"\n\n    def set_base_path(self, path: Path):\n        \"\"\"\n        Sets the base path to use when pulling contents from remote storage to\n        local storage.\n        \"\"\"\n        ...\n\n    @property\n    def pull_interval(self) -> Optional[int]:\n        \"\"\"\n        The interval at which contents from remote storage should be pulled to\n        local storage. If None, remote storage will perform a one-time sync.\n        \"\"\"\n        ...\n\n    @property\n    def destination(self) -> Path:\n        \"\"\"\n        The local file path to pull contents from remote storage to.\n        \"\"\"\n        ...\n\n    async def pull_code(self):\n        \"\"\"\n        Pulls contents from remote storage to the local filesystem.\n        \"\"\"\n        ...\n\n    def to_pull_step(self) -> dict:\n        \"\"\"\n        Returns a dictionary representation of the storage object that can be\n        used as a deployment pull step.\n        \"\"\"\n        ...\n\n    def __eq__(self, __value) -> bool:\n        \"\"\"\n        Equality check for runner storage objects.\n        \"\"\"\n        ...\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.destination","title":"destination: Path property","text":"

    The local file path to pull contents from remote storage to.

    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.pull_interval","title":"pull_interval: Optional[int] property","text":"

    The interval at which contents from remote storage should be pulled to local storage. If None, remote storage will perform a one-time sync.

    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.pull_code","title":"pull_code async","text":"

    Pulls contents from remote storage to the local filesystem.

    Source code in prefect/runner/storage.py
    async def pull_code(self):\n    \"\"\"\n    Pulls contents from remote storage to the local filesystem.\n    \"\"\"\n    ...\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.set_base_path","title":"set_base_path","text":"

    Sets the base path to use when pulling contents from remote storage to local storage.

    Source code in prefect/runner/storage.py
    def set_base_path(self, path: Path):\n    \"\"\"\n    Sets the base path to use when pulling contents from remote storage to\n    local storage.\n    \"\"\"\n    ...\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.RunnerStorage.to_pull_step","title":"to_pull_step","text":"

    Returns a dictionary representation of the storage object that can be used as a deployment pull step.

    Source code in prefect/runner/storage.py
    def to_pull_step(self) -> dict:\n    \"\"\"\n    Returns a dictionary representation of the storage object that can be\n    used as a deployment pull step.\n    \"\"\"\n    ...\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/storage/#prefect.runner.storage.create_storage_from_url","title":"create_storage_from_url","text":"

    Creates a storage object from a URL.

    Parameters:

    Name Type Description Default url str

    The URL to create a storage object from. Supports git and fsspec URLs.

    required pull_interval Optional[int]

    The interval at which to pull contents from remote storage to local storage

    60

    Returns:

    Name Type Description RunnerStorage RunnerStorage

    A runner storage compatible object

    Source code in prefect/runner/storage.py
    def create_storage_from_url(\n    url: str, pull_interval: Optional[int] = 60\n) -> RunnerStorage:\n    \"\"\"\n    Creates a storage object from a URL.\n\n    Args:\n        url: The URL to create a storage object from. Supports git and `fsspec`\n            URLs.\n        pull_interval: The interval at which to pull contents from remote storage to\n            local storage\n\n    Returns:\n        RunnerStorage: A runner storage compatible object\n    \"\"\"\n    parsed_url = urlparse(url)\n    if parsed_url.scheme == \"git\" or parsed_url.path.endswith(\".git\"):\n        return GitRepository(url=url, pull_interval=pull_interval)\n    else:\n        return RemoteStorage(url=url, pull_interval=pull_interval)\n
    ","tags":["Python API","runner"]},{"location":"api-ref/prefect/runner/utils/","title":"utils","text":"","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils","title":"prefect.runner.utils","text":"","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils.inject_schemas_into_openapi","title":"inject_schemas_into_openapi","text":"

    Augments the webserver's OpenAPI schema with additional schemas from deployments.

    Parameters:

    Name Type Description Default webserver FastAPI

    The FastAPI instance representing the webserver.

    required deployment_schemas Dict[str, Any]

    A dictionary of deployment schemas to integrate.

    required

    Returns:

    Type Description Dict[str, Any]

    The augmented OpenAPI schema dictionary.

    Source code in prefect/runner/utils.py
    def inject_schemas_into_openapi(\n    webserver: FastAPI, deployment_schemas: Dict[str, Any]\n) -> Dict[str, Any]:\n    \"\"\"\n    Augments the webserver's OpenAPI schema with additional schemas from deployments.\n\n    Args:\n        webserver: The FastAPI instance representing the webserver.\n        deployment_schemas: A dictionary of deployment schemas to integrate.\n\n    Returns:\n        The augmented OpenAPI schema dictionary.\n    \"\"\"\n    openapi_schema = get_openapi(\n        title=\"FastAPI Prefect Runner\", version=PREFECT_VERSION, routes=webserver.routes\n    )\n\n    augmented_schema = merge_definitions(deployment_schemas, openapi_schema)\n    return update_refs_to_components(augmented_schema)\n
    ","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils.merge_definitions","title":"merge_definitions","text":"

    Integrates definitions from deployment schemas into the OpenAPI components.

    Parameters:

    Name Type Description Default deployment_schemas Dict[str, Any]

    A dictionary of deployment-specific schemas.

    required openapi_schema Dict[str, Any]

    The base OpenAPI schema to update.

    required Source code in prefect/runner/utils.py
    def merge_definitions(\n    deployment_schemas: Dict[str, Any], openapi_schema: Dict[str, Any]\n) -> Dict[str, Any]:\n    \"\"\"\n    Integrates definitions from deployment schemas into the OpenAPI components.\n\n    Args:\n        deployment_schemas: A dictionary of deployment-specific schemas.\n        openapi_schema: The base OpenAPI schema to update.\n    \"\"\"\n    openapi_schema_copy = deepcopy(openapi_schema)\n    components = openapi_schema_copy.setdefault(\"components\", {}).setdefault(\n        \"schemas\", {}\n    )\n    for definitions in deployment_schemas.values():\n        if \"definitions\" in definitions:\n            for def_name, def_schema in definitions[\"definitions\"].items():\n                def_schema_copy = deepcopy(def_schema)\n                update_refs_in_schema(def_schema_copy, \"#/components/schemas/\")\n                components[def_name] = def_schema_copy\n    return openapi_schema_copy\n
    ","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils.update_refs_in_schema","title":"update_refs_in_schema","text":"

    Recursively replaces $ref with a new reference base in a schema item.

    Parameters:

    Name Type Description Default schema_item Any

    A schema or part of a schema to update references in.

    required new_ref str

    The new base string to replace in $ref values.

    required Source code in prefect/runner/utils.py
    def update_refs_in_schema(schema_item: Any, new_ref: str) -> None:\n    \"\"\"\n    Recursively replaces `$ref` with a new reference base in a schema item.\n\n    Args:\n        schema_item: A schema or part of a schema to update references in.\n        new_ref: The new base string to replace in `$ref` values.\n    \"\"\"\n    if isinstance(schema_item, dict):\n        if \"$ref\" in schema_item:\n            schema_item[\"$ref\"] = schema_item[\"$ref\"].replace(\"#/definitions/\", new_ref)\n        for value in schema_item.values():\n            update_refs_in_schema(value, new_ref)\n    elif isinstance(schema_item, list):\n        for item in schema_item:\n            update_refs_in_schema(item, new_ref)\n
    ","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runner/utils/#prefect.runner.utils.update_refs_to_components","title":"update_refs_to_components","text":"

    Updates all $ref fields in the OpenAPI schema to reference the components section.

    Parameters:

    Name Type Description Default openapi_schema Dict[str, Any]

    The OpenAPI schema to modify $ref fields in.

    required Source code in prefect/runner/utils.py
    def update_refs_to_components(openapi_schema: Dict[str, Any]) -> Dict[str, Any]:\n    \"\"\"\n    Updates all `$ref` fields in the OpenAPI schema to reference the components section.\n\n    Args:\n        openapi_schema: The OpenAPI schema to modify `$ref` fields in.\n    \"\"\"\n    for path_item in openapi_schema.get(\"paths\", {}).values():\n        for operation in path_item.values():\n            schema = (\n                operation.get(\"requestBody\", {})\n                .get(\"content\", {})\n                .get(\"application/json\", {})\n                .get(\"schema\", {})\n            )\n            update_refs_in_schema(schema, \"#/components/schemas/\")\n\n    for definition in openapi_schema.get(\"definitions\", {}).values():\n        update_refs_in_schema(definition, \"#/components/schemas/\")\n\n    return openapi_schema\n
    ","tags":["Python API","runner","utilities"]},{"location":"api-ref/prefect/runtime/deployment/","title":"deployment","text":"","tags":["Python API","deployment context","context"]},{"location":"api-ref/prefect/runtime/deployment/#prefect.runtime.deployment","title":"prefect.runtime.deployment","text":"

    Access attributes of the current deployment run dynamically.

    Note that if a deployment is not currently being run, all attributes will return empty values.

    You can mock the runtime attributes for testing purposes by setting environment variables prefixed with PREFECT__RUNTIME__DEPLOYMENT.

    Example usage
    from prefect.runtime import deployment\n\ndef get_task_runner():\n    task_runner_config = deployment.parameters.get(\"runner_config\", \"default config here\")\n    return DummyTaskRunner(task_runner_specs=task_runner_config)\n
    Available attributes
    • id: the deployment's unique ID
    • name: the deployment's name
    • version: the deployment's version
    • flow_run_id: the current flow run ID for this deployment
    • parameters: the parameters that were passed to this run; note that these do not necessarily include default values set on the flow function, only the parameter values set on the deployment object or those directly provided via API for this run
    ","tags":["Python API","deployment context","context"]},{"location":"api-ref/prefect/runtime/flow_run/","title":"flow_run","text":"","tags":["Python API","flow run context","context"]},{"location":"api-ref/prefect/runtime/flow_run/#prefect.runtime.flow_run","title":"prefect.runtime.flow_run","text":"

    Access attributes of the current flow run dynamically.

    Note that if a flow run cannot be discovered, all attributes will return empty values.

    You can mock the runtime attributes for testing purposes by setting environment variables prefixed with PREFECT__RUNTIME__FLOW_RUN.

    Available attributes
    • id: the flow run's unique ID
    • tags: the flow run's set of tags
    • scheduled_start_time: the flow run's expected scheduled start time; defaults to now if not present
    • name: the name of the flow run
    • flow_name: the name of the flow
    • parameters: the parameters that were passed to this run; note that these do not necessarily include default values set on the flow function, only the parameter values explicitly passed for the run
    • parent_flow_run_id: the ID of the flow run that triggered this run, if any
    • parent_deployment_id: the ID of the deployment that triggered this run, if any
    • run_count: the number of times this flow run has been run
    ","tags":["Python API","flow run context","context"]},{"location":"api-ref/prefect/runtime/task_run/","title":"task_run","text":"","tags":["Python API","task run context","context","task run"]},{"location":"api-ref/prefect/runtime/task_run/#prefect.runtime.task_run","title":"prefect.runtime.task_run","text":"

    Access attributes of the current task run dynamically.

    Note that if a task run cannot be discovered, all attributes will return empty values.

    You can mock the runtime attributes for testing purposes by setting environment variables prefixed with PREFECT__RUNTIME__TASK_RUN.

    Available attributes
    • id: the task run's unique ID
    • name: the name of the task run
    • tags: the task run's set of tags
    • parameters: the parameters the task was called with
    • run_count: the number of times this task run has been run
    • task_name: the name of the task
    ","tags":["Python API","task run context","context","task run"]},{"location":"api-ref/prefect/utilities/annotations/","title":"annotations","text":"","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations","title":"prefect.utilities.annotations","text":"","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.BaseAnnotation","title":"BaseAnnotation","text":"

    Bases: namedtuple('BaseAnnotation', field_names='value'), ABC, Generic[T]

    Base class for Prefect annotation types.

    Inherits from namedtuple for unpacking support in another tools.

    Source code in prefect/utilities/annotations.py
    class BaseAnnotation(\n    namedtuple(\"BaseAnnotation\", field_names=\"value\"), ABC, Generic[T]\n):\n    \"\"\"\n    Base class for Prefect annotation types.\n\n    Inherits from `namedtuple` for unpacking support in another tools.\n    \"\"\"\n\n    def unwrap(self) -> T:\n        if sys.version_info < (3, 8):\n            # cannot simply return self.value due to recursion error in Python 3.7\n            # also _asdict does not follow convention; it's not an internal method\n            # https://stackoverflow.com/a/26180604\n            return self._asdict()[\"value\"]\n        else:\n            return self.value\n\n    def rewrap(self, value: T) -> \"BaseAnnotation[T]\":\n        return type(self)(value)\n\n    def __eq__(self, other: object) -> bool:\n        if not type(self) == type(other):\n            return False\n        return self.unwrap() == other.unwrap()\n\n    def __repr__(self) -> str:\n        return f\"{type(self).__name__}({self.value!r})\"\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.NotSet","title":"NotSet","text":"

    Singleton to distinguish None from a value that is not provided by the user.

    Source code in prefect/utilities/annotations.py
    class NotSet:\n    \"\"\"\n    Singleton to distinguish `None` from a value that is not provided by the user.\n    \"\"\"\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.allow_failure","title":"allow_failure","text":"

    Bases: BaseAnnotation[T]

    Wrapper for states or futures.

    Indicates that the upstream run for this input can be failed.

    Generally, Prefect will not allow a downstream run to start if any of its inputs are failed. This annotation allows you to opt into receiving a failed input downstream.

    If the input is from a failed run, the attached exception will be passed to your function.

    Source code in prefect/utilities/annotations.py
    class allow_failure(BaseAnnotation[T]):\n    \"\"\"\n    Wrapper for states or futures.\n\n    Indicates that the upstream run for this input can be failed.\n\n    Generally, Prefect will not allow a downstream run to start if any of its inputs\n    are failed. This annotation allows you to opt into receiving a failed input\n    downstream.\n\n    If the input is from a failed run, the attached exception will be passed to your\n    function.\n    \"\"\"\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.quote","title":"quote","text":"

    Bases: BaseAnnotation[T]

    Simple wrapper to mark an expression as a different type so it will not be coerced by Prefect. For example, if you want to return a state from a flow without having the flow assume that state.

    quote will also instruct prefect to ignore introspection of the wrapped object when passed as flow or task parameter. Parameter introspection can be a significant performance hit when the object is a large collection, e.g. a large dictionary or DataFrame, and each element needs to be visited. This will disable task dependency tracking for the wrapped object, but likely will increase performance.

    @task\ndef my_task(df):\n    ...\n\n@flow\ndef my_flow():\n    my_task(quote(df))\n
    Source code in prefect/utilities/annotations.py
    class quote(BaseAnnotation[T]):\n    \"\"\"\n    Simple wrapper to mark an expression as a different type so it will not be coerced\n    by Prefect. For example, if you want to return a state from a flow without having\n    the flow assume that state.\n\n    quote will also instruct prefect to ignore introspection of the wrapped object\n    when passed as flow or task parameter. Parameter introspection can be a\n    significant performance hit when the object is a large collection,\n    e.g. a large dictionary or DataFrame, and each element needs to be visited. This\n    will disable task dependency tracking for the wrapped object, but likely will\n    increase performance.\n\n    ```\n    @task\n    def my_task(df):\n        ...\n\n    @flow\n    def my_flow():\n        my_task(quote(df))\n    ```\n    \"\"\"\n\n    def unquote(self) -> T:\n        return self.unwrap()\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/annotations/#prefect.utilities.annotations.unmapped","title":"unmapped","text":"

    Bases: BaseAnnotation[T]

    Wrapper for iterables.

    Indicates that this input should be sent as-is to all runs created during a mapping operation instead of being split.

    Source code in prefect/utilities/annotations.py
    class unmapped(BaseAnnotation[T]):\n    \"\"\"\n    Wrapper for iterables.\n\n    Indicates that this input should be sent as-is to all runs created during a mapping\n    operation instead of being split.\n    \"\"\"\n\n    def __getitem__(self, _) -> T:\n        # Internally, this acts as an infinite array where all items are the same value\n        return self.unwrap()\n
    ","tags":["Python API","annotations"]},{"location":"api-ref/prefect/utilities/asyncutils/","title":"asyncutils","text":"","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils","title":"prefect.utilities.asyncutils","text":"

    Utilities for interoperability with async functions and workers from various contexts.

    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.GatherIncomplete","title":"GatherIncomplete","text":"

    Bases: RuntimeError

    Used to indicate retrieving gather results before completion

    Source code in prefect/utilities/asyncutils.py
    class GatherIncomplete(RuntimeError):\n    \"\"\"Used to indicate retrieving gather results before completion\"\"\"\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.GatherTaskGroup","title":"GatherTaskGroup","text":"

    Bases: TaskGroup

    A task group that gathers results.

    AnyIO does not include support gather. This class extends the TaskGroup interface to allow simple gathering.

    See https://github.com/agronholm/anyio/issues/100

    This class should be instantiated with create_gather_task_group.

    Source code in prefect/utilities/asyncutils.py
    class GatherTaskGroup(anyio.abc.TaskGroup):\n    \"\"\"\n    A task group that gathers results.\n\n    AnyIO does not include support `gather`. This class extends the `TaskGroup`\n    interface to allow simple gathering.\n\n    See https://github.com/agronholm/anyio/issues/100\n\n    This class should be instantiated with `create_gather_task_group`.\n    \"\"\"\n\n    def __init__(self, task_group: anyio.abc.TaskGroup):\n        self._results: Dict[UUID, Any] = {}\n        # The concrete task group implementation to use\n        self._task_group: anyio.abc.TaskGroup = task_group\n\n    async def _run_and_store(self, key, fn, args):\n        self._results[key] = await fn(*args)\n\n    def start_soon(self, fn, *args) -> UUID:\n        key = uuid4()\n        # Put a placeholder in-case the result is retrieved earlier\n        self._results[key] = GatherIncomplete\n        self._task_group.start_soon(self._run_and_store, key, fn, args)\n        return key\n\n    async def start(self, fn, *args):\n        \"\"\"\n        Since `start` returns the result of `task_status.started()` but here we must\n        return the key instead, we just won't support this method for now.\n        \"\"\"\n        raise RuntimeError(\"`GatherTaskGroup` does not support `start`.\")\n\n    def get_result(self, key: UUID) -> Any:\n        result = self._results[key]\n        if result is GatherIncomplete:\n            raise GatherIncomplete(\n                \"Task is not complete. \"\n                \"Results should not be retrieved until the task group exits.\"\n            )\n        return result\n\n    async def __aenter__(self):\n        await self._task_group.__aenter__()\n        return self\n\n    async def __aexit__(self, *tb):\n        try:\n            retval = await self._task_group.__aexit__(*tb)\n            return retval\n        finally:\n            del self._task_group\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.GatherTaskGroup.start","title":"start async","text":"

    Since start returns the result of task_status.started() but here we must return the key instead, we just won't support this method for now.

    Source code in prefect/utilities/asyncutils.py
    async def start(self, fn, *args):\n    \"\"\"\n    Since `start` returns the result of `task_status.started()` but here we must\n    return the key instead, we just won't support this method for now.\n    \"\"\"\n    raise RuntimeError(\"`GatherTaskGroup` does not support `start`.\")\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.add_event_loop_shutdown_callback","title":"add_event_loop_shutdown_callback async","text":"

    Adds a callback to the given callable on event loop closure. The callable must be a coroutine function. It will be awaited when the current event loop is shutting down.

    Requires use of asyncio.run() which waits for async generator shutdown by default or explicit call of asyncio.shutdown_asyncgens(). If the application is entered with asyncio.run_until_complete() and the user calls asyncio.close() without the generator shutdown call, this will not trigger callbacks.

    asyncio does not provided any other way to clean up a resource when the event loop is about to close.

    Source code in prefect/utilities/asyncutils.py
    async def add_event_loop_shutdown_callback(coroutine_fn: Callable[[], Awaitable]):\n    \"\"\"\n    Adds a callback to the given callable on event loop closure. The callable must be\n    a coroutine function. It will be awaited when the current event loop is shutting\n    down.\n\n    Requires use of `asyncio.run()` which waits for async generator shutdown by\n    default or explicit call of `asyncio.shutdown_asyncgens()`. If the application\n    is entered with `asyncio.run_until_complete()` and the user calls\n    `asyncio.close()` without the generator shutdown call, this will not trigger\n    callbacks.\n\n    asyncio does not provided _any_ other way to clean up a resource when the event\n    loop is about to close.\n    \"\"\"\n\n    async def on_shutdown(key):\n        # It appears that EVENT_LOOP_GC_REFS is somehow being garbage collected early.\n        # We hold a reference to it so as to preserve it, at least for the lifetime of\n        # this coroutine. See the issue below for the initial report/discussion:\n        # https://github.com/PrefectHQ/prefect/issues/7709#issuecomment-1560021109\n        _ = EVENT_LOOP_GC_REFS\n        try:\n            yield\n        except GeneratorExit:\n            await coroutine_fn()\n            # Remove self from the garbage collection set\n            EVENT_LOOP_GC_REFS.pop(key)\n\n    # Create the iterator and store it in a global variable so it is not garbage\n    # collected. If the iterator is garbage collected before the event loop closes, the\n    # callback will not run. Since this function does not know the scope of the event\n    # loop that is calling it, a reference with global scope is necessary to ensure\n    # garbage collection does not occur until after event loop closure.\n    key = id(on_shutdown)\n    EVENT_LOOP_GC_REFS[key] = on_shutdown(key)\n\n    # Begin iterating so it will be cleaned up as an incomplete generator\n    await EVENT_LOOP_GC_REFS[key].__anext__()\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.create_gather_task_group","title":"create_gather_task_group","text":"

    Create a new task group that gathers results

    Source code in prefect/utilities/asyncutils.py
    def create_gather_task_group() -> GatherTaskGroup:\n    \"\"\"Create a new task group that gathers results\"\"\"\n    # This function matches the AnyIO API which uses callables since the concrete\n    # task group class depends on the async library being used and cannot be\n    # determined until runtime\n    return GatherTaskGroup(anyio.create_task_group())\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.gather","title":"gather async","text":"

    Run calls concurrently and gather their results.

    Unlike asyncio.gather this expects to receive callables not coroutines. This matches anyio semantics.

    Source code in prefect/utilities/asyncutils.py
    async def gather(*calls: Callable[[], Coroutine[Any, Any, T]]) -> List[T]:\n    \"\"\"\n    Run calls concurrently and gather their results.\n\n    Unlike `asyncio.gather` this expects to receive _callables_ not _coroutines_.\n    This matches `anyio` semantics.\n    \"\"\"\n    keys = []\n    async with create_gather_task_group() as tg:\n        for call in calls:\n            keys.append(tg.start_soon(call))\n    return [tg.get_result(key) for key in keys]\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.is_async_fn","title":"is_async_fn","text":"

    Returns True if a function returns a coroutine.

    See https://github.com/microsoft/pyright/issues/2142 for an example use

    Source code in prefect/utilities/asyncutils.py
    def is_async_fn(\n    func: Union[Callable[P, R], Callable[P, Awaitable[R]]]\n) -> TypeGuard[Callable[P, Awaitable[R]]]:\n    \"\"\"\n    Returns `True` if a function returns a coroutine.\n\n    See https://github.com/microsoft/pyright/issues/2142 for an example use\n    \"\"\"\n    while hasattr(func, \"__wrapped__\"):\n        func = func.__wrapped__\n\n    return inspect.iscoroutinefunction(func)\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.is_async_gen_fn","title":"is_async_gen_fn","text":"

    Returns True if a function is an async generator.

    Source code in prefect/utilities/asyncutils.py
    def is_async_gen_fn(func):\n    \"\"\"\n    Returns `True` if a function is an async generator.\n    \"\"\"\n    while hasattr(func, \"__wrapped__\"):\n        func = func.__wrapped__\n\n    return inspect.isasyncgenfunction(func)\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.raise_async_exception_in_thread","title":"raise_async_exception_in_thread","text":"

    Raise an exception in a thread asynchronously.

    This will not interrupt long-running system calls like sleep or wait.

    Source code in prefect/utilities/asyncutils.py
    def raise_async_exception_in_thread(thread: Thread, exc_type: Type[BaseException]):\n    \"\"\"\n    Raise an exception in a thread asynchronously.\n\n    This will not interrupt long-running system calls like `sleep` or `wait`.\n    \"\"\"\n    ret = ctypes.pythonapi.PyThreadState_SetAsyncExc(\n        ctypes.c_long(thread.ident), ctypes.py_object(exc_type)\n    )\n    if ret == 0:\n        raise ValueError(\"Thread not found.\")\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.run_async_from_worker_thread","title":"run_async_from_worker_thread","text":"

    Runs an async function in the main thread's event loop, blocking the worker thread until completion

    Source code in prefect/utilities/asyncutils.py
    def run_async_from_worker_thread(\n    __fn: Callable[..., Awaitable[T]], *args: Any, **kwargs: Any\n) -> T:\n    \"\"\"\n    Runs an async function in the main thread's event loop, blocking the worker\n    thread until completion\n    \"\"\"\n    call = partial(__fn, *args, **kwargs)\n    return anyio.from_thread.run(call)\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.run_sync_in_interruptible_worker_thread","title":"run_sync_in_interruptible_worker_thread async","text":"

    Runs a sync function in a new interruptible worker thread so that the main thread's event loop is not blocked

    Unlike the anyio function, this performs best-effort cancellation of the thread using the C API. Cancellation will not interrupt system calls like sleep.

    Source code in prefect/utilities/asyncutils.py
    async def run_sync_in_interruptible_worker_thread(\n    __fn: Callable[..., T], *args: Any, **kwargs: Any\n) -> T:\n    \"\"\"\n    Runs a sync function in a new interruptible worker thread so that the main\n    thread's event loop is not blocked\n\n    Unlike the anyio function, this performs best-effort cancellation of the\n    thread using the C API. Cancellation will not interrupt system calls like\n    `sleep`.\n    \"\"\"\n\n    class NotSet:\n        pass\n\n    thread: Thread = None\n    result = NotSet\n    event = asyncio.Event()\n    loop = asyncio.get_running_loop()\n\n    def capture_worker_thread_and_result():\n        # Captures the worker thread that AnyIO is using to execute the function so\n        # the main thread can perform actions on it\n        nonlocal thread, result\n        try:\n            thread = threading.current_thread()\n            result = __fn(*args, **kwargs)\n        except BaseException as exc:\n            result = exc\n            raise\n        finally:\n            loop.call_soon_threadsafe(event.set)\n\n    async def send_interrupt_to_thread():\n        # This task waits until the result is returned from the thread, if cancellation\n        # occurs during that time, we will raise the exception in the thread as well\n        try:\n            await event.wait()\n        except anyio.get_cancelled_exc_class():\n            # NOTE: We could send a SIGINT here which allow us to interrupt system\n            # calls but the interrupt bubbles from the child thread into the main thread\n            # and there is not a clear way to prevent it.\n            raise_async_exception_in_thread(thread, anyio.get_cancelled_exc_class())\n            raise\n\n    async with anyio.create_task_group() as tg:\n        tg.start_soon(send_interrupt_to_thread)\n        tg.start_soon(\n            partial(\n                anyio.to_thread.run_sync,\n                capture_worker_thread_and_result,\n                cancellable=True,\n                limiter=get_thread_limiter(),\n            )\n        )\n\n    assert result is not NotSet\n    return result\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.run_sync_in_worker_thread","title":"run_sync_in_worker_thread async","text":"

    Runs a sync function in a new worker thread so that the main thread's event loop is not blocked

    Unlike the anyio function, this defaults to a cancellable thread and does not allow passing arguments to the anyio function so users can pass kwargs to their function.

    Note that cancellation of threads will not result in interrupted computation, the thread may continue running \u2014 the outcome will just be ignored.

    Source code in prefect/utilities/asyncutils.py
    async def run_sync_in_worker_thread(\n    __fn: Callable[..., T], *args: Any, **kwargs: Any\n) -> T:\n    \"\"\"\n    Runs a sync function in a new worker thread so that the main thread's event loop\n    is not blocked\n\n    Unlike the anyio function, this defaults to a cancellable thread and does not allow\n    passing arguments to the anyio function so users can pass kwargs to their function.\n\n    Note that cancellation of threads will not result in interrupted computation, the\n    thread may continue running \u2014 the outcome will just be ignored.\n    \"\"\"\n    call = partial(__fn, *args, **kwargs)\n    return await anyio.to_thread.run_sync(\n        call, cancellable=True, limiter=get_thread_limiter()\n    )\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.sync","title":"sync","text":"

    Call an async function from a synchronous context. Block until completion.

    If in an asynchronous context, we will run the code in a separate loop instead of failing but a warning will be displayed since this is not recommended.

    Source code in prefect/utilities/asyncutils.py
    def sync(__async_fn: Callable[P, Awaitable[T]], *args: P.args, **kwargs: P.kwargs) -> T:\n    \"\"\"\n    Call an async function from a synchronous context. Block until completion.\n\n    If in an asynchronous context, we will run the code in a separate loop instead of\n    failing but a warning will be displayed since this is not recommended.\n    \"\"\"\n    if in_async_main_thread():\n        warnings.warn(\n            \"`sync` called from an asynchronous context; \"\n            \"you should `await` the async function directly instead.\"\n        )\n        with anyio.start_blocking_portal() as portal:\n            return portal.call(partial(__async_fn, *args, **kwargs))\n    elif in_async_worker_thread():\n        # In a sync context but we can access the event loop thread; send the async\n        # call to the parent\n        return run_async_from_worker_thread(__async_fn, *args, **kwargs)\n    else:\n        # In a sync context and there is no event loop; just create an event loop\n        # to run the async code then tear it down\n        return run_async_in_new_loop(__async_fn, *args, **kwargs)\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/asyncutils/#prefect.utilities.asyncutils.sync_compatible","title":"sync_compatible","text":"

    Converts an async function into a dual async and sync function.

    When the returned function is called, we will attempt to determine the best way to enter the async function.

    • If in a thread with a running event loop, we will return the coroutine for the caller to await. This is normal async behavior.
    • If in a blocking worker thread with access to an event loop in another thread, we will submit the async method to the event loop.
    • If we cannot find an event loop, we will create a new one and run the async method then tear down the loop.
    Source code in prefect/utilities/asyncutils.py
    def sync_compatible(async_fn: T) -> T:\n    \"\"\"\n    Converts an async function into a dual async and sync function.\n\n    When the returned function is called, we will attempt to determine the best way\n    to enter the async function.\n\n    - If in a thread with a running event loop, we will return the coroutine for the\n        caller to await. This is normal async behavior.\n    - If in a blocking worker thread with access to an event loop in another thread, we\n        will submit the async method to the event loop.\n    - If we cannot find an event loop, we will create a new one and run the async method\n        then tear down the loop.\n    \"\"\"\n\n    @wraps(async_fn)\n    def coroutine_wrapper(*args, **kwargs):\n        from prefect._internal.concurrency.api import create_call, from_sync\n        from prefect._internal.concurrency.calls import get_current_call, logger\n        from prefect._internal.concurrency.event_loop import get_running_loop\n        from prefect._internal.concurrency.threads import get_global_loop\n\n        global_thread_portal = get_global_loop()\n        current_thread = threading.current_thread()\n        current_call = get_current_call()\n        current_loop = get_running_loop()\n\n        if current_thread.ident == global_thread_portal.thread.ident:\n            logger.debug(f\"{async_fn} --> return coroutine for internal await\")\n            # In the prefect async context; return the coro for us to await\n            return async_fn(*args, **kwargs)\n        elif in_async_main_thread() and (\n            not current_call or is_async_fn(current_call.fn)\n        ):\n            # In the main async context; return the coro for them to await\n            logger.debug(f\"{async_fn} --> return coroutine for user await\")\n            return async_fn(*args, **kwargs)\n        elif in_async_worker_thread():\n            # In a sync context but we can access the event loop thread; send the async\n            # call to the parent\n            return run_async_from_worker_thread(async_fn, *args, **kwargs)\n        elif current_loop is not None:\n            logger.debug(f\"{async_fn} --> run async in global loop portal\")\n            # An event loop is already present but we are in a sync context, run the\n            # call in Prefect's event loop thread\n            return from_sync.call_soon_in_loop_thread(\n                create_call(async_fn, *args, **kwargs)\n            ).result()\n        else:\n            logger.debug(f\"{async_fn} --> run async in new loop\")\n            # Run in a new event loop, but use a `Call` for nested context detection\n            call = create_call(async_fn, *args, **kwargs)\n            return call()\n\n    # TODO: This is breaking type hints on the callable... mypy is behind the curve\n    #       on argument annotations. We can still fix this for editors though.\n    if is_async_fn(async_fn):\n        wrapper = coroutine_wrapper\n    elif is_async_gen_fn(async_fn):\n        raise ValueError(\"Async generators cannot yet be marked as `sync_compatible`\")\n    else:\n        raise TypeError(\"The decorated function must be async.\")\n\n    wrapper.aio = async_fn\n    return wrapper\n
    ","tags":["Python API","async"]},{"location":"api-ref/prefect/utilities/callables/","title":"callables","text":"","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables","title":"prefect.utilities.callables","text":"

    Utilities for working with Python callables.

    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.ParameterSchema","title":"ParameterSchema","text":"

    Bases: BaseModel

    Simple data model corresponding to an OpenAPI Schema.

    Source code in prefect/utilities/callables.py
    class ParameterSchema(pydantic.BaseModel):\n    \"\"\"Simple data model corresponding to an OpenAPI `Schema`.\"\"\"\n\n    title: Literal[\"Parameters\"] = \"Parameters\"\n    type: Literal[\"object\"] = \"object\"\n    properties: Dict[str, Any] = pydantic.Field(default_factory=dict)\n    required: List[str] = None\n    definitions: Dict[str, Any] = None\n\n    def dict(self, *args, **kwargs):\n        \"\"\"Exclude `None` fields by default to comply with\n        the OpenAPI spec.\n        \"\"\"\n        kwargs.setdefault(\"exclude_none\", True)\n        return super().dict(*args, **kwargs)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.ParameterSchema.dict","title":"dict","text":"

    Exclude None fields by default to comply with the OpenAPI spec.

    Source code in prefect/utilities/callables.py
    def dict(self, *args, **kwargs):\n    \"\"\"Exclude `None` fields by default to comply with\n    the OpenAPI spec.\n    \"\"\"\n    kwargs.setdefault(\"exclude_none\", True)\n    return super().dict(*args, **kwargs)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.call_with_parameters","title":"call_with_parameters","text":"

    Call a function with parameters extracted with get_call_parameters

    The function must have an identical signature to the original function or this will fail. If you need to send to a function with a different signature, extract the args/kwargs using parameters_to_positional_and_keyword directly

    Source code in prefect/utilities/callables.py
    def call_with_parameters(fn: Callable, parameters: Dict[str, Any]):\n    \"\"\"\n    Call a function with parameters extracted with `get_call_parameters`\n\n    The function _must_ have an identical signature to the original function or this\n    will fail. If you need to send to a function with a different signature, extract\n    the args/kwargs using `parameters_to_positional_and_keyword` directly\n    \"\"\"\n    args, kwargs = parameters_to_args_kwargs(fn, parameters)\n    return fn(*args, **kwargs)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.cloudpickle_wrapped_call","title":"cloudpickle_wrapped_call","text":"

    Serializes a function call using cloudpickle then returns a callable which will execute that call and return a cloudpickle serialized return value

    This is particularly useful for sending calls to libraries that only use the Python built-in pickler (e.g. anyio.to_process and multiprocessing) but may require a wider range of pickling support.

    Source code in prefect/utilities/callables.py
    def cloudpickle_wrapped_call(\n    __fn: Callable, *args: Any, **kwargs: Any\n) -> Callable[[], bytes]:\n    \"\"\"\n    Serializes a function call using cloudpickle then returns a callable which will\n    execute that call and return a cloudpickle serialized return value\n\n    This is particularly useful for sending calls to libraries that only use the Python\n    built-in pickler (e.g. `anyio.to_process` and `multiprocessing`) but may require\n    a wider range of pickling support.\n    \"\"\"\n    payload = cloudpickle.dumps((__fn, args, kwargs))\n    return partial(_run_serialized_call, payload)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.collapse_variadic_parameters","title":"collapse_variadic_parameters","text":"

    Given a parameter dictionary, move any parameters stored not present in the signature into the variadic keyword argument.

    Example:

    ```python\ndef foo(a, b, **kwargs):\n    pass\n\nparameters = {\"a\": 1, \"b\": 2, \"c\": 3, \"d\": 4}\ncollapse_variadic_parameters(foo, parameters)\n# {\"a\": 1, \"b\": 2, \"kwargs\": {\"c\": 3, \"d\": 4}}\n```\n
    Source code in prefect/utilities/callables.py
    def collapse_variadic_parameters(\n    fn: Callable, parameters: Dict[str, Any]\n) -> Dict[str, Any]:\n    \"\"\"\n    Given a parameter dictionary, move any parameters stored not present in the\n    signature into the variadic keyword argument.\n\n    Example:\n\n        ```python\n        def foo(a, b, **kwargs):\n            pass\n\n        parameters = {\"a\": 1, \"b\": 2, \"c\": 3, \"d\": 4}\n        collapse_variadic_parameters(foo, parameters)\n        # {\"a\": 1, \"b\": 2, \"kwargs\": {\"c\": 3, \"d\": 4}}\n        ```\n    \"\"\"\n    signature_parameters = inspect.signature(fn).parameters\n    variadic_key = None\n    for key, parameter in signature_parameters.items():\n        if parameter.kind == parameter.VAR_KEYWORD:\n            variadic_key = key\n            break\n\n    missing_parameters = set(parameters.keys()) - set(signature_parameters.keys())\n\n    if not variadic_key and missing_parameters:\n        raise ValueError(\n            f\"Signature for {fn} does not include any variadic keyword argument \"\n            \"but parameters were given that are not present in the signature.\"\n        )\n\n    if variadic_key and not missing_parameters:\n        # variadic key is present but no missing parameters, return parameters unchanged\n        return parameters\n\n    new_parameters = parameters.copy()\n    if variadic_key:\n        new_parameters[variadic_key] = {}\n\n    for key in missing_parameters:\n        new_parameters[variadic_key][key] = new_parameters.pop(key)\n\n    return new_parameters\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.explode_variadic_parameter","title":"explode_variadic_parameter","text":"

    Given a parameter dictionary, move any parameters stored in a variadic keyword argument parameter (i.e. **kwargs) into the top level.

    Example:

    ```python\ndef foo(a, b, **kwargs):\n    pass\n\nparameters = {\"a\": 1, \"b\": 2, \"kwargs\": {\"c\": 3, \"d\": 4}}\nexplode_variadic_parameter(foo, parameters)\n# {\"a\": 1, \"b\": 2, \"c\": 3, \"d\": 4}\n```\n
    Source code in prefect/utilities/callables.py
    def explode_variadic_parameter(\n    fn: Callable, parameters: Dict[str, Any]\n) -> Dict[str, Any]:\n    \"\"\"\n    Given a parameter dictionary, move any parameters stored in a variadic keyword\n    argument parameter (i.e. **kwargs) into the top level.\n\n    Example:\n\n        ```python\n        def foo(a, b, **kwargs):\n            pass\n\n        parameters = {\"a\": 1, \"b\": 2, \"kwargs\": {\"c\": 3, \"d\": 4}}\n        explode_variadic_parameter(foo, parameters)\n        # {\"a\": 1, \"b\": 2, \"c\": 3, \"d\": 4}\n        ```\n    \"\"\"\n    variadic_key = None\n    for key, parameter in inspect.signature(fn).parameters.items():\n        if parameter.kind == parameter.VAR_KEYWORD:\n            variadic_key = key\n            break\n\n    if not variadic_key:\n        return parameters\n\n    new_parameters = parameters.copy()\n    for key, value in new_parameters.pop(variadic_key, {}).items():\n        new_parameters[key] = value\n\n    return new_parameters\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.get_call_parameters","title":"get_call_parameters","text":"

    Bind a call to a function to get parameter/value mapping. Default values on the signature will be included if not overridden.

    Raises a ParameterBindError if the arguments/kwargs are not valid for the function

    Source code in prefect/utilities/callables.py
    def get_call_parameters(\n    fn: Callable,\n    call_args: Tuple[Any, ...],\n    call_kwargs: Dict[str, Any],\n    apply_defaults: bool = True,\n) -> Dict[str, Any]:\n    \"\"\"\n    Bind a call to a function to get parameter/value mapping. Default values on the\n    signature will be included if not overridden.\n\n    Raises a ParameterBindError if the arguments/kwargs are not valid for the function\n    \"\"\"\n    try:\n        bound_signature = inspect.signature(fn).bind(*call_args, **call_kwargs)\n    except TypeError as exc:\n        raise ParameterBindError.from_bind_failure(fn, exc, call_args, call_kwargs)\n\n    if apply_defaults:\n        bound_signature.apply_defaults()\n\n    # We cast from `OrderedDict` to `dict` because Dask will not convert futures in an\n    # ordered dictionary to values during execution; this is the default behavior in\n    # Python 3.9 anyway.\n    return dict(bound_signature.arguments)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.get_parameter_defaults","title":"get_parameter_defaults","text":"

    Get default parameter values for a callable.

    Source code in prefect/utilities/callables.py
    def get_parameter_defaults(\n    fn: Callable,\n) -> Dict[str, Any]:\n    \"\"\"\n    Get default parameter values for a callable.\n    \"\"\"\n    signature = inspect.signature(fn)\n\n    parameter_defaults = {}\n\n    for name, param in signature.parameters.items():\n        if param.default is not signature.empty:\n            parameter_defaults[name] = param.default\n\n    return parameter_defaults\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.parameter_docstrings","title":"parameter_docstrings","text":"

    Given a docstring in Google docstring format, parse the parameter section and return a dictionary that maps parameter names to docstring.

    Parameters:

    Name Type Description Default docstring Optional[str]

    The function's docstring.

    required

    Returns:

    Type Description Dict[str, str]

    Mapping from parameter names to docstrings.

    Source code in prefect/utilities/callables.py
    def parameter_docstrings(docstring: Optional[str]) -> Dict[str, str]:\n    \"\"\"\n    Given a docstring in Google docstring format, parse the parameter section\n    and return a dictionary that maps parameter names to docstring.\n\n    Args:\n        docstring: The function's docstring.\n\n    Returns:\n        Mapping from parameter names to docstrings.\n    \"\"\"\n    param_docstrings = {}\n\n    if not docstring:\n        return param_docstrings\n\n    with disable_logger(\"griffe.docstrings.google\"), disable_logger(\n        \"griffe.agents.nodes\"\n    ):\n        parsed = parse(Docstring(docstring), Parser.google)\n        for section in parsed:\n            if section.kind != DocstringSectionKind.parameters:\n                continue\n            param_docstrings = {\n                parameter.name: parameter.description for parameter in section.value\n            }\n\n    return param_docstrings\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.parameter_schema","title":"parameter_schema","text":"

    Given a function, generates an OpenAPI-compatible description of the function's arguments, including: - name - typing information - whether it is required - a default value - additional constraints (like possible enum values)

    Parameters:

    Name Type Description Default fn Callable

    The function whose arguments will be serialized

    required

    Returns:

    Name Type Description ParameterSchema ParameterSchema

    the argument schema

    Source code in prefect/utilities/callables.py
    def parameter_schema(fn: Callable) -> ParameterSchema:\n    \"\"\"Given a function, generates an OpenAPI-compatible description\n    of the function's arguments, including:\n        - name\n        - typing information\n        - whether it is required\n        - a default value\n        - additional constraints (like possible enum values)\n\n    Args:\n        fn (Callable): The function whose arguments will be serialized\n\n    Returns:\n        ParameterSchema: the argument schema\n    \"\"\"\n    signature = inspect.signature(fn)\n    model_fields = {}\n    aliases = {}\n    docstrings = parameter_docstrings(inspect.getdoc(fn))\n\n    class ModelConfig:\n        arbitrary_types_allowed = True\n\n    if HAS_PYDANTIC_V2 and has_v2_model_as_param(signature):\n        create_schema = create_v2_schema\n        process_params = process_v2_params\n    else:\n        create_schema = create_v1_schema\n        process_params = process_v1_params\n\n    for position, param in enumerate(signature.parameters.values()):\n        name, type_, field = process_params(\n            param, position=position, docstrings=docstrings, aliases=aliases\n        )\n        # Generate a Pydantic model at each step so we can check if this parameter\n        # type supports schema generation\n        try:\n            create_schema(\n                \"CheckParameter\", model_cfg=ModelConfig, **{name: (type_, field)}\n            )\n        except ValueError:\n            # This field's type is not valid for schema creation, update it to `Any`\n            type_ = Any\n        model_fields[name] = (type_, field)\n\n    # Generate the final model and schema\n    schema = create_schema(\"Parameters\", model_cfg=ModelConfig, **model_fields)\n    return ParameterSchema(**schema)\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.parameters_to_args_kwargs","title":"parameters_to_args_kwargs","text":"

    Convert a parameters dictionary to positional and keyword arguments

    The function must have an identical signature to the original function or this will return an empty tuple and dict.

    Source code in prefect/utilities/callables.py
    def parameters_to_args_kwargs(\n    fn: Callable,\n    parameters: Dict[str, Any],\n) -> Tuple[Tuple[Any, ...], Dict[str, Any]]:\n    \"\"\"\n    Convert a `parameters` dictionary to positional and keyword arguments\n\n    The function _must_ have an identical signature to the original function or this\n    will return an empty tuple and dict.\n    \"\"\"\n    function_params = dict(inspect.signature(fn).parameters).keys()\n    # Check for parameters that are not present in the function signature\n    unknown_params = parameters.keys() - function_params\n    if unknown_params:\n        raise SignatureMismatchError.from_bad_params(\n            list(function_params), list(parameters.keys())\n        )\n    bound_signature = inspect.signature(fn).bind_partial()\n    bound_signature.arguments = parameters\n\n    return bound_signature.args, bound_signature.kwargs\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/callables/#prefect.utilities.callables.raise_for_reserved_arguments","title":"raise_for_reserved_arguments","text":"

    Raise a ReservedArgumentError if fn has any parameters that conflict with the names contained in reserved_arguments.

    Source code in prefect/utilities/callables.py
    def raise_for_reserved_arguments(fn: Callable, reserved_arguments: Iterable[str]):\n    \"\"\"Raise a ReservedArgumentError if `fn` has any parameters that conflict\n    with the names contained in `reserved_arguments`.\"\"\"\n    function_paremeters = inspect.signature(fn).parameters\n\n    for argument in reserved_arguments:\n        if argument in function_paremeters:\n            raise ReservedArgumentError(\n                f\"{argument!r} is a reserved argument name and cannot be used.\"\n            )\n
    ","tags":["Python API","callables"]},{"location":"api-ref/prefect/utilities/collections/","title":"collections","text":"","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections","title":"prefect.utilities.collections","text":"

    Utilities for extensions of and operations on Python collections.

    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.AutoEnum","title":"AutoEnum","text":"

    Bases: str, Enum

    An enum class that automatically generates value from variable names.

    This guards against common errors where variable names are updated but values are not.

    In addition, because AutoEnums inherit from str, they are automatically JSON-serializable.

    See https://docs.python.org/3/library/enum.html#using-automatic-values

    Example
    class MyEnum(AutoEnum):\n    RED = AutoEnum.auto() # equivalent to RED = 'RED'\n    BLUE = AutoEnum.auto() # equivalent to BLUE = 'BLUE'\n
    Source code in prefect/utilities/collections.py
    class AutoEnum(str, Enum):\n    \"\"\"\n    An enum class that automatically generates value from variable names.\n\n    This guards against common errors where variable names are updated but values are\n    not.\n\n    In addition, because AutoEnums inherit from `str`, they are automatically\n    JSON-serializable.\n\n    See https://docs.python.org/3/library/enum.html#using-automatic-values\n\n    Example:\n        ```python\n        class MyEnum(AutoEnum):\n            RED = AutoEnum.auto() # equivalent to RED = 'RED'\n            BLUE = AutoEnum.auto() # equivalent to BLUE = 'BLUE'\n        ```\n    \"\"\"\n\n    def _generate_next_value_(name, start, count, last_values):\n        return name\n\n    @staticmethod\n    def auto():\n        \"\"\"\n        Exposes `enum.auto()` to avoid requiring a second import to use `AutoEnum`\n        \"\"\"\n        return auto()\n\n    def __repr__(self) -> str:\n        return f\"{type(self).__name__}.{self.value}\"\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.AutoEnum.auto","title":"auto staticmethod","text":"

    Exposes enum.auto() to avoid requiring a second import to use AutoEnum

    Source code in prefect/utilities/collections.py
    @staticmethod\ndef auto():\n    \"\"\"\n    Exposes `enum.auto()` to avoid requiring a second import to use `AutoEnum`\n    \"\"\"\n    return auto()\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.StopVisiting","title":"StopVisiting","text":"

    Bases: BaseException

    A special exception used to stop recursive visits in visit_collection.

    When raised, the expression is returned without modification and recursive visits in that path will end.

    Source code in prefect/utilities/collections.py
    class StopVisiting(BaseException):\n    \"\"\"\n    A special exception used to stop recursive visits in `visit_collection`.\n\n    When raised, the expression is returned without modification and recursive visits\n    in that path will end.\n    \"\"\"\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.batched_iterable","title":"batched_iterable","text":"

    Yield batches of a certain size from an iterable

    Parameters:

    Name Type Description Default iterable Iterable

    An iterable

    required size int

    The batch size to return

    required

    Yields:

    Name Type Description tuple T

    A batch of the iterable

    Source code in prefect/utilities/collections.py
    def batched_iterable(iterable: Iterable[T], size: int) -> Iterator[Tuple[T, ...]]:\n    \"\"\"\n    Yield batches of a certain size from an iterable\n\n    Args:\n        iterable (Iterable): An iterable\n        size (int): The batch size to return\n\n    Yields:\n        tuple: A batch of the iterable\n    \"\"\"\n    it = iter(iterable)\n    while True:\n        batch = tuple(itertools.islice(it, size))\n        if not batch:\n            break\n        yield batch\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.dict_to_flatdict","title":"dict_to_flatdict","text":"

    Converts a (nested) dictionary to a flattened representation.

    Each key of the flat dict will be a CompoundKey tuple containing the \"chain of keys\" for the corresponding value.

    Parameters:

    Name Type Description Default dct dict

    The dictionary to flatten

    required _parent Tuple

    The current parent for recursion

    None

    Returns:

    Type Description Dict[Tuple[KT, ...], Any]

    A flattened dict of the same type as dct

    Source code in prefect/utilities/collections.py
    def dict_to_flatdict(\n    dct: Dict[KT, Union[Any, Dict[KT, Any]]], _parent: Tuple[KT, ...] = None\n) -> Dict[Tuple[KT, ...], Any]:\n    \"\"\"Converts a (nested) dictionary to a flattened representation.\n\n    Each key of the flat dict will be a CompoundKey tuple containing the \"chain of keys\"\n    for the corresponding value.\n\n    Args:\n        dct (dict): The dictionary to flatten\n        _parent (Tuple, optional): The current parent for recursion\n\n    Returns:\n        A flattened dict of the same type as dct\n    \"\"\"\n    typ = cast(Type[Dict[Tuple[KT, ...], Any]], type(dct))\n    items: List[Tuple[Tuple[KT, ...], Any]] = []\n    parent = _parent or tuple()\n\n    for k, v in dct.items():\n        k_parent = tuple(parent + (k,))\n        # if v is a non-empty dict, recurse\n        if isinstance(v, dict) and v:\n            items.extend(dict_to_flatdict(v, _parent=k_parent).items())\n        else:\n            items.append((k_parent, v))\n    return typ(items)\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.extract_instances","title":"extract_instances","text":"

    Extract objects from a file and returns a dict of type -> instances

    Parameters:

    Name Type Description Default objects Iterable

    An iterable of objects

    required types Union[Type[T], Tuple[Type[T], ...]]

    A type or tuple of types to extract, defaults to all objects

    object

    Returns:

    Type Description Union[List[T], Dict[Type[T], T]]

    If a single type is given: a list of instances of that type

    Union[List[T], Dict[Type[T], T]]

    If a tuple of types is given: a mapping of type to a list of instances

    Source code in prefect/utilities/collections.py
    def extract_instances(\n    objects: Iterable,\n    types: Union[Type[T], Tuple[Type[T], ...]] = object,\n) -> Union[List[T], Dict[Type[T], T]]:\n    \"\"\"\n    Extract objects from a file and returns a dict of type -> instances\n\n    Args:\n        objects: An iterable of objects\n        types: A type or tuple of types to extract, defaults to all objects\n\n    Returns:\n        If a single type is given: a list of instances of that type\n        If a tuple of types is given: a mapping of type to a list of instances\n    \"\"\"\n    types = ensure_iterable(types)\n\n    # Create a mapping of type -> instance from the exec values\n    ret = defaultdict(list)\n\n    for o in objects:\n        # We iterate here so that the key is the passed type rather than type(o)\n        for type_ in types:\n            if isinstance(o, type_):\n                ret[type_].append(o)\n\n    if len(types) == 1:\n        return ret[types[0]]\n\n    return ret\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.flatdict_to_dict","title":"flatdict_to_dict","text":"

    Converts a flattened dictionary back to a nested dictionary.

    Parameters:

    Name Type Description Default dct dict

    The dictionary to be nested. Each key should be a tuple of keys as generated by dict_to_flatdict

    required

    Returns A nested dict of the same type as dct

    Source code in prefect/utilities/collections.py
    def flatdict_to_dict(\n    dct: Dict[Tuple[KT, ...], VT]\n) -> Dict[KT, Union[VT, Dict[KT, VT]]]:\n    \"\"\"Converts a flattened dictionary back to a nested dictionary.\n\n    Args:\n        dct (dict): The dictionary to be nested. Each key should be a tuple of keys\n            as generated by `dict_to_flatdict`\n\n    Returns\n        A nested dict of the same type as dct\n    \"\"\"\n    typ = type(dct)\n    result = cast(Dict[KT, Union[VT, Dict[KT, VT]]], typ())\n    for key_tuple, value in dct.items():\n        current_dict = result\n        for prefix_key in key_tuple[:-1]:\n            # Build nested dictionaries up for the current key tuple\n            # Use `setdefault` in case the nested dict has already been created\n            current_dict = current_dict.setdefault(prefix_key, typ())  # type: ignore\n        # Set the value\n        current_dict[key_tuple[-1]] = value\n\n    return result\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.get_from_dict","title":"get_from_dict","text":"

    Fetch a value from a nested dictionary or list using a sequence of keys.

    This function allows to fetch a value from a deeply nested structure of dictionaries and lists using either a dot-separated string or a list of keys. If a requested key does not exist, the function returns the provided default value.

    Parameters:

    Name Type Description Default dct Dict

    The nested dictionary or list from which to fetch the value.

    required keys Union[str, List[str]]

    The sequence of keys to use for access. Can be a dot-separated string or a list of keys. List indices can be included in the sequence as either integer keys or as string indices in square brackets.

    required default Any

    The default value to return if the requested key path does not exist. Defaults to None.

    None

    Returns:

    Type Description Any

    The fetched value if the key exists, or the default value if it does not.

    get_from_dict({'a': {'b': {'c': [1, 2, 3, 4]}}}, 'a.b.c[1]') 2 get_from_dict({'a': {'b': [0, {'c': [1, 2]}]}}, ['a', 'b', 1, 'c', 1]) 2 get_from_dict({'a': {'b': [0, {'c': [1, 2]}]}}, 'a.b.1.c.2', 'default') 'default'

    Source code in prefect/utilities/collections.py
    def get_from_dict(dct: Dict, keys: Union[str, List[str]], default: Any = None) -> Any:\n    \"\"\"\n    Fetch a value from a nested dictionary or list using a sequence of keys.\n\n    This function allows to fetch a value from a deeply nested structure\n    of dictionaries and lists using either a dot-separated string or a list\n    of keys. If a requested key does not exist, the function returns the\n    provided default value.\n\n    Args:\n        dct: The nested dictionary or list from which to fetch the value.\n        keys: The sequence of keys to use for access. Can be a\n            dot-separated string or a list of keys. List indices can be included\n            in the sequence as either integer keys or as string indices in square\n            brackets.\n        default: The default value to return if the requested key path does not\n            exist. Defaults to None.\n\n    Returns:\n        The fetched value if the key exists, or the default value if it does not.\n\n    Examples:\n    >>> get_from_dict({'a': {'b': {'c': [1, 2, 3, 4]}}}, 'a.b.c[1]')\n    2\n    >>> get_from_dict({'a': {'b': [0, {'c': [1, 2]}]}}, ['a', 'b', 1, 'c', 1])\n    2\n    >>> get_from_dict({'a': {'b': [0, {'c': [1, 2]}]}}, 'a.b.1.c.2', 'default')\n    'default'\n    \"\"\"\n    if isinstance(keys, str):\n        keys = keys.replace(\"[\", \".\").replace(\"]\", \"\").split(\".\")\n    try:\n        for key in keys:\n            try:\n                # Try to cast to int to handle list indices\n                key = int(key)\n            except ValueError:\n                # If it's not an int, use the key as-is\n                # for dict lookup\n                pass\n            dct = dct[key]\n        return dct\n    except (TypeError, KeyError, IndexError):\n        return default\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.isiterable","title":"isiterable","text":"

    Return a boolean indicating if an object is iterable.

    Excludes types that are iterable but typically used as singletons: - str - bytes - IO objects

    Source code in prefect/utilities/collections.py
    def isiterable(obj: Any) -> bool:\n    \"\"\"\n    Return a boolean indicating if an object is iterable.\n\n    Excludes types that are iterable but typically used as singletons:\n    - str\n    - bytes\n    - IO objects\n    \"\"\"\n    try:\n        iter(obj)\n    except TypeError:\n        return False\n    else:\n        return not isinstance(obj, (str, bytes, io.IOBase))\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.remove_nested_keys","title":"remove_nested_keys","text":"

    Recurses a dictionary returns a copy without all keys that match an entry in key_to_remove. Return obj unchanged if not a dictionary.

    Parameters:

    Name Type Description Default keys_to_remove List[Hashable]

    A list of keys to remove from obj obj: The object to remove keys from.

    required

    Returns:

    Type Description

    obj without keys matching an entry in keys_to_remove if obj is a dictionary. obj if obj is not a dictionary.

    Source code in prefect/utilities/collections.py
    def remove_nested_keys(keys_to_remove: List[Hashable], obj):\n    \"\"\"\n    Recurses a dictionary returns a copy without all keys that match an entry in\n    `key_to_remove`. Return `obj` unchanged if not a dictionary.\n\n    Args:\n        keys_to_remove: A list of keys to remove from obj obj: The object to remove keys\n            from.\n\n    Returns:\n        `obj` without keys matching an entry in `keys_to_remove` if `obj` is a\n            dictionary. `obj` if `obj` is not a dictionary.\n    \"\"\"\n    if not isinstance(obj, dict):\n        return obj\n    return {\n        key: remove_nested_keys(keys_to_remove, value)\n        for key, value in obj.items()\n        if key not in keys_to_remove\n    }\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/collections/#prefect.utilities.collections.visit_collection","title":"visit_collection","text":"

    This function visits every element of an arbitrary Python collection. If an element is a Python collection, it will be visited recursively. If an element is not a collection, visit_fn will be called with the element. The return value of visit_fn can be used to alter the element if return_data is set.

    Note that when using return_data a copy of each collection is created to avoid mutating the original object. This may have significant performance penalties and should only be used if you intend to transform the collection.

    Supported types: - List - Tuple - Set - Dict (note: keys are also visited recursively) - Dataclass - Pydantic model - Prefect annotations

    Parameters:

    Name Type Description Default expr Any

    a Python object or expression

    required visit_fn Callable[[Any], Awaitable[Any]]

    an async function that will be applied to every non-collection element of expr.

    required return_data bool

    if True, a copy of expr containing data modified by visit_fn will be returned. This is slower than return_data=False (the default).

    False max_depth int

    Controls the depth of recursive visitation. If set to zero, no recursion will occur. If set to a positive integer N, visitation will only descend to N layers deep. If set to any negative integer, no limit will be enforced and recursion will continue until terminal items are reached. By default, recursion is unlimited.

    -1 context Optional[dict]

    An optional dictionary. If passed, the context will be sent to each call to the visit_fn. The context can be mutated by each visitor and will be available for later visits to expressions at the given depth. Values will not be available \"up\" a level from a given expression.

    The context will be automatically populated with an 'annotation' key when visiting collections within a BaseAnnotation type. This requires the caller to pass context={} and will not be activated by default.

    None remove_annotations bool

    If set, annotations will be replaced by their contents. By default, annotations are preserved but their contents are visited.

    False Source code in prefect/utilities/collections.py
    def visit_collection(\n    expr,\n    visit_fn: Callable[[Any], Any],\n    return_data: bool = False,\n    max_depth: int = -1,\n    context: Optional[dict] = None,\n    remove_annotations: bool = False,\n):\n    \"\"\"\n    This function visits every element of an arbitrary Python collection. If an element\n    is a Python collection, it will be visited recursively. If an element is not a\n    collection, `visit_fn` will be called with the element. The return value of\n    `visit_fn` can be used to alter the element if `return_data` is set.\n\n    Note that when using `return_data` a copy of each collection is created to avoid\n    mutating the original object. This may have significant performance penalties and\n    should only be used if you intend to transform the collection.\n\n    Supported types:\n    - List\n    - Tuple\n    - Set\n    - Dict (note: keys are also visited recursively)\n    - Dataclass\n    - Pydantic model\n    - Prefect annotations\n\n    Args:\n        expr (Any): a Python object or expression\n        visit_fn (Callable[[Any], Awaitable[Any]]): an async function that\n            will be applied to every non-collection element of expr.\n        return_data (bool): if `True`, a copy of `expr` containing data modified\n            by `visit_fn` will be returned. This is slower than `return_data=False`\n            (the default).\n        max_depth: Controls the depth of recursive visitation. If set to zero, no\n            recursion will occur. If set to a positive integer N, visitation will only\n            descend to N layers deep. If set to any negative integer, no limit will be\n            enforced and recursion will continue until terminal items are reached. By\n            default, recursion is unlimited.\n        context: An optional dictionary. If passed, the context will be sent to each\n            call to the `visit_fn`. The context can be mutated by each visitor and will\n            be available for later visits to expressions at the given depth. Values\n            will not be available \"up\" a level from a given expression.\n\n            The context will be automatically populated with an 'annotation' key when\n            visiting collections within a `BaseAnnotation` type. This requires the\n            caller to pass `context={}` and will not be activated by default.\n        remove_annotations: If set, annotations will be replaced by their contents. By\n            default, annotations are preserved but their contents are visited.\n    \"\"\"\n\n    def visit_nested(expr):\n        # Utility for a recursive call, preserving options and updating the depth.\n        return visit_collection(\n            expr,\n            visit_fn=visit_fn,\n            return_data=return_data,\n            remove_annotations=remove_annotations,\n            max_depth=max_depth - 1,\n            # Copy the context on nested calls so it does not \"propagate up\"\n            context=context.copy() if context is not None else None,\n        )\n\n    def visit_expression(expr):\n        if context is not None:\n            return visit_fn(expr, context)\n        else:\n            return visit_fn(expr)\n\n    # Visit every expression\n    try:\n        result = visit_expression(expr)\n    except StopVisiting:\n        max_depth = 0\n        result = expr\n\n    if return_data:\n        # Only mutate the expression while returning data, otherwise it could be null\n        expr = result\n\n    # Then, visit every child of the expression recursively\n\n    # If we have reached the maximum depth, do not perform any recursion\n    if max_depth == 0:\n        return result if return_data else None\n\n    # Get the expression type; treat iterators like lists\n    typ = list if isinstance(expr, IteratorABC) and isiterable(expr) else type(expr)\n    typ = cast(type, typ)  # mypy treats this as 'object' otherwise and complains\n\n    # Then visit every item in the expression if it is a collection\n    if isinstance(expr, Mock):\n        # Do not attempt to recurse into mock objects\n        result = expr\n\n    elif isinstance(expr, BaseAnnotation):\n        if context is not None:\n            context[\"annotation\"] = expr\n        value = visit_nested(expr.unwrap())\n\n        if remove_annotations:\n            result = value if return_data else None\n        else:\n            result = expr.rewrap(value) if return_data else None\n\n    elif typ in (list, tuple, set):\n        items = [visit_nested(o) for o in expr]\n        result = typ(items) if return_data else None\n\n    elif typ in (dict, OrderedDict):\n        assert isinstance(expr, (dict, OrderedDict))  # typecheck assertion\n        items = [(visit_nested(k), visit_nested(v)) for k, v in expr.items()]\n        result = typ(items) if return_data else None\n\n    elif is_dataclass(expr) and not isinstance(expr, type):\n        values = [visit_nested(getattr(expr, f.name)) for f in fields(expr)]\n        items = {field.name: value for field, value in zip(fields(expr), values)}\n        result = typ(**items) if return_data else None\n\n    elif isinstance(expr, pydantic.BaseModel):\n        # NOTE: This implementation *does not* traverse private attributes\n        # Pydantic does not expose extras in `__fields__` so we use `__fields_set__`\n        # as well to get all of the relevant attributes\n        # Check for presence of attrs even if they're in the field set due to pydantic#4916\n        model_fields = {\n            f for f in expr.__fields_set__.union(expr.__fields__) if hasattr(expr, f)\n        }\n        items = [visit_nested(getattr(expr, key)) for key in model_fields]\n\n        if return_data:\n            # Collect fields with aliases so reconstruction can use the correct field name\n            aliases = {\n                key: value.alias\n                for key, value in expr.__fields__.items()\n                if value.has_alias\n            }\n\n            model_instance = typ(\n                **{\n                    aliases.get(key) or key: value\n                    for key, value in zip(model_fields, items)\n                }\n            )\n\n            # Private attributes are not included in `__fields_set__` but we do not want\n            # to drop them from the model so we restore them after constructing a new\n            # model\n            for attr in expr.__private_attributes__:\n                # Use `object.__setattr__` to avoid errors on immutable models\n                object.__setattr__(model_instance, attr, getattr(expr, attr))\n\n            # Preserve data about which fields were explicitly set on the original model\n            object.__setattr__(model_instance, \"__fields_set__\", expr.__fields_set__)\n            result = model_instance\n        else:\n            result = None\n\n    else:\n        result = result if return_data else None\n\n    return result\n
    ","tags":["Python API","collections"]},{"location":"api-ref/prefect/utilities/compat/","title":"compat","text":"","tags":["Python API","utilities","compatibility"]},{"location":"api-ref/prefect/utilities/compat/#prefect.utilities.compat","title":"prefect.utilities.compat","text":"

    Utilities for Python version compatibility

    ","tags":["Python API","utilities","compatibility"]},{"location":"api-ref/prefect/utilities/context/","title":"context","text":"","tags":["Python API","utilities","context"]},{"location":"api-ref/prefect/utilities/context/#prefect.utilities.context","title":"prefect.utilities.context","text":"","tags":["Python API","utilities","context"]},{"location":"api-ref/prefect/utilities/dispatch/","title":"dispatch","text":"","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch","title":"prefect.utilities.dispatch","text":"

    Provides methods for performing dynamic dispatch for actions on base type to one of its subtypes.

    Example:

    @register_base_type\nclass Base:\n    @classmethod\n    def __dispatch_key__(cls):\n        return cls.__name__.lower()\n\n\nclass Foo(Base):\n    ...\n\nkey = get_dispatch_key(Foo)  # 'foo'\nlookup_type(Base, key) # Foo\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.get_dispatch_key","title":"get_dispatch_key","text":"

    Retrieve the unique dispatch key for a class type or instance.

    This key is defined at the __dispatch_key__ attribute. If it is a callable, it will be resolved.

    If allow_missing is False, an exception will be raised if the attribute is not defined or the key is null. If True, None will be returned in these cases.

    Source code in prefect/utilities/dispatch.py
    def get_dispatch_key(\n    cls_or_instance: Any, allow_missing: bool = False\n) -> Optional[str]:\n    \"\"\"\n    Retrieve the unique dispatch key for a class type or instance.\n\n    This key is defined at the `__dispatch_key__` attribute. If it is a callable, it\n    will be resolved.\n\n    If `allow_missing` is `False`, an exception will be raised if the attribute is not\n    defined or the key is null. If `True`, `None` will be returned in these cases.\n    \"\"\"\n    dispatch_key = getattr(cls_or_instance, \"__dispatch_key__\", None)\n\n    type_name = (\n        cls_or_instance.__name__\n        if isinstance(cls_or_instance, type)\n        else type(cls_or_instance).__name__\n    )\n\n    if dispatch_key is None:\n        if allow_missing:\n            return None\n        raise ValueError(\n            f\"Type {type_name!r} does not define a value for \"\n            \"'__dispatch_key__' which is required for registry lookup.\"\n        )\n\n    if callable(dispatch_key):\n        dispatch_key = dispatch_key()\n\n    if allow_missing and dispatch_key is None:\n        return None\n\n    if not isinstance(dispatch_key, str):\n        raise TypeError(\n            f\"Type {type_name!r} has a '__dispatch_key__' of type \"\n            f\"{type(dispatch_key).__name__} but a type of 'str' is required.\"\n        )\n\n    return dispatch_key\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.get_registry_for_type","title":"get_registry_for_type","text":"

    Get the first matching registry for a class or any of its base classes.

    If not found, None is returned.

    Source code in prefect/utilities/dispatch.py
    def get_registry_for_type(cls: T) -> Optional[Dict[str, T]]:\n    \"\"\"\n    Get the first matching registry for a class or any of its base classes.\n\n    If not found, `None` is returned.\n    \"\"\"\n    return next(\n        filter(\n            lambda registry: registry is not None,\n            (_TYPE_REGISTRIES.get(cls) for cls in cls.mro()),\n        ),\n        None,\n    )\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.lookup_type","title":"lookup_type","text":"

    Look up a dispatch key in the type registry for the given class.

    Source code in prefect/utilities/dispatch.py
    def lookup_type(cls: T, dispatch_key: str) -> T:\n    \"\"\"\n    Look up a dispatch key in the type registry for the given class.\n    \"\"\"\n    # Get the first matching registry for the class or one of its bases\n    registry = get_registry_for_type(cls)\n\n    # Look up this type in the registry\n    subcls = registry.get(dispatch_key)\n\n    if subcls is None:\n        raise KeyError(\n            f\"No class found for dispatch key {dispatch_key!r} in registry for type \"\n            f\"{cls.__name__!r}.\"\n        )\n\n    return subcls\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.register_base_type","title":"register_base_type","text":"

    Register a base type allowing child types to be registered for dispatch with register_type.

    The base class may or may not define a __dispatch_key__ to allow lookups of the base type.

    Source code in prefect/utilities/dispatch.py
    def register_base_type(cls: T) -> T:\n    \"\"\"\n    Register a base type allowing child types to be registered for dispatch with\n    `register_type`.\n\n    The base class may or may not define a `__dispatch_key__` to allow lookups of the\n    base type.\n    \"\"\"\n    registry = _TYPE_REGISTRIES.setdefault(cls, {})\n    base_key = get_dispatch_key(cls, allow_missing=True)\n    if base_key is not None:\n        registry[base_key] = cls\n\n    # Add automatic subtype registration\n    cls.__init_subclass_original__ = getattr(cls, \"__init_subclass__\")\n    cls.__init_subclass__ = _register_subclass_of_base_type\n\n    return cls\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dispatch/#prefect.utilities.dispatch.register_type","title":"register_type","text":"

    Register a type for lookup with dispatch.

    The type or one of its parents must define a unique __dispatch_key__.

    One of the classes base types must be registered using register_base_type.

    Source code in prefect/utilities/dispatch.py
    def register_type(cls: T) -> T:\n    \"\"\"\n    Register a type for lookup with dispatch.\n\n    The type or one of its parents must define a unique `__dispatch_key__`.\n\n    One of the classes base types must be registered using `register_base_type`.\n    \"\"\"\n    # Lookup the registry for this type\n    registry = get_registry_for_type(cls)\n\n    # Check if a base type is registered\n    if registry is None:\n        # Include a description of registered base types\n        known = \", \".join(repr(base.__name__) for base in _TYPE_REGISTRIES)\n        known_message = (\n            f\" Did you mean to inherit from one of the following known types: {known}.\"\n            if known\n            else \"\"\n        )\n\n        # And a list of all base types for the type they tried to register\n        bases = \", \".join(\n            repr(base.__name__) for base in cls.mro() if base not in (object, cls)\n        )\n\n        raise ValueError(\n            f\"No registry found for type {cls.__name__!r} with bases {bases}.\"\n            + known_message\n        )\n\n    key = get_dispatch_key(cls)\n    existing_value = registry.get(key)\n    if existing_value is not None and id(existing_value) != id(cls):\n        # Get line numbers for debugging\n        file = inspect.getsourcefile(cls)\n        line_number = inspect.getsourcelines(cls)[1]\n        existing_file = inspect.getsourcefile(existing_value)\n        existing_line_number = inspect.getsourcelines(existing_value)[1]\n        warnings.warn(\n            f\"Type {cls.__name__!r} at {file}:{line_number} has key {key!r} that \"\n            f\"matches existing registered type {existing_value.__name__!r} from \"\n            f\"{existing_file}:{existing_line_number}. The existing type will be \"\n            \"overridden.\"\n        )\n\n    # Add to the registry\n    registry[key] = cls\n\n    return cls\n
    ","tags":["Python API","dispatch"]},{"location":"api-ref/prefect/utilities/dockerutils/","title":"dockerutils","text":"","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils","title":"prefect.utilities.dockerutils","text":"","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.BuildError","title":"BuildError","text":"

    Bases: Exception

    Raised when a Docker build fails

    Source code in prefect/utilities/dockerutils.py
    class BuildError(Exception):\n    \"\"\"Raised when a Docker build fails\"\"\"\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder","title":"ImageBuilder","text":"

    An interface for preparing Docker build contexts and building images

    Source code in prefect/utilities/dockerutils.py
    class ImageBuilder:\n    \"\"\"An interface for preparing Docker build contexts and building images\"\"\"\n\n    base_directory: Path\n    context: Optional[Path]\n    platform: Optional[str]\n    dockerfile_lines: List[str]\n\n    def __init__(\n        self,\n        base_image: str,\n        base_directory: Path = None,\n        platform: str = None,\n        context: Path = None,\n    ):\n        \"\"\"Create an ImageBuilder\n\n        Args:\n            base_image: the base image to use\n            base_directory: the starting point on your host for relative file locations,\n                defaulting to the current directory\n            context: use this path as the build context (if not provided, will create a\n                temporary directory for the context)\n\n        Returns:\n            The image ID\n        \"\"\"\n        self.base_directory = base_directory or context or Path().absolute()\n        self.temporary_directory = None\n        self.context = context\n        self.platform = platform\n        self.dockerfile_lines = []\n\n        if self.context:\n            dockerfile_path: Path = self.context / \"Dockerfile\"\n            if dockerfile_path.exists():\n                raise ValueError(f\"There is already a Dockerfile at {context}\")\n\n        self.add_line(f\"FROM {base_image}\")\n\n    def __enter__(self) -> Self:\n        if self.context and not self.temporary_directory:\n            return self\n\n        self.temporary_directory = TemporaryDirectory()\n        self.context = Path(self.temporary_directory.__enter__())\n        return self\n\n    def __exit__(\n        self, exc: Type[BaseException], value: BaseException, traceback: TracebackType\n    ) -> None:\n        if not self.temporary_directory:\n            return\n\n        self.temporary_directory.__exit__(exc, value, traceback)\n        self.temporary_directory = None\n        self.context = None\n\n    def add_line(self, line: str) -> None:\n        \"\"\"Add a line to this image's Dockerfile\"\"\"\n        self.add_lines([line])\n\n    def add_lines(self, lines: Iterable[str]) -> None:\n        \"\"\"Add lines to this image's Dockerfile\"\"\"\n        self.dockerfile_lines.extend(lines)\n\n    def copy(self, source: Union[str, Path], destination: Union[str, PurePosixPath]):\n        \"\"\"Copy a file to this image\"\"\"\n        if not self.context:\n            raise Exception(\"No context available\")\n\n        if not isinstance(destination, PurePosixPath):\n            destination = PurePosixPath(destination)\n\n        if not isinstance(source, Path):\n            source = Path(source)\n\n        if source.is_absolute():\n            source = source.resolve().relative_to(self.base_directory)\n\n        if self.temporary_directory:\n            os.makedirs(self.context / source.parent, exist_ok=True)\n\n            if source.is_dir():\n                shutil.copytree(self.base_directory / source, self.context / source)\n            else:\n                shutil.copy2(self.base_directory / source, self.context / source)\n\n        self.add_line(f\"COPY {source} {destination}\")\n\n    def write_text(self, text: str, destination: Union[str, PurePosixPath]):\n        if not self.context:\n            raise Exception(\"No context available\")\n\n        if not isinstance(destination, PurePosixPath):\n            destination = PurePosixPath(destination)\n\n        source_hash = hashlib.sha256(text.encode()).hexdigest()\n        (self.context / f\".{source_hash}\").write_text(text)\n        self.add_line(f\"COPY .{source_hash} {destination}\")\n\n    def build(\n        self, pull: bool = False, stream_progress_to: Optional[TextIO] = None\n    ) -> str:\n        \"\"\"Build the Docker image from the current state of the ImageBuilder\n\n        Args:\n            pull: True to pull the base image during the build\n            stream_progress_to: an optional stream (like sys.stdout, or an io.TextIO)\n                that will collect the build output as it is reported by Docker\n\n        Returns:\n            The image ID\n        \"\"\"\n        dockerfile_path: Path = self.context / \"Dockerfile\"\n\n        with dockerfile_path.open(\"w\") as dockerfile:\n            dockerfile.writelines(line + \"\\n\" for line in self.dockerfile_lines)\n\n        try:\n            return build_image(\n                self.context,\n                platform=self.platform,\n                pull=pull,\n                stream_progress_to=stream_progress_to,\n            )\n        finally:\n            os.unlink(dockerfile_path)\n\n    def assert_has_line(self, line: str) -> None:\n        \"\"\"Asserts that the given line is in the Dockerfile\"\"\"\n        all_lines = \"\\n\".join(\n            [f\"  {i+1:>3}: {line}\" for i, line in enumerate(self.dockerfile_lines)]\n        )\n        message = (\n            f\"Expected {line!r} not found in Dockerfile.  Dockerfile:\\n{all_lines}\"\n        )\n        assert line in self.dockerfile_lines, message\n\n    def assert_line_absent(self, line: str) -> None:\n        \"\"\"Asserts that the given line is absent from the Dockerfile\"\"\"\n        if line not in self.dockerfile_lines:\n            return\n\n        i = self.dockerfile_lines.index(line)\n\n        surrounding_lines = \"\\n\".join(\n            [\n                f\"  {i+1:>3}: {line}\"\n                for i, line in enumerate(self.dockerfile_lines[i - 2 : i + 2])\n            ]\n        )\n        message = (\n            f\"Unexpected {line!r} found in Dockerfile at line {i+1}.  \"\n            f\"Surrounding lines:\\n{surrounding_lines}\"\n        )\n\n        assert line not in self.dockerfile_lines, message\n\n    def assert_line_before(self, first: str, second: str) -> None:\n        \"\"\"Asserts that the first line appears before the second line\"\"\"\n        self.assert_has_line(first)\n        self.assert_has_line(second)\n\n        first_index = self.dockerfile_lines.index(first)\n        second_index = self.dockerfile_lines.index(second)\n\n        surrounding_lines = \"\\n\".join(\n            [\n                f\"  {i+1:>3}: {line}\"\n                for i, line in enumerate(\n                    self.dockerfile_lines[second_index - 2 : first_index + 2]\n                )\n            ]\n        )\n\n        message = (\n            f\"Expected {first!r} to appear before {second!r} in the Dockerfile, but \"\n            f\"{first!r} was at line {first_index+1} and {second!r} as at line \"\n            f\"{second_index+1}.  Surrounding lines:\\n{surrounding_lines}\"\n        )\n\n        assert first_index < second_index, message\n\n    def assert_line_after(self, second: str, first: str) -> None:\n        \"\"\"Asserts that the second line appears after the first line\"\"\"\n        self.assert_line_before(first, second)\n\n    def assert_has_file(self, source: Path, container_path: PurePosixPath) -> None:\n        \"\"\"Asserts that the given file or directory will be copied into the container\n        at the given path\"\"\"\n        if source.is_absolute():\n            source = source.relative_to(self.base_directory)\n\n        self.assert_has_line(f\"COPY {source} {container_path}\")\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.add_line","title":"add_line","text":"

    Add a line to this image's Dockerfile

    Source code in prefect/utilities/dockerutils.py
    def add_line(self, line: str) -> None:\n    \"\"\"Add a line to this image's Dockerfile\"\"\"\n    self.add_lines([line])\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.add_lines","title":"add_lines","text":"

    Add lines to this image's Dockerfile

    Source code in prefect/utilities/dockerutils.py
    def add_lines(self, lines: Iterable[str]) -> None:\n    \"\"\"Add lines to this image's Dockerfile\"\"\"\n    self.dockerfile_lines.extend(lines)\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_has_file","title":"assert_has_file","text":"

    Asserts that the given file or directory will be copied into the container at the given path

    Source code in prefect/utilities/dockerutils.py
    def assert_has_file(self, source: Path, container_path: PurePosixPath) -> None:\n    \"\"\"Asserts that the given file or directory will be copied into the container\n    at the given path\"\"\"\n    if source.is_absolute():\n        source = source.relative_to(self.base_directory)\n\n    self.assert_has_line(f\"COPY {source} {container_path}\")\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_has_line","title":"assert_has_line","text":"

    Asserts that the given line is in the Dockerfile

    Source code in prefect/utilities/dockerutils.py
    def assert_has_line(self, line: str) -> None:\n    \"\"\"Asserts that the given line is in the Dockerfile\"\"\"\n    all_lines = \"\\n\".join(\n        [f\"  {i+1:>3}: {line}\" for i, line in enumerate(self.dockerfile_lines)]\n    )\n    message = (\n        f\"Expected {line!r} not found in Dockerfile.  Dockerfile:\\n{all_lines}\"\n    )\n    assert line in self.dockerfile_lines, message\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_line_absent","title":"assert_line_absent","text":"

    Asserts that the given line is absent from the Dockerfile

    Source code in prefect/utilities/dockerutils.py
    def assert_line_absent(self, line: str) -> None:\n    \"\"\"Asserts that the given line is absent from the Dockerfile\"\"\"\n    if line not in self.dockerfile_lines:\n        return\n\n    i = self.dockerfile_lines.index(line)\n\n    surrounding_lines = \"\\n\".join(\n        [\n            f\"  {i+1:>3}: {line}\"\n            for i, line in enumerate(self.dockerfile_lines[i - 2 : i + 2])\n        ]\n    )\n    message = (\n        f\"Unexpected {line!r} found in Dockerfile at line {i+1}.  \"\n        f\"Surrounding lines:\\n{surrounding_lines}\"\n    )\n\n    assert line not in self.dockerfile_lines, message\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_line_after","title":"assert_line_after","text":"

    Asserts that the second line appears after the first line

    Source code in prefect/utilities/dockerutils.py
    def assert_line_after(self, second: str, first: str) -> None:\n    \"\"\"Asserts that the second line appears after the first line\"\"\"\n    self.assert_line_before(first, second)\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.assert_line_before","title":"assert_line_before","text":"

    Asserts that the first line appears before the second line

    Source code in prefect/utilities/dockerutils.py
    def assert_line_before(self, first: str, second: str) -> None:\n    \"\"\"Asserts that the first line appears before the second line\"\"\"\n    self.assert_has_line(first)\n    self.assert_has_line(second)\n\n    first_index = self.dockerfile_lines.index(first)\n    second_index = self.dockerfile_lines.index(second)\n\n    surrounding_lines = \"\\n\".join(\n        [\n            f\"  {i+1:>3}: {line}\"\n            for i, line in enumerate(\n                self.dockerfile_lines[second_index - 2 : first_index + 2]\n            )\n        ]\n    )\n\n    message = (\n        f\"Expected {first!r} to appear before {second!r} in the Dockerfile, but \"\n        f\"{first!r} was at line {first_index+1} and {second!r} as at line \"\n        f\"{second_index+1}.  Surrounding lines:\\n{surrounding_lines}\"\n    )\n\n    assert first_index < second_index, message\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.build","title":"build","text":"

    Build the Docker image from the current state of the ImageBuilder

    Parameters:

    Name Type Description Default pull bool

    True to pull the base image during the build

    False stream_progress_to Optional[TextIO]

    an optional stream (like sys.stdout, or an io.TextIO) that will collect the build output as it is reported by Docker

    None

    Returns:

    Type Description str

    The image ID

    Source code in prefect/utilities/dockerutils.py
    def build(\n    self, pull: bool = False, stream_progress_to: Optional[TextIO] = None\n) -> str:\n    \"\"\"Build the Docker image from the current state of the ImageBuilder\n\n    Args:\n        pull: True to pull the base image during the build\n        stream_progress_to: an optional stream (like sys.stdout, or an io.TextIO)\n            that will collect the build output as it is reported by Docker\n\n    Returns:\n        The image ID\n    \"\"\"\n    dockerfile_path: Path = self.context / \"Dockerfile\"\n\n    with dockerfile_path.open(\"w\") as dockerfile:\n        dockerfile.writelines(line + \"\\n\" for line in self.dockerfile_lines)\n\n    try:\n        return build_image(\n            self.context,\n            platform=self.platform,\n            pull=pull,\n            stream_progress_to=stream_progress_to,\n        )\n    finally:\n        os.unlink(dockerfile_path)\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.ImageBuilder.copy","title":"copy","text":"

    Copy a file to this image

    Source code in prefect/utilities/dockerutils.py
    def copy(self, source: Union[str, Path], destination: Union[str, PurePosixPath]):\n    \"\"\"Copy a file to this image\"\"\"\n    if not self.context:\n        raise Exception(\"No context available\")\n\n    if not isinstance(destination, PurePosixPath):\n        destination = PurePosixPath(destination)\n\n    if not isinstance(source, Path):\n        source = Path(source)\n\n    if source.is_absolute():\n        source = source.resolve().relative_to(self.base_directory)\n\n    if self.temporary_directory:\n        os.makedirs(self.context / source.parent, exist_ok=True)\n\n        if source.is_dir():\n            shutil.copytree(self.base_directory / source, self.context / source)\n        else:\n            shutil.copy2(self.base_directory / source, self.context / source)\n\n    self.add_line(f\"COPY {source} {destination}\")\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.PushError","title":"PushError","text":"

    Bases: Exception

    Raised when a Docker image push fails

    Source code in prefect/utilities/dockerutils.py
    class PushError(Exception):\n    \"\"\"Raised when a Docker image push fails\"\"\"\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.build_image","title":"build_image","text":"

    Builds a Docker image, returning the image ID

    Parameters:

    Name Type Description Default context Path

    the root directory for the Docker build context

    required dockerfile str

    the path to the Dockerfile, relative to the context

    'Dockerfile' tag Optional[str]

    the tag to give this image

    None pull bool

    True to pull the base image during the build

    False stream_progress_to Optional[TextIO]

    an optional stream (like sys.stdout, or an io.TextIO) that will collect the build output as it is reported by Docker

    None

    Returns:

    Type Description str

    The image ID

    Source code in prefect/utilities/dockerutils.py
    @silence_docker_warnings()\ndef build_image(\n    context: Path,\n    dockerfile: str = \"Dockerfile\",\n    tag: Optional[str] = None,\n    pull: bool = False,\n    platform: str = None,\n    stream_progress_to: Optional[TextIO] = None,\n    **kwargs,\n) -> str:\n    \"\"\"Builds a Docker image, returning the image ID\n\n    Args:\n        context: the root directory for the Docker build context\n        dockerfile: the path to the Dockerfile, relative to the context\n        tag: the tag to give this image\n        pull: True to pull the base image during the build\n        stream_progress_to: an optional stream (like sys.stdout, or an io.TextIO) that\n            will collect the build output as it is reported by Docker\n\n    Returns:\n        The image ID\n    \"\"\"\n\n    if not context:\n        raise ValueError(\"context required to build an image\")\n\n    if not Path(context).exists():\n        raise ValueError(f\"Context path {context} does not exist\")\n\n    kwargs = {key: kwargs[key] for key in kwargs if key not in [\"decode\", \"labels\"]}\n\n    image_id = None\n    with docker_client() as client:\n        events = client.api.build(\n            path=str(context),\n            tag=tag,\n            dockerfile=dockerfile,\n            pull=pull,\n            decode=True,\n            labels=IMAGE_LABELS,\n            platform=platform,\n            **kwargs,\n        )\n\n        try:\n            for event in events:\n                if \"stream\" in event:\n                    if not stream_progress_to:\n                        continue\n                    stream_progress_to.write(event[\"stream\"])\n                    stream_progress_to.flush()\n                elif \"aux\" in event:\n                    image_id = event[\"aux\"][\"ID\"]\n                elif \"error\" in event:\n                    raise BuildError(event[\"error\"])\n                elif \"message\" in event:\n                    raise BuildError(event[\"message\"])\n        except docker.errors.APIError as e:\n            raise BuildError(e.explanation) from e\n\n    assert image_id, \"The Docker daemon did not return an image ID\"\n    return image_id\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.docker_client","title":"docker_client","text":"

    Get the environmentally-configured Docker client

    Source code in prefect/utilities/dockerutils.py
    @contextmanager\ndef docker_client() -> Generator[\"DockerClient\", None, None]:\n    \"\"\"Get the environmentally-configured Docker client\"\"\"\n    with silence_docker_warnings():\n        client = docker.DockerClient.from_env()\n\n    try:\n        yield client\n    finally:\n        client.close()\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.format_outlier_version_name","title":"format_outlier_version_name","text":"

    Formats outlier docker version names to pass packaging.version.parse validation - Current cases are simple, but creates stub for more complicated formatting if eventually needed. - Example outlier versions that throw a parsing exception: - \"20.10.0-ce\" (variant of community edition label) - \"20.10.0-ee\" (variant of enterprise edition label)

    Parameters:

    Name Type Description Default version str

    raw docker version value

    required

    Returns:

    Name Type Description str

    value that can pass packaging.version.parse validation

    Source code in prefect/utilities/dockerutils.py
    def format_outlier_version_name(version: str):\n    \"\"\"\n    Formats outlier docker version names to pass `packaging.version.parse` validation\n    - Current cases are simple, but creates stub for more complicated formatting if eventually needed.\n    - Example outlier versions that throw a parsing exception:\n      - \"20.10.0-ce\" (variant of community edition label)\n      - \"20.10.0-ee\" (variant of enterprise edition label)\n\n    Args:\n        version (str): raw docker version value\n\n    Returns:\n        str: value that can pass `packaging.version.parse` validation\n    \"\"\"\n    return version.replace(\"-ce\", \"\").replace(\"-ee\", \"\")\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.generate_default_dockerfile","title":"generate_default_dockerfile","text":"

    Generates a default Dockerfile used for deploying flows. The Dockerfile is written to a temporary file and yielded. The temporary file is removed after the context manager exits.

    Parameters:

    Name Type Description Default - context

    The context to use for the Dockerfile. Defaults to the current working directory.

    required Source code in prefect/utilities/dockerutils.py
    @contextmanager\ndef generate_default_dockerfile(context: Optional[Path] = None):\n    \"\"\"\n    Generates a default Dockerfile used for deploying flows. The Dockerfile is written\n    to a temporary file and yielded. The temporary file is removed after the context\n    manager exits.\n\n    Args:\n        - context: The context to use for the Dockerfile. Defaults to\n            the current working directory.\n    \"\"\"\n    if not context:\n        context = Path.cwd()\n    lines = []\n    base_image = get_prefect_image_name()\n    lines.append(f\"FROM {base_image}\")\n    dir_name = context.name\n\n    if (context / \"requirements.txt\").exists():\n        lines.append(f\"COPY requirements.txt /opt/prefect/{dir_name}/requirements.txt\")\n        lines.append(\n            f\"RUN python -m pip install -r /opt/prefect/{dir_name}/requirements.txt\"\n        )\n\n    lines.append(f\"COPY . /opt/prefect/{dir_name}/\")\n    lines.append(f\"WORKDIR /opt/prefect/{dir_name}/\")\n\n    temp_dockerfile = context / \"Dockerfile\"\n    if Path(temp_dockerfile).exists():\n        raise RuntimeError(\n            \"Failed to generate Dockerfile. Dockerfile already exists in the\"\n            \" current directory.\"\n        )\n\n    with Path(temp_dockerfile).open(\"w\") as f:\n        f.writelines(line + \"\\n\" for line in lines)\n\n    try:\n        yield temp_dockerfile\n    finally:\n        temp_dockerfile.unlink()\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.get_prefect_image_name","title":"get_prefect_image_name","text":"

    Get the Prefect image name matching the current Prefect and Python versions.

    Parameters:

    Name Type Description Default prefect_version str

    An optional override for the Prefect version.

    None python_version str

    An optional override for the Python version; must be at the minor level e.g. '3.9'.

    None flavor str

    An optional alternative image flavor to build, like 'conda'

    None Source code in prefect/utilities/dockerutils.py
    def get_prefect_image_name(\n    prefect_version: str = None, python_version: str = None, flavor: str = None\n) -> str:\n    \"\"\"\n    Get the Prefect image name matching the current Prefect and Python versions.\n\n    Args:\n        prefect_version: An optional override for the Prefect version.\n        python_version: An optional override for the Python version; must be at the\n            minor level e.g. '3.9'.\n        flavor: An optional alternative image flavor to build, like 'conda'\n    \"\"\"\n    parsed_version = (prefect_version or prefect.__version__).split(\"+\")\n    is_prod_build = len(parsed_version) == 1\n    prefect_version = (\n        parsed_version[0]\n        if is_prod_build\n        else \"sha-\" + prefect.__version_info__[\"full-revisionid\"][:7]\n    )\n\n    python_version = python_version or python_version_minor()\n\n    tag = slugify(\n        f\"{prefect_version}-python{python_version}\" + (f\"-{flavor}\" if flavor else \"\"),\n        lowercase=False,\n        max_length=128,\n        # Docker allows these characters for tag names\n        regex_pattern=r\"[^a-zA-Z0-9_.-]+\",\n    )\n\n    image = \"prefect\" if is_prod_build else \"prefect-dev\"\n    return f\"prefecthq/{image}:{tag}\"\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.parse_image_tag","title":"parse_image_tag","text":"

    Parse Docker Image String

    • If a tag exists, this function parses and returns the image registry and tag, separately as a tuple.
    • Example 1: 'prefecthq/prefect:latest' -> ('prefecthq/prefect', 'latest')
    • Example 2: 'hostname.io:5050/folder/subfolder:latest' -> ('hostname.io:5050/folder/subfolder', 'latest')
    • Supports parsing Docker Image strings that follow Docker Image Specification v1.1.0
    • Image building tools typically enforce this standard

    Parameters:

    Name Type Description Default name str

    Name of Docker Image

    required Return Source code in prefect/utilities/dockerutils.py
    def parse_image_tag(name: str) -> Tuple[str, Optional[str]]:\n    \"\"\"\n    Parse Docker Image String\n\n    - If a tag exists, this function parses and returns the image registry and tag,\n      separately as a tuple.\n      - Example 1: 'prefecthq/prefect:latest' -> ('prefecthq/prefect', 'latest')\n      - Example 2: 'hostname.io:5050/folder/subfolder:latest' -> ('hostname.io:5050/folder/subfolder', 'latest')\n    - Supports parsing Docker Image strings that follow Docker Image Specification v1.1.0\n      - Image building tools typically enforce this standard\n\n    Args:\n        name (str): Name of Docker Image\n\n    Return:\n        tuple: image registry, image tag\n    \"\"\"\n    tag = None\n    name_parts = name.split(\"/\")\n    # First handles the simplest image names (DockerHub-based, index-free, potentionally with a tag)\n    # - Example: simplename:latest\n    if len(name_parts) == 1:\n        if \":\" in name_parts[0]:\n            image_name, tag = name_parts[0].split(\":\")\n        else:\n            image_name = name_parts[0]\n    else:\n        # 1. Separates index (hostname.io or prefecthq) from path:tag (folder/subfolder:latest or prefect:latest)\n        # 2. Separates path and tag (if tag exists)\n        # 3. Reunites index and path (without tag) as image name\n        index_name = name_parts[0]\n        image_path = \"/\".join(name_parts[1:])\n        if \":\" in image_path:\n            image_path, tag = image_path.split(\":\")\n        image_name = f\"{index_name}/{image_path}\"\n    return image_name, tag\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.push_image","title":"push_image","text":"

    Pushes a local image to a Docker registry, returning the registry-qualified tag for that image

    This assumes that the environment's Docker daemon is already authenticated to the given registry, and currently makes no attempt to authenticate.

    Parameters:

    Name Type Description Default image_id str

    a Docker image ID

    required registry_url str

    the URL of a Docker registry

    required name str

    the name of this image

    required tag str

    the tag to give this image (defaults to a short representation of the image's ID)

    None stream_progress_to Optional[TextIO]

    an optional stream (like sys.stdout, or an io.TextIO) that will collect the build output as it is reported by Docker

    None

    Returns:

    Type Description str

    A registry-qualified tag, like my-registry.example.com/my-image:abcdefg

    Source code in prefect/utilities/dockerutils.py
    @silence_docker_warnings()\ndef push_image(\n    image_id: str,\n    registry_url: str,\n    name: str,\n    tag: Optional[str] = None,\n    stream_progress_to: Optional[TextIO] = None,\n) -> str:\n    \"\"\"Pushes a local image to a Docker registry, returning the registry-qualified tag\n    for that image\n\n    This assumes that the environment's Docker daemon is already authenticated to the\n    given registry, and currently makes no attempt to authenticate.\n\n    Args:\n        image_id (str): a Docker image ID\n        registry_url (str): the URL of a Docker registry\n        name (str): the name of this image\n        tag (str): the tag to give this image (defaults to a short representation of\n            the image's ID)\n        stream_progress_to: an optional stream (like sys.stdout, or an io.TextIO) that\n            will collect the build output as it is reported by Docker\n\n    Returns:\n        A registry-qualified tag, like my-registry.example.com/my-image:abcdefg\n    \"\"\"\n\n    if not tag:\n        tag = slugify(pendulum.now(\"utc\").isoformat())\n\n    _, registry, _, _, _ = urlsplit(registry_url)\n    repository = f\"{registry}/{name}\"\n\n    with docker_client() as client:\n        image: \"docker.Image\" = client.images.get(image_id)\n        image.tag(repository, tag=tag)\n        events = client.api.push(repository, tag=tag, stream=True, decode=True)\n        try:\n            for event in events:\n                if \"status\" in event:\n                    if not stream_progress_to:\n                        continue\n                    stream_progress_to.write(event[\"status\"])\n                    if \"progress\" in event:\n                        stream_progress_to.write(\" \" + event[\"progress\"])\n                    stream_progress_to.write(\"\\n\")\n                    stream_progress_to.flush()\n                elif \"error\" in event:\n                    raise PushError(event[\"error\"])\n        finally:\n            client.api.remove_image(f\"{repository}:{tag}\", noprune=True)\n\n    return f\"{repository}:{tag}\"\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.split_repository_path","title":"split_repository_path","text":"

    Splits a Docker repository path into its namespace and repository components.

    Parameters:

    Name Type Description Default repository_path str

    The Docker repository path to split.

    required

    Returns:

    Type Description Tuple[Optional[str], str]

    Tuple[Optional[str], str]: A tuple containing the namespace and repository components. - namespace (Optional[str]): The Docker namespace, combining the registry and organization. None if not present. - repository (Optionals[str]): The repository name.

    Source code in prefect/utilities/dockerutils.py
    def split_repository_path(repository_path: str) -> Tuple[Optional[str], str]:\n    \"\"\"\n    Splits a Docker repository path into its namespace and repository components.\n\n    Args:\n        repository_path: The Docker repository path to split.\n\n    Returns:\n        Tuple[Optional[str], str]: A tuple containing the namespace and repository components.\n            - namespace (Optional[str]): The Docker namespace, combining the registry and organization. None if not present.\n            - repository (Optionals[str]): The repository name.\n    \"\"\"\n    parts = repository_path.split(\"/\", 2)\n\n    # Check if the path includes a registry and organization or just organization/repository\n    if len(parts) == 3 or (len(parts) == 2 and (\".\" in parts[0] or \":\" in parts[0])):\n        # Namespace includes registry and organization\n        namespace = \"/\".join(parts[:-1])\n        repository = parts[-1]\n    elif len(parts) == 2:\n        # Only organization/repository provided, so namespace is just the first part\n        namespace = parts[0]\n        repository = parts[1]\n    else:\n        # No namespace provided\n        namespace = None\n        repository = parts[0]\n\n    return namespace, repository\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/dockerutils/#prefect.utilities.dockerutils.to_run_command","title":"to_run_command","text":"

    Convert a process-style list of command arguments to a single Dockerfile RUN instruction.

    Source code in prefect/utilities/dockerutils.py
    def to_run_command(command: List[str]) -> str:\n    \"\"\"\n    Convert a process-style list of command arguments to a single Dockerfile RUN\n    instruction.\n    \"\"\"\n    if not command:\n        return \"\"\n\n    run_command = f\"RUN {command[0]}\"\n    if len(command) > 1:\n        run_command += \" \" + \" \".join([repr(arg) for arg in command[1:]])\n\n    # TODO: Consider performing text-wrapping to improve readability of the generated\n    #       Dockerfile\n    # return textwrap.wrap(\n    #     run_command,\n    #     subsequent_indent=\" \" * 4,\n    #     break_on_hyphens=False,\n    #     break_long_words=False\n    # )\n\n    return run_command\n
    ","tags":["Python API","Docker"]},{"location":"api-ref/prefect/utilities/filesystem/","title":"filesystem","text":"","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem","title":"prefect.utilities.filesystem","text":"

    Utilities for working with file systems

    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.create_default_ignore_file","title":"create_default_ignore_file","text":"

    Creates default ignore file in the provided path if one does not already exist; returns boolean specifying whether a file was created.

    Source code in prefect/utilities/filesystem.py
    def create_default_ignore_file(path: str) -> bool:\n    \"\"\"\n    Creates default ignore file in the provided path if one does not already exist; returns boolean specifying\n    whether a file was created.\n    \"\"\"\n    path = pathlib.Path(path)\n    ignore_file = path / \".prefectignore\"\n    if ignore_file.exists():\n        return False\n    default_file = pathlib.Path(prefect.__module_path__) / \".prefectignore\"\n    with ignore_file.open(mode=\"w\") as f:\n        f.write(default_file.read_text())\n    return True\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.filename","title":"filename","text":"

    Extract the file name from a path with remote file system support

    Source code in prefect/utilities/filesystem.py
    def filename(path: str) -> str:\n    \"\"\"Extract the file name from a path with remote file system support\"\"\"\n    try:\n        of: OpenFile = fsspec.open(path)\n        sep = of.fs.sep\n    except (ImportError, AttributeError):\n        sep = \"\\\\\" if \"\\\\\" in path else \"/\"\n    return path.split(sep)[-1]\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.filter_files","title":"filter_files","text":"

    This function accepts a root directory path and a list of file patterns to ignore, and returns a list of files that excludes those that should be ignored.

    The specification matches that of .gitignore files.

    Source code in prefect/utilities/filesystem.py
    def filter_files(\n    root: str = \".\", ignore_patterns: list = None, include_dirs: bool = True\n) -> set:\n    \"\"\"\n    This function accepts a root directory path and a list of file patterns to ignore, and returns\n    a list of files that excludes those that should be ignored.\n\n    The specification matches that of [.gitignore files](https://git-scm.com/docs/gitignore).\n    \"\"\"\n    if ignore_patterns is None:\n        ignore_patterns = []\n    spec = pathspec.PathSpec.from_lines(\"gitwildmatch\", ignore_patterns)\n    ignored_files = {p.path for p in spec.match_tree_entries(root)}\n    if include_dirs:\n        all_files = {p.path for p in pathspec.util.iter_tree_entries(root)}\n    else:\n        all_files = set(pathspec.util.iter_tree_files(root))\n    included_files = all_files - ignored_files\n    return included_files\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.get_open_file_limit","title":"get_open_file_limit","text":"

    Get the maximum number of open files allowed for the current process

    Source code in prefect/utilities/filesystem.py
    def get_open_file_limit() -> int:\n    \"\"\"Get the maximum number of open files allowed for the current process\"\"\"\n\n    try:\n        if os.name == \"nt\":\n            import ctypes\n\n            return ctypes.cdll.ucrtbase._getmaxstdio()\n        else:\n            import resource\n\n            soft_limit, _ = resource.getrlimit(resource.RLIMIT_NOFILE)\n            return soft_limit\n    except Exception:\n        # Catch all exceptions, as ctypes can raise several errors\n        # depending on what went wrong. Return a safe default if we\n        # can't get the limit from the OS.\n        return 200\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.is_local_path","title":"is_local_path","text":"

    Check if the given path points to a local or remote file system

    Source code in prefect/utilities/filesystem.py
    def is_local_path(path: Union[str, pathlib.Path, OpenFile]):\n    \"\"\"Check if the given path points to a local or remote file system\"\"\"\n    if isinstance(path, str):\n        try:\n            of = fsspec.open(path)\n        except ImportError:\n            # The path is a remote file system that uses a lib that is not installed\n            return False\n    elif isinstance(path, pathlib.Path):\n        return True\n    elif isinstance(path, OpenFile):\n        of = path\n    else:\n        raise TypeError(f\"Invalid path of type {type(path).__name__!r}\")\n\n    return type(of.fs) == LocalFileSystem\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.relative_path_to_current_platform","title":"relative_path_to_current_platform","text":"

    Converts a relative path generated on any platform to a relative path for the current platform.

    Source code in prefect/utilities/filesystem.py
    def relative_path_to_current_platform(path_str: str) -> Path:\n    \"\"\"\n    Converts a relative path generated on any platform to a relative path for the\n    current platform.\n    \"\"\"\n\n    return Path(PureWindowsPath(path_str).as_posix())\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.tmpchdir","title":"tmpchdir","text":"

    Change current-working directories for the duration of the context

    Source code in prefect/utilities/filesystem.py
    @contextmanager\ndef tmpchdir(path: str):\n    \"\"\"\n    Change current-working directories for the duration of the context\n    \"\"\"\n    path = os.path.abspath(path)\n    if os.path.isfile(path) or (not os.path.exists(path) and not path.endswith(\"/\")):\n        path = os.path.dirname(path)\n\n    owd = os.getcwd()\n\n    try:\n        os.chdir(path)\n        yield path\n    finally:\n        os.chdir(owd)\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/filesystem/#prefect.utilities.filesystem.to_display_path","title":"to_display_path","text":"

    Convert a path to a displayable path. The absolute path or relative path to the current (or given) directory will be returned, whichever is shorter.

    Source code in prefect/utilities/filesystem.py
    def to_display_path(\n    path: Union[pathlib.Path, str], relative_to: Union[pathlib.Path, str] = None\n) -> str:\n    \"\"\"\n    Convert a path to a displayable path. The absolute path or relative path to the\n    current (or given) directory will be returned, whichever is shorter.\n    \"\"\"\n    path, relative_to = (\n        pathlib.Path(path).resolve(),\n        pathlib.Path(relative_to or \".\").resolve(),\n    )\n    relative_path = str(path.relative_to(relative_to))\n    absolute_path = str(path)\n    return relative_path if len(relative_path) < len(absolute_path) else absolute_path\n
    ","tags":["Python API","files","filesystems"]},{"location":"api-ref/prefect/utilities/hashing/","title":"hashing","text":"","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/hashing/#prefect.utilities.hashing","title":"prefect.utilities.hashing","text":"","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/hashing/#prefect.utilities.hashing.file_hash","title":"file_hash","text":"

    Given a path to a file, produces a stable hash of the file contents.

    Parameters:

    Name Type Description Default path str

    the path to a file

    required hash_algo

    Hash algorithm from hashlib to use.

    _md5

    Returns:

    Name Type Description str str

    a hash of the file contents

    Source code in prefect/utilities/hashing.py
    def file_hash(path: str, hash_algo=_md5) -> str:\n    \"\"\"Given a path to a file, produces a stable hash of the file contents.\n\n    Args:\n        path (str): the path to a file\n        hash_algo: Hash algorithm from hashlib to use.\n\n    Returns:\n        str: a hash of the file contents\n    \"\"\"\n    contents = Path(path).read_bytes()\n    return stable_hash(contents, hash_algo=hash_algo)\n
    ","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/hashing/#prefect.utilities.hashing.hash_objects","title":"hash_objects","text":"

    Attempt to hash objects by dumping to JSON or serializing with cloudpickle. On failure of both, None will be returned

    Source code in prefect/utilities/hashing.py
    def hash_objects(*args, hash_algo=_md5, **kwargs) -> Optional[str]:\n    \"\"\"\n    Attempt to hash objects by dumping to JSON or serializing with cloudpickle.\n    On failure of both, `None` will be returned\n    \"\"\"\n    try:\n        serializer = JSONSerializer(dumps_kwargs={\"sort_keys\": True})\n        return stable_hash(serializer.dumps((args, kwargs)), hash_algo=hash_algo)\n    except Exception:\n        pass\n\n    try:\n        return stable_hash(cloudpickle.dumps((args, kwargs)), hash_algo=hash_algo)\n    except Exception:\n        pass\n\n    return None\n
    ","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/hashing/#prefect.utilities.hashing.stable_hash","title":"stable_hash","text":"

    Given some arguments, produces a stable 64-bit hash of their contents.

    Supports bytes and strings. Strings will be UTF-8 encoded.

    Parameters:

    Name Type Description Default *args Union[str, bytes]

    Items to include in the hash.

    () hash_algo

    Hash algorithm from hashlib to use.

    _md5

    Returns:

    Type Description str

    A hex hash.

    Source code in prefect/utilities/hashing.py
    def stable_hash(*args: Union[str, bytes], hash_algo=_md5) -> str:\n    \"\"\"Given some arguments, produces a stable 64-bit hash of their contents.\n\n    Supports bytes and strings. Strings will be UTF-8 encoded.\n\n    Args:\n        *args: Items to include in the hash.\n        hash_algo: Hash algorithm from hashlib to use.\n\n    Returns:\n        A hex hash.\n    \"\"\"\n    h = hash_algo()\n    for a in args:\n        if isinstance(a, str):\n            a = a.encode()\n        h.update(a)\n    return h.hexdigest()\n
    ","tags":["Python API","hashes","hashing"]},{"location":"api-ref/prefect/utilities/importtools/","title":"importtools","text":"","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools","title":"prefect.utilities.importtools","text":"","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.AliasedModuleDefinition","title":"AliasedModuleDefinition","text":"

    Bases: NamedTuple

    A definition for the AliasedModuleFinder.

    Parameters:

    Name Type Description Default alias

    The import name to create

    required real

    The import name of the module to reference for the alias

    required callback

    A function to call when the alias module is loaded

    required Source code in prefect/utilities/importtools.py
    class AliasedModuleDefinition(NamedTuple):\n    \"\"\"\n    A definition for the `AliasedModuleFinder`.\n\n    Args:\n        alias: The import name to create\n        real: The import name of the module to reference for the alias\n        callback: A function to call when the alias module is loaded\n    \"\"\"\n\n    alias: str\n    real: str\n    callback: Optional[Callable[[str], None]]\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.AliasedModuleFinder","title":"AliasedModuleFinder","text":"

    Bases: MetaPathFinder

    Source code in prefect/utilities/importtools.py
    class AliasedModuleFinder(MetaPathFinder):\n    def __init__(self, aliases: Iterable[AliasedModuleDefinition]):\n        \"\"\"\n        See `AliasedModuleDefinition` for alias specification.\n\n        Aliases apply to all modules nested within an alias.\n        \"\"\"\n        self.aliases = aliases\n\n    def find_spec(\n        self,\n        fullname: str,\n        path=None,\n        target=None,\n    ) -> Optional[ModuleSpec]:\n        \"\"\"\n        The fullname is the imported path, e.g. \"foo.bar\". If there is an alias \"phi\"\n        for \"foo\" then on import of \"phi.bar\" we will find the spec for \"foo.bar\" and\n        create a new spec for \"phi.bar\" that points to \"foo.bar\".\n        \"\"\"\n        for alias, real, callback in self.aliases:\n            if fullname.startswith(alias):\n                # Retrieve the spec of the real module\n                real_spec = importlib.util.find_spec(fullname.replace(alias, real, 1))\n                # Create a new spec for the alias\n                return ModuleSpec(\n                    fullname,\n                    AliasedModuleLoader(fullname, callback, real_spec),\n                    origin=real_spec.origin,\n                    is_package=real_spec.submodule_search_locations is not None,\n                )\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.AliasedModuleFinder.find_spec","title":"find_spec","text":"

    The fullname is the imported path, e.g. \"foo.bar\". If there is an alias \"phi\" for \"foo\" then on import of \"phi.bar\" we will find the spec for \"foo.bar\" and create a new spec for \"phi.bar\" that points to \"foo.bar\".

    Source code in prefect/utilities/importtools.py
    def find_spec(\n    self,\n    fullname: str,\n    path=None,\n    target=None,\n) -> Optional[ModuleSpec]:\n    \"\"\"\n    The fullname is the imported path, e.g. \"foo.bar\". If there is an alias \"phi\"\n    for \"foo\" then on import of \"phi.bar\" we will find the spec for \"foo.bar\" and\n    create a new spec for \"phi.bar\" that points to \"foo.bar\".\n    \"\"\"\n    for alias, real, callback in self.aliases:\n        if fullname.startswith(alias):\n            # Retrieve the spec of the real module\n            real_spec = importlib.util.find_spec(fullname.replace(alias, real, 1))\n            # Create a new spec for the alias\n            return ModuleSpec(\n                fullname,\n                AliasedModuleLoader(fullname, callback, real_spec),\n                origin=real_spec.origin,\n                is_package=real_spec.submodule_search_locations is not None,\n            )\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.DelayedImportErrorModule","title":"DelayedImportErrorModule","text":"

    Bases: ModuleType

    A fake module returned by lazy_import when the module cannot be found. When any of the module's attributes are accessed, we will throw a ModuleNotFoundError.

    Adapted from lazy_loader

    Source code in prefect/utilities/importtools.py
    class DelayedImportErrorModule(ModuleType):\n    \"\"\"\n    A fake module returned by `lazy_import` when the module cannot be found. When any\n    of the module's attributes are accessed, we will throw a `ModuleNotFoundError`.\n\n    Adapted from [lazy_loader][1]\n\n    [1]: https://github.com/scientific-python/lazy_loader\n    \"\"\"\n\n    def __init__(self, frame_data, help_message, *args, **kwargs):\n        self.__frame_data = frame_data\n        self.__help_message = (\n            help_message or \"Import errors for this module are only reported when used.\"\n        )\n        super().__init__(*args, **kwargs)\n\n    def __getattr__(self, attr):\n        if attr in (\"__class__\", \"__file__\", \"__frame_data\", \"__help_message\"):\n            super().__getattr__(attr)\n        else:\n            fd = self.__frame_data\n            raise ModuleNotFoundError(\n                f\"No module named '{fd['spec']}'\\n\\nThis module was originally imported\"\n                f\" at:\\n  File \\\"{fd['filename']}\\\", line {fd['lineno']}, in\"\n                f\" {fd['function']}\\n\\n    {''.join(fd['code_context']).strip()}\\n\"\n                + self.__help_message\n            )\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.from_qualified_name","title":"from_qualified_name","text":"

    Import an object given a fully-qualified name.

    Parameters:

    Name Type Description Default name str

    The fully-qualified name of the object to import.

    required

    Returns:

    Type Description Any

    the imported object

    Examples:

    >>> obj = from_qualified_name(\"random.randint\")\n>>> import random\n>>> obj == random.randint\nTrue\n
    Source code in prefect/utilities/importtools.py
    def from_qualified_name(name: str) -> Any:\n    \"\"\"\n    Import an object given a fully-qualified name.\n\n    Args:\n        name: The fully-qualified name of the object to import.\n\n    Returns:\n        the imported object\n\n    Examples:\n        >>> obj = from_qualified_name(\"random.randint\")\n        >>> import random\n        >>> obj == random.randint\n        True\n    \"\"\"\n    # Try importing it first so we support \"module\" or \"module.sub_module\"\n    try:\n        module = importlib.import_module(name)\n        return module\n    except ImportError:\n        # If no subitem was included raise the import error\n        if \".\" not in name:\n            raise\n\n    # Otherwise, we'll try to load it as an attribute of a module\n    mod_name, attr_name = name.rsplit(\".\", 1)\n    module = importlib.import_module(mod_name)\n    return getattr(module, attr_name)\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.import_object","title":"import_object","text":"

    Load an object from an import path.

    Import paths can be formatted as one of: - module.object - module:object - /path/to/script.py:object

    This function is not thread safe as it modifies the 'sys' module during execution.

    Source code in prefect/utilities/importtools.py
    def import_object(import_path: str):\n    \"\"\"\n    Load an object from an import path.\n\n    Import paths can be formatted as one of:\n    - module.object\n    - module:object\n    - /path/to/script.py:object\n\n    This function is not thread safe as it modifies the 'sys' module during execution.\n    \"\"\"\n    if \".py:\" in import_path:\n        script_path, object_name = import_path.rsplit(\":\", 1)\n        module = load_script_as_module(script_path)\n    else:\n        if \":\" in import_path:\n            module_name, object_name = import_path.rsplit(\":\", 1)\n        elif \".\" in import_path:\n            module_name, object_name = import_path.rsplit(\".\", 1)\n        else:\n            raise ValueError(\n                f\"Invalid format for object import. Received {import_path!r}.\"\n            )\n\n        module = load_module(module_name)\n\n    return getattr(module, object_name)\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.lazy_import","title":"lazy_import","text":"

    Create a lazily-imported module to use in place of the module of the given name. Use this to retain module-level imports for libraries that we don't want to actually import until they are needed.

    Adapted from the Python documentation and lazy_loader

    Source code in prefect/utilities/importtools.py
    def lazy_import(\n    name: str, error_on_import: bool = False, help_message: str = \"\"\n) -> ModuleType:\n    \"\"\"\n    Create a lazily-imported module to use in place of the module of the given name.\n    Use this to retain module-level imports for libraries that we don't want to\n    actually import until they are needed.\n\n    Adapted from the [Python documentation][1] and [lazy_loader][2]\n\n    [1]: https://docs.python.org/3/library/importlib.html#implementing-lazy-imports\n    [2]: https://github.com/scientific-python/lazy_loader\n    \"\"\"\n\n    try:\n        return sys.modules[name]\n    except KeyError:\n        pass\n\n    spec = importlib.util.find_spec(name)\n    if spec is None:\n        if error_on_import:\n            raise ModuleNotFoundError(f\"No module named '{name}'.\\n{help_message}\")\n        else:\n            try:\n                parent = inspect.stack()[1]\n                frame_data = {\n                    \"spec\": name,\n                    \"filename\": parent.filename,\n                    \"lineno\": parent.lineno,\n                    \"function\": parent.function,\n                    \"code_context\": parent.code_context,\n                }\n                return DelayedImportErrorModule(\n                    frame_data, help_message, \"DelayedImportErrorModule\"\n                )\n            finally:\n                del parent\n\n    module = importlib.util.module_from_spec(spec)\n    sys.modules[name] = module\n\n    loader = importlib.util.LazyLoader(spec.loader)\n    loader.exec_module(module)\n\n    return module\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.load_module","title":"load_module","text":"

    Import a module with support for relative imports within the module.

    Source code in prefect/utilities/importtools.py
    def load_module(module_name: str) -> ModuleType:\n    \"\"\"\n    Import a module with support for relative imports within the module.\n    \"\"\"\n    # Ensure relative imports within the imported module work if the user is in the\n    # correct working directory\n    working_directory = os.getcwd()\n    sys.path.insert(0, working_directory)\n\n    try:\n        return importlib.import_module(module_name)\n    finally:\n        sys.path.remove(working_directory)\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.load_script_as_module","title":"load_script_as_module","text":"

    Execute a script at the given path.

    Sets the module name to __prefect_loader__.

    If an exception occurs during execution of the script, a prefect.exceptions.ScriptError is created to wrap the exception and raised.

    During the duration of this function call, sys is modified to support loading. These changes are reverted after completion, but this function is not thread safe and use of it in threaded contexts may result in undesirable behavior.

    See https://docs.python.org/3/library/importlib.html#importing-a-source-file-directly

    Source code in prefect/utilities/importtools.py
    def load_script_as_module(path: str) -> ModuleType:\n    \"\"\"\n    Execute a script at the given path.\n\n    Sets the module name to `__prefect_loader__`.\n\n    If an exception occurs during execution of the script, a\n    `prefect.exceptions.ScriptError` is created to wrap the exception and raised.\n\n    During the duration of this function call, `sys` is modified to support loading.\n    These changes are reverted after completion, but this function is not thread safe\n    and use of it in threaded contexts may result in undesirable behavior.\n\n    See https://docs.python.org/3/library/importlib.html#importing-a-source-file-directly\n    \"\"\"\n    # We will add the parent directory to search locations to support relative imports\n    # during execution of the script\n    if not path.endswith(\".py\"):\n        raise ValueError(f\"The provided path does not point to a python file: {path!r}\")\n\n    parent_path = str(Path(path).resolve().parent)\n    working_directory = os.getcwd()\n\n    spec = importlib.util.spec_from_file_location(\n        \"__prefect_loader__\",\n        path,\n        # Support explicit relative imports i.e. `from .foo import bar`\n        submodule_search_locations=[parent_path, working_directory],\n    )\n    module = importlib.util.module_from_spec(spec)\n    sys.modules[\"__prefect_loader__\"] = module\n\n    # Support implicit relative imports i.e. `from foo import bar`\n    sys.path.insert(0, working_directory)\n    sys.path.insert(0, parent_path)\n    try:\n        spec.loader.exec_module(module)\n    except Exception as exc:\n        raise ScriptError(user_exc=exc, path=path) from exc\n    finally:\n        sys.modules.pop(\"__prefect_loader__\")\n        sys.path.remove(parent_path)\n        sys.path.remove(working_directory)\n\n    return module\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.objects_from_script","title":"objects_from_script","text":"

    Run a python script and return all the global variables

    Supports remote paths by copying to a local temporary file.

    WARNING: The Python documentation does not recommend using runpy for this pattern.

    Furthermore, any functions and classes defined by the executed code are not guaranteed to work correctly after a runpy function has returned. If that limitation is not acceptable for a given use case, importlib is likely to be a more suitable choice than this module.

    The function load_script_as_module uses importlib instead and should be used instead for loading objects from scripts.

    Parameters:

    Name Type Description Default path str

    The path to the script to run

    required text Union[str, bytes]

    Optionally, the text of the script. Skips loading the contents if given.

    None

    Returns:

    Type Description Dict[str, Any]

    A dictionary mapping variable name to value

    Raises:

    Type Description ScriptError

    if the script raises an exception during execution

    Source code in prefect/utilities/importtools.py
    def objects_from_script(path: str, text: Union[str, bytes] = None) -> Dict[str, Any]:\n    \"\"\"\n    Run a python script and return all the global variables\n\n    Supports remote paths by copying to a local temporary file.\n\n    WARNING: The Python documentation does not recommend using runpy for this pattern.\n\n    > Furthermore, any functions and classes defined by the executed code are not\n    > guaranteed to work correctly after a runpy function has returned. If that\n    > limitation is not acceptable for a given use case, importlib is likely to be a\n    > more suitable choice than this module.\n\n    The function `load_script_as_module` uses importlib instead and should be used\n    instead for loading objects from scripts.\n\n    Args:\n        path: The path to the script to run\n        text: Optionally, the text of the script. Skips loading the contents if given.\n\n    Returns:\n        A dictionary mapping variable name to value\n\n    Raises:\n        ScriptError: if the script raises an exception during execution\n    \"\"\"\n\n    def run_script(run_path: str):\n        # Cast to an absolute path before changing directories to ensure relative paths\n        # are not broken\n        abs_run_path = os.path.abspath(run_path)\n        with tmpchdir(run_path):\n            try:\n                return runpy.run_path(abs_run_path)\n            except Exception as exc:\n                raise ScriptError(user_exc=exc, path=path) from exc\n\n    if text:\n        with NamedTemporaryFile(\n            mode=\"wt\" if isinstance(text, str) else \"wb\",\n            prefix=f\"run-{filename(path)}\",\n            suffix=\".py\",\n        ) as tmpfile:\n            tmpfile.write(text)\n            tmpfile.flush()\n            return run_script(tmpfile.name)\n\n    else:\n        if not is_local_path(path):\n            # Remote paths need to be local to run\n            with fsspec.open(path) as f:\n                contents = f.read()\n            return objects_from_script(path, contents)\n        else:\n            return run_script(path)\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/importtools/#prefect.utilities.importtools.to_qualified_name","title":"to_qualified_name","text":"

    Given an object, returns its fully-qualified name: a string that represents its Python import path.

    Parameters:

    Name Type Description Default obj Any

    an importable Python object

    required

    Returns:

    Name Type Description str str

    the qualified name

    Source code in prefect/utilities/importtools.py
    def to_qualified_name(obj: Any) -> str:\n    \"\"\"\n    Given an object, returns its fully-qualified name: a string that represents its\n    Python import path.\n\n    Args:\n        obj (Any): an importable Python object\n\n    Returns:\n        str: the qualified name\n    \"\"\"\n    if sys.version_info < (3, 10):\n        # These attributes are only available in Python 3.10+\n        if isinstance(obj, (classmethod, staticmethod)):\n            obj = obj.__func__\n    return obj.__module__ + \".\" + obj.__qualname__\n
    ","tags":["Python API","imports"]},{"location":"api-ref/prefect/utilities/math/","title":"math","text":"","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math","title":"prefect.utilities.math","text":"","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math.bounded_poisson_interval","title":"bounded_poisson_interval","text":"

    Bounds Poisson \"inter-arrival times\" to a range.

    Unlike clamped_poisson_interval this does not take a target average interval. Instead, the interval is predetermined and the average is calculated as their midpoint. This allows Poisson intervals to be used in cases where a lower bound must be enforced.

    Source code in prefect/utilities/math.py
    def bounded_poisson_interval(lower_bound, upper_bound):\n    \"\"\"\n    Bounds Poisson \"inter-arrival times\" to a range.\n\n    Unlike `clamped_poisson_interval` this does not take a target average interval.\n    Instead, the interval is predetermined and the average is calculated as their\n    midpoint. This allows Poisson intervals to be used in cases where a lower bound\n    must be enforced.\n    \"\"\"\n    average = (float(lower_bound) + float(upper_bound)) / 2.0\n    upper_rv = exponential_cdf(upper_bound, average)\n    lower_rv = exponential_cdf(lower_bound, average)\n    return poisson_interval(average, lower_rv, upper_rv)\n
    ","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math.clamped_poisson_interval","title":"clamped_poisson_interval","text":"

    Bounds Poisson \"inter-arrival times\" to a range defined by the clamping factor.

    The upper bound for this random variate is: average_interval * (1 + clamping_factor). A lower bound is picked so that the average interval remains approximately fixed.

    Source code in prefect/utilities/math.py
    def clamped_poisson_interval(average_interval, clamping_factor=0.3):\n    \"\"\"\n    Bounds Poisson \"inter-arrival times\" to a range defined by the clamping factor.\n\n    The upper bound for this random variate is: average_interval * (1 + clamping_factor).\n    A lower bound is picked so that the average interval remains approximately fixed.\n    \"\"\"\n    if clamping_factor <= 0:\n        raise ValueError(\"`clamping_factor` must be >= 0.\")\n\n    upper_clamp_multiple = 1 + clamping_factor\n    upper_bound = average_interval * upper_clamp_multiple\n    lower_bound = max(0, average_interval * lower_clamp_multiple(upper_clamp_multiple))\n\n    upper_rv = exponential_cdf(upper_bound, average_interval)\n    lower_rv = exponential_cdf(lower_bound, average_interval)\n    return poisson_interval(average_interval, lower_rv, upper_rv)\n
    ","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math.lower_clamp_multiple","title":"lower_clamp_multiple","text":"

    Computes a lower clamp multiple that can be used to bound a random variate drawn from an exponential distribution.

    Given an upper clamp multiple k (and corresponding upper bound k * average_interval), this function computes a lower clamp multiple c (corresponding to a lower bound c * average_interval) where the probability mass between the lower bound and the median is equal to the probability mass between the median and the upper bound.

    Source code in prefect/utilities/math.py
    def lower_clamp_multiple(k):\n    \"\"\"\n    Computes a lower clamp multiple that can be used to bound a random variate drawn\n    from an exponential distribution.\n\n    Given an upper clamp multiple `k` (and corresponding upper bound k * average_interval),\n    this function computes a lower clamp multiple `c` (corresponding to a lower bound\n    c * average_interval) where the probability mass between the lower bound and the\n    median is equal to the probability mass between the median and the upper bound.\n    \"\"\"\n    if k >= 50:\n        # return 0 for large values of `k` to prevent numerical overflow\n        return 0.0\n\n    return math.log(max(2**k / (2**k - 1), 1e-10), 2)\n
    ","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/math/#prefect.utilities.math.poisson_interval","title":"poisson_interval","text":"

    Generates an \"inter-arrival time\" for a Poisson process.

    Draws a random variable from an exponential distribution using the inverse-CDF method. Can optionally be passed a lower and upper bound between (0, 1] to clamp the potential output values.

    Source code in prefect/utilities/math.py
    def poisson_interval(average_interval, lower=0, upper=1):\n    \"\"\"\n    Generates an \"inter-arrival time\" for a Poisson process.\n\n    Draws a random variable from an exponential distribution using the inverse-CDF\n    method. Can optionally be passed a lower and upper bound between (0, 1] to clamp\n    the potential output values.\n    \"\"\"\n\n    # note that we ensure the argument to the logarithm is stabilized to prevent\n    # calling log(0), which results in a DomainError\n    return -math.log(max(1 - random.uniform(lower, upper), 1e-10)) * average_interval\n
    ","tags":["Python API","math"]},{"location":"api-ref/prefect/utilities/names/","title":"names","text":"","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/names/#prefect.utilities.names","title":"prefect.utilities.names","text":"","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/names/#prefect.utilities.names.generate_slug","title":"generate_slug","text":"

    Generates a random slug.

    Parameters:

    Name Type Description Default - n_words (int

    the number of words in the slug

    required Source code in prefect/utilities/names.py
    def generate_slug(n_words: int) -> str:\n    \"\"\"\n    Generates a random slug.\n\n    Args:\n        - n_words (int): the number of words in the slug\n    \"\"\"\n    words = coolname.generate(n_words)\n\n    # regenerate words if they include ignored words\n    while IGNORE_LIST.intersection(words):\n        words = coolname.generate(n_words)\n\n    return \"-\".join(words)\n
    ","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/names/#prefect.utilities.names.obfuscate","title":"obfuscate","text":"

    Obfuscates any data type's string representation. See obfuscate_string.

    Source code in prefect/utilities/names.py
    def obfuscate(s: Any, show_tail=False) -> str:\n    \"\"\"\n    Obfuscates any data type's string representation. See `obfuscate_string`.\n    \"\"\"\n    if s is None:\n        return OBFUSCATED_PREFIX + \"*\" * 4\n\n    return obfuscate_string(str(s), show_tail=show_tail)\n
    ","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/names/#prefect.utilities.names.obfuscate_string","title":"obfuscate_string","text":"

    Obfuscates a string by returning a new string of 8 characters. If the input string is longer than 10 characters and show_tail is True, then up to 4 of its final characters will become final characters of the obfuscated string; all other characters are \"*\".

    \"abc\" -> \"*\" \"abcdefgh\" -> \"*\" \"abcdefghijk\" -> \"*k\" \"abcdefghijklmnopqrs\" -> \"****pqrs\"

    Source code in prefect/utilities/names.py
    def obfuscate_string(s: str, show_tail=False) -> str:\n    \"\"\"\n    Obfuscates a string by returning a new string of 8 characters. If the input\n    string is longer than 10 characters and show_tail is True, then up to 4 of\n    its final characters will become final characters of the obfuscated string;\n    all other characters are \"*\".\n\n    \"abc\"      -> \"********\"\n    \"abcdefgh\" -> \"********\"\n    \"abcdefghijk\" -> \"*******k\"\n    \"abcdefghijklmnopqrs\" -> \"****pqrs\"\n    \"\"\"\n    result = OBFUSCATED_PREFIX + \"*\" * 4\n    # take up to 4 characters, but only after the 10th character\n    suffix = s[10:][-4:]\n    if suffix and show_tail:\n        result = f\"{result[:-len(suffix)]}{suffix}\"\n    return result\n
    ","tags":["Python API","names"]},{"location":"api-ref/prefect/utilities/processutils/","title":"processutils","text":""},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils","title":"prefect.utilities.processutils","text":""},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.forward_signal_handler","title":"forward_signal_handler","text":"

    Forward subsequent signum events (e.g. interrupts) to respective signums.

    Source code in prefect/utilities/processutils.py
    def forward_signal_handler(\n    pid: int, signum: int, *signums: int, process_name: str, print_fn: Callable\n):\n    \"\"\"Forward subsequent signum events (e.g. interrupts) to respective signums.\"\"\"\n    current_signal, future_signals = signums[0], signums[1:]\n\n    # avoid RecursionError when setting up a direct signal forward to the same signal for the main pid\n    avoid_infinite_recursion = signum == current_signal and pid == os.getpid()\n    if avoid_infinite_recursion:\n        # store the vanilla handler so it can be temporarily restored below\n        original_handler = signal.getsignal(current_signal)\n\n    def handler(*args):\n        print_fn(\n            f\"Received {getattr(signum, 'name', signum)}. \"\n            f\"Sending {getattr(current_signal, 'name', current_signal)} to\"\n            f\" {process_name} (PID {pid})...\"\n        )\n        if avoid_infinite_recursion:\n            signal.signal(current_signal, original_handler)\n        os.kill(pid, current_signal)\n        if future_signals:\n            forward_signal_handler(\n                pid,\n                signum,\n                *future_signals,\n                process_name=process_name,\n                print_fn=print_fn,\n            )\n\n    # register current and future signal handlers\n    _register_signal(signum, handler)\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.open_process","title":"open_process async","text":"

    Like anyio.open_process but with: - Support for Windows command joining - Termination of the process on exception during yield - Forced cleanup of process resources during cancellation

    Source code in prefect/utilities/processutils.py
    @asynccontextmanager\nasync def open_process(command: List[str], **kwargs):\n    \"\"\"\n    Like `anyio.open_process` but with:\n    - Support for Windows command joining\n    - Termination of the process on exception during yield\n    - Forced cleanup of process resources during cancellation\n    \"\"\"\n    # Passing a string to open_process is equivalent to shell=True which is\n    # generally necessary for Unix-like commands on Windows but otherwise should\n    # be avoided\n    if not isinstance(command, list):\n        raise TypeError(\n            \"The command passed to open process must be a list. You passed the command\"\n            f\"'{command}', which is type '{type(command)}'.\"\n        )\n\n    if sys.platform == \"win32\":\n        command = \" \".join(command)\n        process = await _open_anyio_process(command, **kwargs)\n    else:\n        process = await anyio.open_process(command, **kwargs)\n\n    # if there's a creationflags kwarg and it contains CREATE_NEW_PROCESS_GROUP,\n    # use SetConsoleCtrlHandler to handle CTRL-C\n    win32_process_group = False\n    if (\n        sys.platform == \"win32\"\n        and \"creationflags\" in kwargs\n        and kwargs[\"creationflags\"] & subprocess.CREATE_NEW_PROCESS_GROUP\n    ):\n        win32_process_group = True\n        _windows_process_group_pids.add(process.pid)\n        # Add a handler for CTRL-C. Re-adding the handler is safe as Windows\n        # will not add a duplicate handler if _win32_ctrl_handler is\n        # already registered.\n        windll.kernel32.SetConsoleCtrlHandler(_win32_ctrl_handler, 1)\n\n    try:\n        async with process:\n            yield process\n    finally:\n        try:\n            process.terminate()\n            if win32_process_group:\n                _windows_process_group_pids.remove(process.pid)\n\n        except OSError:\n            # Occurs if the process is already terminated\n            pass\n\n        # Ensure the process resource is closed. If not shielded from cancellation,\n        # this resource can be left open and the subprocess output can appear after\n        # the parent process has exited.\n        with anyio.CancelScope(shield=True):\n            await process.aclose()\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.run_process","title":"run_process async","text":"

    Like anyio.run_process but with:

    • Use of our open_process utility to ensure resources are cleaned up
    • Simple stream_output support to connect the subprocess to the parent stdout/err
    • Support for submission with TaskGroup.start marking as 'started' after the process has been created. When used, the PID is returned to the task status.
    Source code in prefect/utilities/processutils.py
    async def run_process(\n    command: List[str],\n    stream_output: Union[bool, Tuple[Optional[TextSink], Optional[TextSink]]] = False,\n    task_status: Optional[anyio.abc.TaskStatus] = None,\n    task_status_handler: Optional[Callable[[anyio.abc.Process], Any]] = None,\n    **kwargs,\n):\n    \"\"\"\n    Like `anyio.run_process` but with:\n\n    - Use of our `open_process` utility to ensure resources are cleaned up\n    - Simple `stream_output` support to connect the subprocess to the parent stdout/err\n    - Support for submission with `TaskGroup.start` marking as 'started' after the\n        process has been created. When used, the PID is returned to the task status.\n\n    \"\"\"\n    if stream_output is True:\n        stream_output = (sys.stdout, sys.stderr)\n\n    async with open_process(\n        command,\n        stdout=subprocess.PIPE if stream_output else subprocess.DEVNULL,\n        stderr=subprocess.PIPE if stream_output else subprocess.DEVNULL,\n        **kwargs,\n    ) as process:\n        if task_status is not None:\n            if not task_status_handler:\n\n                def task_status_handler(process):\n                    return process.pid\n\n            task_status.started(task_status_handler(process))\n\n        if stream_output:\n            await consume_process_output(\n                process, stdout_sink=stream_output[0], stderr_sink=stream_output[1]\n            )\n\n        await process.wait()\n\n    return process\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.setup_signal_handlers_agent","title":"setup_signal_handlers_agent","text":"

    Handle interrupts of the agent gracefully.

    Source code in prefect/utilities/processutils.py
    def setup_signal_handlers_agent(pid: int, process_name: str, print_fn: Callable):\n    \"\"\"Handle interrupts of the agent gracefully.\"\"\"\n    setup_handler = partial(\n        forward_signal_handler, pid, process_name=process_name, print_fn=print_fn\n    )\n    # when agent receives SIGINT, it stops dequeueing new FlowRuns, and runs until the subprocesses finish\n    # the signal is not forwarded to subprocesses, so they can continue to run and hopefully still complete\n    if sys.platform == \"win32\":\n        # on Windows, use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        setup_handler(signal.SIGINT, signal.CTRL_BREAK_EVENT)\n    else:\n        # forward first SIGINT directly, send SIGKILL on subsequent interrupt\n        setup_handler(signal.SIGINT, signal.SIGINT, signal.SIGKILL)\n        # first SIGTERM: send SIGINT, send SIGKILL on subsequent SIGTERM\n        setup_handler(signal.SIGTERM, signal.SIGINT, signal.SIGKILL)\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.setup_signal_handlers_server","title":"setup_signal_handlers_server","text":"

    Handle interrupts of the server gracefully.

    Source code in prefect/utilities/processutils.py
    def setup_signal_handlers_server(pid: int, process_name: str, print_fn: Callable):\n    \"\"\"Handle interrupts of the server gracefully.\"\"\"\n    setup_handler = partial(\n        forward_signal_handler, pid, process_name=process_name, print_fn=print_fn\n    )\n    # when server receives a signal, it needs to be propagated to the uvicorn subprocess\n    if sys.platform == \"win32\":\n        # on Windows, use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        setup_handler(signal.SIGINT, signal.CTRL_BREAK_EVENT)\n    else:\n        # first interrupt: SIGTERM, second interrupt: SIGKILL\n        setup_handler(signal.SIGINT, signal.SIGTERM, signal.SIGKILL)\n        # forward first SIGTERM directly, send SIGKILL on subsequent SIGTERM\n        setup_handler(signal.SIGTERM, signal.SIGTERM, signal.SIGKILL)\n
    "},{"location":"api-ref/prefect/utilities/processutils/#prefect.utilities.processutils.setup_signal_handlers_worker","title":"setup_signal_handlers_worker","text":"

    Handle interrupts of workers gracefully.

    Source code in prefect/utilities/processutils.py
    def setup_signal_handlers_worker(pid: int, process_name: str, print_fn: Callable):\n    \"\"\"Handle interrupts of workers gracefully.\"\"\"\n    setup_handler = partial(\n        forward_signal_handler, pid, process_name=process_name, print_fn=print_fn\n    )\n    # when agent receives SIGINT, it stops dequeueing new FlowRuns, and runs until the subprocesses finish\n    # the signal is not forwarded to subprocesses, so they can continue to run and hopefully still complete\n    if sys.platform == \"win32\":\n        # on Windows, use CTRL_BREAK_EVENT as SIGTERM is useless:\n        # https://bugs.python.org/issue26350\n        setup_handler(signal.SIGINT, signal.CTRL_BREAK_EVENT)\n    else:\n        # forward first SIGINT directly, send SIGKILL on subsequent interrupt\n        setup_handler(signal.SIGINT, signal.SIGINT, signal.SIGKILL)\n        # first SIGTERM: send SIGINT, send SIGKILL on subsequent SIGTERM\n        setup_handler(signal.SIGTERM, signal.SIGINT, signal.SIGKILL)\n
    "},{"location":"api-ref/prefect/utilities/pydantic/","title":"pydantic","text":"","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic","title":"prefect.utilities.pydantic","text":"","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic.PartialModel","title":"PartialModel","text":"

    Bases: Generic[M]

    A utility for creating a Pydantic model in several steps.

    Fields may be set at initialization, via attribute assignment, or at finalization when the concrete model is returned.

    Pydantic validation does not occur until finalization.

    Each field can only be set once and a ValueError will be raised on assignment if a field already has a value.

    Example

    class MyModel(pydantic.BaseModel): x: int y: str z: float

    partial_model = PartialModel(MyModel, x=1) partial_model.y = \"two\" model = partial_model.finalize(z=3.0)

    Source code in prefect/utilities/pydantic.py
    class PartialModel(Generic[M]):\n    \"\"\"\n    A utility for creating a Pydantic model in several steps.\n\n    Fields may be set at initialization, via attribute assignment, or at finalization\n    when the concrete model is returned.\n\n    Pydantic validation does not occur until finalization.\n\n    Each field can only be set once and a `ValueError` will be raised on assignment if\n    a field already has a value.\n\n    Example:\n        >>> class MyModel(pydantic.BaseModel):\n        >>>     x: int\n        >>>     y: str\n        >>>     z: float\n        >>>\n        >>> partial_model = PartialModel(MyModel, x=1)\n        >>> partial_model.y = \"two\"\n        >>> model = partial_model.finalize(z=3.0)\n    \"\"\"\n\n    def __init__(self, __model_cls: Type[M], **kwargs: Any) -> None:\n        self.fields = kwargs\n        # Set fields first to avoid issues if `fields` is also set on the `model_cls`\n        # in our custom `setattr` implementation.\n        self.model_cls = __model_cls\n\n        for name in kwargs.keys():\n            self.raise_if_not_in_model(name)\n\n    def finalize(self, **kwargs: Any) -> M:\n        for name in kwargs.keys():\n            self.raise_if_already_set(name)\n            self.raise_if_not_in_model(name)\n        return self.model_cls(**self.fields, **kwargs)\n\n    def raise_if_already_set(self, name):\n        if name in self.fields:\n            raise ValueError(f\"Field {name!r} has already been set.\")\n\n    def raise_if_not_in_model(self, name):\n        if name not in self.model_cls.__fields__:\n            raise ValueError(f\"Field {name!r} is not present in the model.\")\n\n    def __setattr__(self, __name: str, __value: Any) -> None:\n        if __name in {\"fields\", \"model_cls\"}:\n            return super().__setattr__(__name, __value)\n\n        self.raise_if_already_set(__name)\n        self.raise_if_not_in_model(__name)\n        self.fields[__name] = __value\n\n    def __repr__(self) -> str:\n        dsp_fields = \", \".join(\n            f\"{key}={repr(value)}\" for key, value in self.fields.items()\n        )\n        return f\"PartialModel(cls={self.model_cls.__name__}, {dsp_fields})\"\n
    ","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic.add_cloudpickle_reduction","title":"add_cloudpickle_reduction","text":"

    Adds a __reducer__ to the given class that ensures it is cloudpickle compatible.

    Workaround for issues with cloudpickle when using cythonized pydantic which throws exceptions when attempting to pickle the class which has \"compiled\" validator methods dynamically attached to it.

    We cannot define this utility in the model class itself because the class is the type that contains unserializable methods.

    Any model using some features of Pydantic (e.g. Path validation) with a Cython compiled Pydantic installation may encounter pickling issues.

    See related issue at https://github.com/cloudpipe/cloudpickle/issues/408

    Source code in prefect/utilities/pydantic.py
    def add_cloudpickle_reduction(__model_cls: Type[M] = None, **kwargs: Any):\n    \"\"\"\n    Adds a `__reducer__` to the given class that ensures it is cloudpickle compatible.\n\n    Workaround for issues with cloudpickle when using cythonized pydantic which\n    throws exceptions when attempting to pickle the class which has \"compiled\"\n    validator methods dynamically attached to it.\n\n    We cannot define this utility in the model class itself because the class is the\n    type that contains unserializable methods.\n\n    Any model using some features of Pydantic (e.g. `Path` validation) with a Cython\n    compiled Pydantic installation may encounter pickling issues.\n\n    See related issue at https://github.com/cloudpipe/cloudpickle/issues/408\n    \"\"\"\n    if __model_cls:\n        __model_cls.__reduce__ = _reduce_model\n        __model_cls.__reduce_kwargs__ = kwargs\n        return __model_cls\n    else:\n        return cast(\n            Callable[[Type[M]], Type[M]],\n            partial(\n                add_cloudpickle_reduction,\n                **kwargs,\n            ),\n        )\n
    ","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic.add_type_dispatch","title":"add_type_dispatch","text":"

    Extend a Pydantic model to add a 'type' field that is used a discriminator field to dynamically determine the subtype that when deserializing models.

    This allows automatic resolution to subtypes of the decorated model.

    If a type field already exists, it should be a string literal field that has a constant value for each subclass. The default value of this field will be used as the dispatch key.

    If a type field does not exist, one will be added. In this case, the value of the field will be set to the value of the __dispatch_key__. The base class should define a __dispatch_key__ class method that is used to determine the unique key for each subclass. Alternatively, each subclass can define the __dispatch_key__ as a string literal.

    The base class must not define a 'type' field. If it is not desirable to add a field to the model and the dispatch key can be tracked separately, the lower level utilities in prefect.utilities.dispatch should be used directly.

    Source code in prefect/utilities/pydantic.py
    def add_type_dispatch(model_cls: Type[M]) -> Type[M]:\n    \"\"\"\n    Extend a Pydantic model to add a 'type' field that is used a discriminator field\n    to dynamically determine the subtype that when deserializing models.\n\n    This allows automatic resolution to subtypes of the decorated model.\n\n    If a type field already exists, it should be a string literal field that has a\n    constant value for each subclass. The default value of this field will be used as\n    the dispatch key.\n\n    If a type field does not exist, one will be added. In this case, the value of the\n    field will be set to the value of the `__dispatch_key__`. The base class should\n    define a `__dispatch_key__` class method that is used to determine the unique key\n    for each subclass. Alternatively, each subclass can define the `__dispatch_key__`\n    as a string literal.\n\n    The base class must not define a 'type' field. If it is not desirable to add a field\n    to the model and the dispatch key can be tracked separately, the lower level\n    utilities in `prefect.utilities.dispatch` should be used directly.\n    \"\"\"\n    defines_dispatch_key = hasattr(\n        model_cls, \"__dispatch_key__\"\n    ) or \"__dispatch_key__\" in getattr(model_cls, \"__annotations__\", {})\n\n    defines_type_field = \"type\" in model_cls.__fields__\n\n    if not defines_dispatch_key and not defines_type_field:\n        raise ValueError(\n            f\"Model class {model_cls.__name__!r} does not define a `__dispatch_key__` \"\n            \"or a type field. One of these is required for dispatch.\"\n        )\n\n    elif defines_dispatch_key and not defines_type_field:\n        # Add a type field to store the value of the dispatch key\n        model_cls.__fields__[\"type\"] = pydantic.fields.ModelField(\n            name=\"type\",\n            type_=str,\n            required=True,\n            class_validators=None,\n            model_config=model_cls.__config__,\n        )\n\n    elif not defines_dispatch_key and defines_type_field:\n        field_type_annotation = model_cls.__fields__[\"type\"].type_\n        if field_type_annotation != str:\n            raise TypeError(\n                f\"Model class {model_cls.__name__!r} defines a 'type' field with \"\n                f\"type {field_type_annotation.__name__!r} but it must be 'str'.\"\n            )\n\n        # Set the dispatch key to retrieve the value from the type field\n        @classmethod\n        def dispatch_key_from_type_field(cls):\n            return cls.__fields__[\"type\"].default\n\n        model_cls.__dispatch_key__ = dispatch_key_from_type_field\n\n    else:\n        raise ValueError(\n            f\"Model class {model_cls.__name__!r} defines a `__dispatch_key__` \"\n            \"and a type field. Only one of these may be defined for dispatch.\"\n        )\n\n    cls_init = model_cls.__init__\n    cls_new = model_cls.__new__\n\n    def __init__(__pydantic_self__, **data: Any) -> None:\n        type_string = (\n            get_dispatch_key(__pydantic_self__)\n            if type(__pydantic_self__) != model_cls\n            else \"__base__\"\n        )\n        data.setdefault(\"type\", type_string)\n        cls_init(__pydantic_self__, **data)\n\n    def __new__(cls: Type[Self], **kwargs) -> Self:\n        if \"type\" in kwargs:\n            try:\n                subcls = lookup_type(cls, dispatch_key=kwargs[\"type\"])\n            except KeyError as exc:\n                raise pydantic.ValidationError(errors=[exc], model=cls)\n            return cls_new(subcls)\n        else:\n            return cls_new(cls)\n\n    model_cls.__init__ = __init__\n    model_cls.__new__ = __new__\n\n    register_base_type(model_cls)\n\n    return model_cls\n
    ","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/pydantic/#prefect.utilities.pydantic.get_class_fields_only","title":"get_class_fields_only","text":"

    Gets all the field names defined on the model class but not any parent classes. Any fields that are on the parent but redefined on the subclass are included.

    Source code in prefect/utilities/pydantic.py
    def get_class_fields_only(model: Type[pydantic.BaseModel]) -> set:\n    \"\"\"\n    Gets all the field names defined on the model class but not any parent classes.\n    Any fields that are on the parent but redefined on the subclass are included.\n    \"\"\"\n    subclass_class_fields = set(model.__annotations__.keys())\n    parent_class_fields = set()\n\n    for base in model.__class__.__bases__:\n        if issubclass(base, pydantic.BaseModel):\n            parent_class_fields.update(base.__annotations__.keys())\n\n    return (subclass_class_fields - parent_class_fields) | (\n        subclass_class_fields & parent_class_fields\n    )\n
    ","tags":["Python API","pydantic"]},{"location":"api-ref/prefect/utilities/render_swagger/","title":"render_swagger","text":"","tags":["Python API","Swagger"]},{"location":"api-ref/prefect/utilities/render_swagger/#prefect.utilities.render_swagger","title":"prefect.utilities.render_swagger","text":"","tags":["Python API","Swagger"]},{"location":"api-ref/prefect/utilities/render_swagger/#prefect.utilities.render_swagger.swagger_lib","title":"swagger_lib","text":"

    Provides the actual swagger library used

    Source code in prefect/utilities/render_swagger.py
    def swagger_lib(config) -> dict:\n    \"\"\"\n    Provides the actual swagger library used\n    \"\"\"\n    lib_swagger = {\n        \"css\": \"https://unpkg.com/swagger-ui-dist@3/swagger-ui.css\",\n        \"js\": \"https://unpkg.com/swagger-ui-dist@3/swagger-ui-bundle.js\",\n    }\n\n    extra_javascript = config.get(\"extra_javascript\", [])\n    extra_css = config.get(\"extra_css\", [])\n    for lib in extra_javascript:\n        if os.path.basename(urllib.parse.urlparse(lib).path) == \"swagger-ui-bundle.js\":\n            lib_swagger[\"js\"] = lib\n            break\n\n    for css in extra_css:\n        if os.path.basename(urllib.parse.urlparse(css).path) == \"swagger-ui.css\":\n            lib_swagger[\"css\"] = css\n            break\n    return lib_swagger\n
    ","tags":["Python API","Swagger"]},{"location":"api-ref/prefect/utilities/services/","title":"services","text":"","tags":["Python API","services"]},{"location":"api-ref/prefect/utilities/services/#prefect.utilities.services","title":"prefect.utilities.services","text":"","tags":["Python API","services"]},{"location":"api-ref/prefect/utilities/services/#prefect.utilities.services.critical_service_loop","title":"critical_service_loop async","text":"

    Runs the given workload function on the specified interval, while being forgiving of intermittent issues like temporary HTTP errors. If more than a certain number of consecutive errors occur, print a summary of up to memory recent exceptions to printer, then begin backoff.

    The loop will exit after reaching the consecutive error limit backoff times. On each backoff, the interval will be doubled. On a successful loop, the backoff will be reset.

    Parameters:

    Name Type Description Default workload Callable[..., Coroutine]

    the function to call

    required interval float

    how frequently to call it

    required memory int

    how many recent errors to remember

    10 consecutive int

    how many consecutive errors must we see before we begin backoff

    3 backoff int

    how many times we should allow consecutive errors before exiting

    1 printer Callable[..., None]

    a print-like function where errors will be reported

    print run_once bool

    if set, the loop will only run once then return

    False jitter_range float

    if set, the interval will be a random variable (rv) drawn from a clamped Poisson distribution where lambda = interval and the rv is bound between interval * (1 - range) < rv < interval * (1 + range)

    None Source code in prefect/utilities/services.py
    async def critical_service_loop(\n    workload: Callable[..., Coroutine],\n    interval: float,\n    memory: int = 10,\n    consecutive: int = 3,\n    backoff: int = 1,\n    printer: Callable[..., None] = print,\n    run_once: bool = False,\n    jitter_range: float = None,\n):\n    \"\"\"\n    Runs the given `workload` function on the specified `interval`, while being\n    forgiving of intermittent issues like temporary HTTP errors.  If more than a certain\n    number of `consecutive` errors occur, print a summary of up to `memory` recent\n    exceptions to `printer`, then begin backoff.\n\n    The loop will exit after reaching the consecutive error limit `backoff` times.\n    On each backoff, the interval will be doubled. On a successful loop, the backoff\n    will be reset.\n\n    Args:\n        workload: the function to call\n        interval: how frequently to call it\n        memory: how many recent errors to remember\n        consecutive: how many consecutive errors must we see before we begin backoff\n        backoff: how many times we should allow consecutive errors before exiting\n        printer: a `print`-like function where errors will be reported\n        run_once: if set, the loop will only run once then return\n        jitter_range: if set, the interval will be a random variable (rv) drawn from\n            a clamped Poisson distribution where lambda = interval and the rv is bound\n            between `interval * (1 - range) < rv < interval * (1 + range)`\n    \"\"\"\n\n    track_record: Deque[bool] = deque([True] * consecutive, maxlen=consecutive)\n    failures: Deque[Tuple[Exception, TracebackType]] = deque(maxlen=memory)\n    backoff_count = 0\n\n    while True:\n        try:\n            logger.debug(f\"Starting run of {workload!r}\")\n            await workload()\n\n            # Reset the backoff count on success; we may want to consider resetting\n            # this only if the track record is _all_ successful to avoid ending backoff\n            # prematurely\n            if backoff_count > 0:\n                printer(\"Resetting backoff due to successful run.\")\n                backoff_count = 0\n\n            track_record.append(True)\n        except httpx.TransportError as exc:\n            # httpx.TransportError is the base class for any kind of communications\n            # error, like timeouts, connection failures, etc.  This does _not_ cover\n            # routine HTTP error codes (even 5xx errors like 502/503) so this\n            # handler should not be attempting to cover cases where the Prefect server\n            # or Prefect Cloud is having an outage (which will be covered by the\n            # exception clause below)\n            track_record.append(False)\n            failures.append((exc, sys.exc_info()[-1]))\n            logger.debug(\n                f\"Run of {workload!r} failed with TransportError\", exc_info=exc\n            )\n        except httpx.HTTPStatusError as exc:\n            if exc.response.status_code >= 500:\n                # 5XX codes indicate a potential outage of the Prefect API which is\n                # likely to be temporary and transient.  Don't quit over these unless\n                # it is prolonged.\n                track_record.append(False)\n                failures.append((exc, sys.exc_info()[-1]))\n                logger.debug(\n                    f\"Run of {workload!r} failed with HTTPStatusError\", exc_info=exc\n                )\n            else:\n                raise\n\n        # Decide whether to exit now based on recent history.\n        #\n        # Given some typical background error rate of, say, 1%, we may still observe\n        # quite a few errors in our recent samples, but this is not necessarily a cause\n        # for concern.\n        #\n        # Imagine two distributions that could reflect our situation at any time: the\n        # everything-is-fine distribution of errors, and the everything-is-on-fire\n        # distribution of errors. We are trying to determine which of those two worlds\n        # we are currently experiencing.  We compare the likelihood that we'd draw N\n        # consecutive errors from each.  In the everything-is-fine distribution, that\n        # would be a very low-probability occurrence, but in the everything-is-on-fire\n        # distribution, that is a high-probability occurrence.\n        #\n        # Remarkably, we only need to look back for a small number of consecutive\n        # errors to have reasonable confidence that this is indeed an anomaly.\n        # @anticorrelator and @chrisguidry estimated that we should only need to look\n        # back for 3 consecutive errors.\n        if not any(track_record):\n            # We've failed enough times to be sure something is wrong, the writing is\n            # on the wall.  Let's explain what we've seen and exit.\n            printer(\n                f\"\\nFailed the last {consecutive} attempts. \"\n                \"Please check your environment and configuration.\"\n            )\n\n            printer(\"Examples of recent errors:\\n\")\n\n            failures_by_type = distinct(\n                reversed(failures),\n                key=lambda pair: type(pair[0]),  # Group by the type of exception\n            )\n            for exception, traceback in failures_by_type:\n                printer(\"\".join(format_exception(None, exception, traceback)))\n                printer()\n\n            backoff_count += 1\n\n            if backoff_count >= backoff:\n                raise RuntimeError(\"Service exceeded error threshold.\")\n\n            # Reset the track record\n            track_record.extend([True] * consecutive)\n            failures.clear()\n            printer(\n                \"Backing off due to consecutive errors, using increased interval of \"\n                f\" {interval * 2**backoff_count}s.\"\n            )\n\n        if run_once:\n            return\n\n        if jitter_range is not None:\n            sleep = clamped_poisson_interval(interval, clamping_factor=jitter_range)\n        else:\n            sleep = interval * 2**backoff_count\n\n        await anyio.sleep(sleep)\n
    ","tags":["Python API","services"]},{"location":"api-ref/prefect/utilities/slugify/","title":"slugify","text":"","tags":["Python API","slugify"]},{"location":"api-ref/prefect/utilities/slugify/#prefect.utilities.slugify","title":"prefect.utilities.slugify","text":"","tags":["Python API","slugify"]},{"location":"api-ref/prefect/utilities/templating/","title":"templating","text":"","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating","title":"prefect.utilities.templating","text":"","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.apply_values","title":"apply_values","text":"

    Replaces placeholders in a template with values from a supplied dictionary.

    Will recursively replace placeholders in dictionaries and lists.

    If a value has no placeholders, it will be returned unchanged.

    If a template contains only a single placeholder, the placeholder will be fully replaced with the value.

    If a template contains text before or after a placeholder or there are multiple placeholders, the placeholders will be replaced with the corresponding variable values.

    If a template contains a placeholder that is not in values, NotSet will be returned to signify that no placeholder replacement occurred. If template is a dictionary that contains a key with a value of NotSet, the key will be removed in the return value unless remove_notset is set to False.

    Parameters:

    Name Type Description Default template T

    template to discover and replace values in

    required values Dict[str, Any]

    The values to apply to placeholders in the template

    required remove_notset bool

    If True, remove keys with an unset value

    True

    Returns:

    Type Description Union[T, Type[NotSet]]

    The template with the values applied

    Source code in prefect/utilities/templating.py
    def apply_values(\n    template: T, values: Dict[str, Any], remove_notset: bool = True\n) -> Union[T, Type[NotSet]]:\n    \"\"\"\n    Replaces placeholders in a template with values from a supplied dictionary.\n\n    Will recursively replace placeholders in dictionaries and lists.\n\n    If a value has no placeholders, it will be returned unchanged.\n\n    If a template contains only a single placeholder, the placeholder will be\n    fully replaced with the value.\n\n    If a template contains text before or after a placeholder or there are\n    multiple placeholders, the placeholders will be replaced with the\n    corresponding variable values.\n\n    If a template contains a placeholder that is not in `values`, NotSet will\n    be returned to signify that no placeholder replacement occurred. If\n    `template` is a dictionary that contains a key with a value of NotSet,\n    the key will be removed in the return value unless `remove_notset` is set to False.\n\n    Args:\n        template: template to discover and replace values in\n        values: The values to apply to placeholders in the template\n        remove_notset: If True, remove keys with an unset value\n\n    Returns:\n        The template with the values applied\n    \"\"\"\n    if isinstance(template, (int, float, bool, type(NotSet), type(None))):\n        return template\n    if isinstance(template, str):\n        placeholders = find_placeholders(template)\n        if not placeholders:\n            # If there are no values, we can just use the template\n            return template\n        elif (\n            len(placeholders) == 1\n            and list(placeholders)[0].full_match == template\n            and list(placeholders)[0].type is PlaceholderType.STANDARD\n        ):\n            # If there is only one variable with no surrounding text,\n            # we can replace it. If there is no variable value, we\n            # return NotSet to indicate that the value should not be included.\n            return get_from_dict(values, list(placeholders)[0].name, NotSet)\n        else:\n            for full_match, name, placeholder_type in placeholders:\n                if placeholder_type is PlaceholderType.STANDARD:\n                    value = get_from_dict(values, name, NotSet)\n                elif placeholder_type is PlaceholderType.ENV_VAR:\n                    name = name.lstrip(ENV_VAR_PLACEHOLDER_PREFIX)\n                    value = os.environ.get(name, NotSet)\n                else:\n                    continue\n\n                if value is NotSet and not remove_notset:\n                    continue\n                elif value is NotSet:\n                    template = template.replace(full_match, \"\")\n                else:\n                    template = template.replace(full_match, str(value))\n\n            return template\n    elif isinstance(template, dict):\n        updated_template = {}\n        for key, value in template.items():\n            updated_value = apply_values(value, values, remove_notset=remove_notset)\n            if updated_value is not NotSet:\n                updated_template[key] = updated_value\n            elif not remove_notset:\n                updated_template[key] = value\n\n        return updated_template\n    elif isinstance(template, list):\n        updated_list = []\n        for value in template:\n            updated_value = apply_values(value, values, remove_notset=remove_notset)\n            if updated_value is not NotSet:\n                updated_list.append(updated_value)\n        return updated_list\n    else:\n        raise ValueError(f\"Unexpected template type {type(template).__name__!r}\")\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.determine_placeholder_type","title":"determine_placeholder_type","text":"

    Determines the type of a placeholder based on its name.

    Parameters:

    Name Type Description Default name str

    The name of the placeholder

    required

    Returns:

    Type Description PlaceholderType

    The type of the placeholder

    Source code in prefect/utilities/templating.py
    def determine_placeholder_type(name: str) -> PlaceholderType:\n    \"\"\"\n    Determines the type of a placeholder based on its name.\n\n    Args:\n        name: The name of the placeholder\n\n    Returns:\n        The type of the placeholder\n    \"\"\"\n    if name.startswith(BLOCK_DOCUMENT_PLACEHOLDER_PREFIX):\n        return PlaceholderType.BLOCK_DOCUMENT\n    elif name.startswith(VARIABLE_PLACEHOLDER_PREFIX):\n        return PlaceholderType.VARIABLE\n    elif name.startswith(ENV_VAR_PLACEHOLDER_PREFIX):\n        return PlaceholderType.ENV_VAR\n    else:\n        return PlaceholderType.STANDARD\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.find_placeholders","title":"find_placeholders","text":"

    Finds all placeholders in a template.

    Parameters:

    Name Type Description Default template T

    template to discover placeholders in

    required

    Returns:

    Type Description Set[Placeholder]

    A set of all placeholders in the template

    Source code in prefect/utilities/templating.py
    def find_placeholders(template: T) -> Set[Placeholder]:\n    \"\"\"\n    Finds all placeholders in a template.\n\n    Args:\n        template: template to discover placeholders in\n\n    Returns:\n        A set of all placeholders in the template\n    \"\"\"\n    if isinstance(template, (int, float, bool)):\n        return set()\n    if isinstance(template, str):\n        result = PLACEHOLDER_CAPTURE_REGEX.findall(template)\n        return {\n            Placeholder(full_match, name, determine_placeholder_type(name))\n            for full_match, name in result\n        }\n    elif isinstance(template, dict):\n        return set().union(\n            *[find_placeholders(value) for key, value in template.items()]\n        )\n    elif isinstance(template, list):\n        return set().union(*[find_placeholders(item) for item in template])\n    else:\n        raise ValueError(f\"Unexpected type: {type(template)}\")\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.resolve_block_document_references","title":"resolve_block_document_references async","text":"

    Resolve block document references in a template by replacing each reference with the data of the block document.

    Recursively searches for block document references in dictionaries and lists.

    Identifies block document references by the as dictionary with the following structure:

    {\n    \"$ref\": {\n        \"block_document_id\": <block_document_id>\n    }\n}\n
    where <block_document_id> is the ID of the block document to resolve.

    Once the block document is retrieved from the API, the data of the block document is used to replace the reference.

    Parameters:

    Name Type Description Default template T

    The template to resolve block documents in

    required

    Returns:

    Type Description Union[T, Dict[str, Any]]

    The template with block documents resolved

    Source code in prefect/utilities/templating.py
    @inject_client\nasync def resolve_block_document_references(\n    template: T, client: \"PrefectClient\" = None\n) -> Union[T, Dict[str, Any]]:\n    \"\"\"\n    Resolve block document references in a template by replacing each reference with\n    the data of the block document.\n\n    Recursively searches for block document references in dictionaries and lists.\n\n    Identifies block document references by the as dictionary with the following\n    structure:\n    ```\n    {\n        \"$ref\": {\n            \"block_document_id\": <block_document_id>\n        }\n    }\n    ```\n    where `<block_document_id>` is the ID of the block document to resolve.\n\n    Once the block document is retrieved from the API, the data of the block document\n    is used to replace the reference.\n\n    Args:\n        template: The template to resolve block documents in\n\n    Returns:\n        The template with block documents resolved\n    \"\"\"\n    if isinstance(template, dict):\n        block_document_id = template.get(\"$ref\", {}).get(\"block_document_id\")\n        if block_document_id:\n            block_document = await client.read_block_document(block_document_id)\n            return block_document.data\n        updated_template = {}\n        for key, value in template.items():\n            updated_value = await resolve_block_document_references(\n                value, client=client\n            )\n            updated_template[key] = updated_value\n        return updated_template\n    elif isinstance(template, list):\n        return [\n            await resolve_block_document_references(item, client=client)\n            for item in template\n        ]\n    elif isinstance(template, str):\n        placeholders = find_placeholders(template)\n        has_block_document_placeholder = any(\n            placeholder.type is PlaceholderType.BLOCK_DOCUMENT\n            for placeholder in placeholders\n        )\n        if len(placeholders) == 0 or not has_block_document_placeholder:\n            return template\n        elif (\n            len(placeholders) == 1\n            and list(placeholders)[0].full_match == template\n            and list(placeholders)[0].type is PlaceholderType.BLOCK_DOCUMENT\n        ):\n            block_type_slug, block_document_name = (\n                list(placeholders)[0]\n                .name.replace(BLOCK_DOCUMENT_PLACEHOLDER_PREFIX, \"\")\n                .split(\".\")\n            )\n            block_document = await client.read_block_document_by_name(\n                name=block_document_name, block_type_slug=block_type_slug\n            )\n            # Handling for system blocks like Secret that have a value field\n            # These blocks will be replaced by variables in the future and this\n            # logic can be removed at that time.\n            value = block_document.data.get(\"value\", block_document.data)\n            return value\n        else:\n            raise ValueError(\n                f\"Invalid template: {template!r}. Only a single block placeholder is\"\n                \" allowed in a string and no surrounding text is allowed.\"\n            )\n\n    return template\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/templating/#prefect.utilities.templating.resolve_variables","title":"resolve_variables async","text":"

    Resolve variables in a template by replacing each variable placeholder with the value of the variable.

    Recursively searches for variable placeholders in dictionaries and lists.

    Strips variable placeholders if the variable is not found.

    Parameters:

    Name Type Description Default template T

    The template to resolve variables in

    required

    Returns:

    Type Description

    The template with variables resolved

    Source code in prefect/utilities/templating.py
    @inject_client\nasync def resolve_variables(template: T, client: \"PrefectClient\" = None):\n    \"\"\"\n    Resolve variables in a template by replacing each variable placeholder with the\n    value of the variable.\n\n    Recursively searches for variable placeholders in dictionaries and lists.\n\n    Strips variable placeholders if the variable is not found.\n\n    Args:\n        template: The template to resolve variables in\n\n    Returns:\n        The template with variables resolved\n    \"\"\"\n    if isinstance(template, str):\n        placeholders = find_placeholders(template)\n        has_variable_placeholder = any(\n            placeholder.type is PlaceholderType.VARIABLE for placeholder in placeholders\n        )\n        if not placeholders or not has_variable_placeholder:\n            # If there are no values, we can just use the template\n            return template\n        elif (\n            len(placeholders) == 1\n            and list(placeholders)[0].full_match == template\n            and list(placeholders)[0].type is PlaceholderType.VARIABLE\n        ):\n            variable_name = list(placeholders)[0].name.replace(\n                VARIABLE_PLACEHOLDER_PREFIX, \"\"\n            )\n            variable = await client.read_variable_by_name(name=variable_name)\n            if variable is None:\n                return \"\"\n            else:\n                return variable.value\n        else:\n            for full_match, name, placeholder_type in placeholders:\n                if placeholder_type is PlaceholderType.VARIABLE:\n                    variable_name = name.replace(VARIABLE_PLACEHOLDER_PREFIX, \"\")\n                    variable = await client.read_variable_by_name(name=variable_name)\n                    if variable is None:\n                        template = template.replace(full_match, \"\")\n                    else:\n                        template = template.replace(full_match, variable.value)\n            return template\n    elif isinstance(template, dict):\n        return {\n            key: await resolve_variables(value, client=client)\n            for key, value in template.items()\n        }\n    elif isinstance(template, list):\n        return [await resolve_variables(item, client=client) for item in template]\n    else:\n        return template\n
    ","tags":["Python API","templating"]},{"location":"api-ref/prefect/utilities/text/","title":"text","text":"","tags":["Python API","text"]},{"location":"api-ref/prefect/utilities/text/#prefect.utilities.text","title":"prefect.utilities.text","text":"","tags":["Python API","text"]},{"location":"api-ref/prefect/utilities/validation/","title":"validation","text":"","tags":["Python API","validation"]},{"location":"api-ref/prefect/utilities/validation/#prefect.utilities.validation","title":"prefect.utilities.validation","text":"","tags":["Python API","validation"]},{"location":"api-ref/prefect/utilities/validation/#prefect.utilities.validation.validate_schema","title":"validate_schema","text":"

    Validate that the provided schema is a valid json schema.

    Parameters:

    Name Type Description Default schema dict

    The schema to validate.

    required

    Raises:

    Type Description ValueError

    If the provided schema is not a valid json schema.

    Source code in prefect/utilities/validation.py
    def validate_schema(schema: dict):\n    \"\"\"\n    Validate that the provided schema is a valid json schema.\n\n    Args:\n        schema: The schema to validate.\n\n    Raises:\n        ValueError: If the provided schema is not a valid json schema.\n\n    \"\"\"\n    try:\n        if schema is not None:\n            # Most closely matches the schemas generated by pydantic\n            jsonschema.Draft4Validator.check_schema(schema)\n    except jsonschema.SchemaError as exc:\n        raise ValueError(\n            \"The provided schema is not a valid json schema. Schema error:\"\n            f\" {exc.message}\"\n        ) from exc\n
    ","tags":["Python API","validation"]},{"location":"api-ref/prefect/utilities/validation/#prefect.utilities.validation.validate_values_conform_to_schema","title":"validate_values_conform_to_schema","text":"

    Validate that the provided values conform to the provided json schema.

    Parameters:

    Name Type Description Default values dict

    The values to validate.

    required schema dict

    The schema to validate against.

    required ignore_required bool

    Whether to ignore the required fields in the schema. Should be used when a partial set of values is acceptable.

    False

    Raises:

    Type Description ValueError

    If the parameters do not conform to the schema.

    Source code in prefect/utilities/validation.py
    def validate_values_conform_to_schema(\n    values: dict, schema: dict, ignore_required: bool = False\n):\n    \"\"\"\n    Validate that the provided values conform to the provided json schema.\n\n    Args:\n        values: The values to validate.\n        schema: The schema to validate against.\n        ignore_required: Whether to ignore the required fields in the schema. Should be\n            used when a partial set of values is acceptable.\n\n    Raises:\n        ValueError: If the parameters do not conform to the schema.\n\n    \"\"\"\n    if ignore_required:\n        schema = remove_nested_keys([\"required\"], schema)\n\n    try:\n        if schema is not None and values is not None:\n            jsonschema.validate(values, schema)\n    except jsonschema.ValidationError as exc:\n        if exc.json_path == \"$\":\n            error_message = \"Validation failed.\"\n        else:\n            error_message = (\n                f\"Validation failed for field {exc.json_path.replace('$.', '')!r}.\"\n            )\n        error_message += f\" Failure reason: {exc.message}\"\n        raise ValueError(error_message) from exc\n    except jsonschema.SchemaError as exc:\n        raise ValueError(\n            \"The provided schema is not a valid json schema. Schema error:\"\n            f\" {exc.message}\"\n        ) from exc\n
    ","tags":["Python API","validation"]},{"location":"api-ref/prefect/utilities/visualization/","title":"visualization","text":"","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization","title":"prefect.utilities.visualization","text":"

    Utilities for working with Flow.visualize()

    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.TaskVizTracker","title":"TaskVizTracker","text":"Source code in prefect/utilities/visualization.py
    class TaskVizTracker:\n    def __init__(self):\n        self.tasks = []\n        self.dynamic_task_counter = {}\n        self.object_id_to_task = {}\n\n    def add_task(self, task: VizTask):\n        if task.name not in self.dynamic_task_counter:\n            self.dynamic_task_counter[task.name] = 0\n        else:\n            self.dynamic_task_counter[task.name] += 1\n\n        task.name = f\"{task.name}-{self.dynamic_task_counter[task.name]}\"\n        self.tasks.append(task)\n\n    def __enter__(self):\n        TaskVizTrackerState.current = self\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        TaskVizTrackerState.current = None\n\n    def link_viz_return_value_to_viz_task(\n        self, viz_return_value: Any, viz_task: VizTask\n    ) -> None:\n        \"\"\"\n        We cannot track booleans, Ellipsis, None, NotImplemented, or the integers from -5 to 256\n        because they are singletons.\n        \"\"\"\n        from prefect.engine import UNTRACKABLE_TYPES\n\n        if (type(viz_return_value) in UNTRACKABLE_TYPES) or (\n            isinstance(viz_return_value, int) and (-5 <= viz_return_value <= 256)\n        ):\n            return\n        self.object_id_to_task[id(viz_return_value)] = viz_task\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.TaskVizTracker.link_viz_return_value_to_viz_task","title":"link_viz_return_value_to_viz_task","text":"

    We cannot track booleans, Ellipsis, None, NotImplemented, or the integers from -5 to 256 because they are singletons.

    Source code in prefect/utilities/visualization.py
    def link_viz_return_value_to_viz_task(\n    self, viz_return_value: Any, viz_task: VizTask\n) -> None:\n    \"\"\"\n    We cannot track booleans, Ellipsis, None, NotImplemented, or the integers from -5 to 256\n    because they are singletons.\n    \"\"\"\n    from prefect.engine import UNTRACKABLE_TYPES\n\n    if (type(viz_return_value) in UNTRACKABLE_TYPES) or (\n        isinstance(viz_return_value, int) and (-5 <= viz_return_value <= 256)\n    ):\n        return\n    self.object_id_to_task[id(viz_return_value)] = viz_task\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.build_task_dependencies","title":"build_task_dependencies","text":"

    Constructs a Graphviz directed graph object that represents the dependencies between tasks in the given TaskVizTracker.

    • task_run_tracker (TaskVizTracker): An object containing tasks and their dependencies.
    • graphviz.Digraph: A directed graph object depicting the relationships and dependencies between tasks.

    Raises: - GraphvizImportError: If there's an ImportError related to graphviz. - FlowVisualizationError: If there's any other error during the visualization process or if return values of tasks are directly accessed without specifying a viz_return_value.

    Source code in prefect/utilities/visualization.py
    def build_task_dependencies(task_run_tracker: TaskVizTracker):\n    \"\"\"\n    Constructs a Graphviz directed graph object that represents the dependencies\n    between tasks in the given TaskVizTracker.\n\n    Parameters:\n    - task_run_tracker (TaskVizTracker): An object containing tasks and their\n      dependencies.\n\n    Returns:\n    - graphviz.Digraph: A directed graph object depicting the relationships and\n      dependencies between tasks.\n\n    Raises:\n    - GraphvizImportError: If there's an ImportError related to graphviz.\n    - FlowVisualizationError: If there's any other error during the visualization\n      process or if return values of tasks are directly accessed without\n      specifying a `viz_return_value`.\n    \"\"\"\n    try:\n        g = graphviz.Digraph()\n        for task in task_run_tracker.tasks:\n            g.node(task.name)\n            for upstream in task.upstream_tasks:\n                g.edge(upstream.name, task.name)\n        return g\n    except ImportError as exc:\n        raise GraphvizImportError from exc\n    except Exception:\n        raise FlowVisualizationError(\n            \"Something went wrong building the flow's visualization.\"\n            \" If you're interacting with the return value of a task\"\n            \" directly inside of your flow, you must set a set a `viz_return_value`\"\n            \", for example `@task(viz_return_value=[1, 2, 3])`.\"\n        )\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.track_viz_task","title":"track_viz_task","text":"

    Return a result if sync otherwise return a coroutine that returns the result

    Source code in prefect/utilities/visualization.py
    def track_viz_task(\n    is_async: bool,\n    task_name: str,\n    parameters: dict,\n    viz_return_value: Optional[Any] = None,\n):\n    \"\"\"Return a result if sync otherwise return a coroutine that returns the result\"\"\"\n    if is_async:\n        return from_async.wait_for_call_in_loop_thread(\n            partial(_track_viz_task, task_name, parameters, viz_return_value)\n        )\n    else:\n        return _track_viz_task(task_name, parameters, viz_return_value)\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/utilities/visualization/#prefect.utilities.visualization.visualize_task_dependencies","title":"visualize_task_dependencies","text":"

    Renders and displays a Graphviz directed graph representing task dependencies.

    The graph is rendered in PNG format and saved with the name specified by flow_run_name. After rendering, the visualization is opened and displayed.

    Parameters: - graph (graphviz.Digraph): The directed graph object to visualize. - flow_run_name (str): The name to use when saving the rendered graph image.

    Raises: - GraphvizExecutableNotFoundError: If Graphviz isn't found on the system. - FlowVisualizationError: If there's any other error during the visualization process or if return values of tasks are directly accessed without specifying a viz_return_value.

    Source code in prefect/utilities/visualization.py
    def visualize_task_dependencies(graph: graphviz.Digraph, flow_run_name: str):\n    \"\"\"\n    Renders and displays a Graphviz directed graph representing task dependencies.\n\n    The graph is rendered in PNG format and saved with the name specified by\n    flow_run_name. After rendering, the visualization is opened and displayed.\n\n    Parameters:\n    - graph (graphviz.Digraph): The directed graph object to visualize.\n    - flow_run_name (str): The name to use when saving the rendered graph image.\n\n    Raises:\n    - GraphvizExecutableNotFoundError: If Graphviz isn't found on the system.\n    - FlowVisualizationError: If there's any other error during the visualization\n      process or if return values of tasks are directly accessed without\n      specifying a `viz_return_value`.\n    \"\"\"\n    try:\n        graph.render(filename=flow_run_name, view=True, format=\"png\", cleanup=True)\n    except graphviz.backend.ExecutableNotFound as exc:\n        msg = (\n            \"It appears you do not have Graphviz installed, or it is not on your \"\n            \"PATH. Please install Graphviz from http://www.graphviz.org/download/. \"\n            \"Note: Just installing the `graphviz` python package is not \"\n            \"sufficient.\"\n        )\n        raise GraphvizExecutableNotFoundError(msg) from exc\n    except Exception:\n        raise FlowVisualizationError(\n            \"Something went wrong building the flow's visualization.\"\n            \" If you're interacting with the return value of a task\"\n            \" directly inside of your flow, you must set a set a `viz_return_value`\"\n            \", for example `@task(viz_return_value=[1, 2, 3])`.\"\n        )\n
    ","tags":["Python API","visualization"]},{"location":"api-ref/prefect/workers/base/","title":"base","text":"","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base","title":"prefect.workers.base","text":"","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseJobConfiguration","title":"BaseJobConfiguration","text":"

    Bases: BaseModel

    Source code in prefect/workers/base.py
    class BaseJobConfiguration(BaseModel):\n    command: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The command to use when starting a flow run. \"\n            \"In most cases, this should be left blank and the command \"\n            \"will be automatically generated by the worker.\"\n        ),\n    )\n    env: Dict[str, Optional[str]] = Field(\n        default_factory=dict,\n        title=\"Environment Variables\",\n        description=\"Environment variables to set when starting a flow run.\",\n    )\n    labels: Dict[str, str] = Field(\n        default_factory=dict,\n        description=(\n            \"Labels applied to infrastructure created by the worker using \"\n            \"this job configuration.\"\n        ),\n    )\n    name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"Name given to infrastructure created by the worker using this \"\n            \"job configuration.\"\n        ),\n    )\n\n    _related_objects: Dict[str, Any] = PrivateAttr(default_factory=dict)\n\n    @property\n    def is_using_a_runner(self):\n        return self.command is not None and \"prefect flow-run execute\" in self.command\n\n    @validator(\"command\")\n    def _coerce_command(cls, v):\n        \"\"\"Make sure that empty strings are treated as None\"\"\"\n        if not v:\n            return None\n        return v\n\n    @staticmethod\n    def _get_base_config_defaults(variables: dict) -> dict:\n        \"\"\"Get default values from base config for all variables that have them.\"\"\"\n        defaults = dict()\n        for variable_name, attrs in variables.items():\n            if \"default\" in attrs:\n                defaults[variable_name] = attrs[\"default\"]\n\n        return defaults\n\n    @classmethod\n    @inject_client\n    async def from_template_and_values(\n        cls, base_job_template: dict, values: dict, client: \"PrefectClient\" = None\n    ):\n        \"\"\"Creates a valid worker configuration object from the provided base\n        configuration and overrides.\n\n        Important: this method expects that the base_job_template was already\n        validated server-side.\n        \"\"\"\n        job_config: Dict[str, Any] = base_job_template[\"job_configuration\"]\n        variables_schema = base_job_template[\"variables\"]\n        variables = cls._get_base_config_defaults(\n            variables_schema.get(\"properties\", {})\n        )\n        variables.update(values)\n\n        populated_configuration = apply_values(template=job_config, values=variables)\n        populated_configuration = await resolve_block_document_references(\n            template=populated_configuration, client=client\n        )\n        populated_configuration = await resolve_variables(\n            template=populated_configuration, client=client\n        )\n        return cls(**populated_configuration)\n\n    @classmethod\n    def json_template(cls) -> dict:\n        \"\"\"Returns a dict with job configuration as keys and the corresponding templates as values\n\n        Defaults to using the job configuration parameter name as the template variable name.\n\n        e.g.\n        {\n            key1: '{{ key1 }}',     # default variable template\n            key2: '{{ template2 }}', # `template2` specifically provide as template\n        }\n        \"\"\"\n        configuration = {}\n        properties = cls.schema()[\"properties\"]\n        for k, v in properties.items():\n            if v.get(\"template\"):\n                template = v[\"template\"]\n            else:\n                template = \"{{ \" + k + \" }}\"\n            configuration[k] = template\n\n        return configuration\n\n    def prepare_for_flow_run(\n        self,\n        flow_run: \"FlowRun\",\n        deployment: Optional[\"DeploymentResponse\"] = None,\n        flow: Optional[\"Flow\"] = None,\n    ):\n        \"\"\"\n        Prepare the job configuration for a flow run.\n\n        This method is called by the worker before starting a flow run. It\n        should be used to set any configuration values that are dependent on\n        the flow run.\n\n        Args:\n            flow_run: The flow run to be executed.\n            deployment: The deployment that the flow run is associated with.\n            flow: The flow that the flow run is associated with.\n        \"\"\"\n\n        self._related_objects = {\n            \"deployment\": deployment,\n            \"flow\": flow,\n            \"flow-run\": flow_run,\n        }\n        if deployment is not None:\n            deployment_labels = self._base_deployment_labels(deployment)\n        else:\n            deployment_labels = {}\n\n        if flow is not None:\n            flow_labels = self._base_flow_labels(flow)\n        else:\n            flow_labels = {}\n\n        env = {\n            **self._base_environment(),\n            **self._base_flow_run_environment(flow_run),\n            **self.env,\n        }\n        self.env = {key: value for key, value in env.items() if value is not None}\n        self.labels = {\n            **self._base_flow_run_labels(flow_run),\n            **deployment_labels,\n            **flow_labels,\n            **self.labels,\n        }\n        self.name = self.name or flow_run.name\n        self.command = self.command or self._base_flow_run_command()\n\n    @staticmethod\n    def _base_flow_run_command() -> str:\n        \"\"\"\n        Generate a command for a flow run job.\n        \"\"\"\n        if experiment_enabled(\"enhanced_cancellation\"):\n            if (\n                PREFECT_EXPERIMENTAL_WARN\n                and PREFECT_EXPERIMENTAL_WARN_ENHANCED_CANCELLATION\n            ):\n                warnings.warn(\n                    EXPERIMENTAL_WARNING.format(\n                        feature=\"Enhanced flow run cancellation\",\n                        group=\"enhanced_cancellation\",\n                        help=\"\",\n                    ),\n                    ExperimentalFeature,\n                    stacklevel=3,\n                )\n            return \"prefect flow-run execute\"\n        return \"python -m prefect.engine\"\n\n    @staticmethod\n    def _base_flow_run_labels(flow_run: \"FlowRun\") -> Dict[str, str]:\n        \"\"\"\n        Generate a dictionary of labels for a flow run job.\n        \"\"\"\n        return {\n            \"prefect.io/flow-run-id\": str(flow_run.id),\n            \"prefect.io/flow-run-name\": flow_run.name,\n            \"prefect.io/version\": prefect.__version__,\n        }\n\n    @classmethod\n    def _base_environment(cls) -> Dict[str, str]:\n        \"\"\"\n        Environment variables that should be passed to all created infrastructure.\n\n        These values should be overridable with the `env` field.\n        \"\"\"\n        return get_current_settings().to_environment_variables(exclude_unset=True)\n\n    @staticmethod\n    def _base_flow_run_environment(flow_run: \"FlowRun\") -> Dict[str, str]:\n        \"\"\"\n        Generate a dictionary of environment variables for a flow run job.\n        \"\"\"\n        return {\"PREFECT__FLOW_RUN_ID\": str(flow_run.id)}\n\n    @staticmethod\n    def _base_deployment_labels(deployment: \"DeploymentResponse\") -> Dict[str, str]:\n        labels = {\n            \"prefect.io/deployment-id\": str(deployment.id),\n            \"prefect.io/deployment-name\": deployment.name,\n        }\n        if deployment.updated is not None:\n            labels[\"prefect.io/deployment-updated\"] = deployment.updated.in_timezone(\n                \"utc\"\n            ).to_iso8601_string()\n        return labels\n\n    @staticmethod\n    def _base_flow_labels(flow: \"Flow\") -> Dict[str, str]:\n        return {\n            \"prefect.io/flow-id\": str(flow.id),\n            \"prefect.io/flow-name\": flow.name,\n        }\n\n    def _related_resources(self) -> List[RelatedResource]:\n        tags = set()\n        related = []\n\n        for kind, obj in self._related_objects.items():\n            if obj is None:\n                continue\n            if hasattr(obj, \"tags\"):\n                tags.update(obj.tags)\n            related.append(object_as_related_resource(kind=kind, role=kind, object=obj))\n\n        return related + tags_as_related_resources(tags)\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseJobConfiguration.from_template_and_values","title":"from_template_and_values async classmethod","text":"

    Creates a valid worker configuration object from the provided base configuration and overrides.

    Important: this method expects that the base_job_template was already validated server-side.

    Source code in prefect/workers/base.py
    @classmethod\n@inject_client\nasync def from_template_and_values(\n    cls, base_job_template: dict, values: dict, client: \"PrefectClient\" = None\n):\n    \"\"\"Creates a valid worker configuration object from the provided base\n    configuration and overrides.\n\n    Important: this method expects that the base_job_template was already\n    validated server-side.\n    \"\"\"\n    job_config: Dict[str, Any] = base_job_template[\"job_configuration\"]\n    variables_schema = base_job_template[\"variables\"]\n    variables = cls._get_base_config_defaults(\n        variables_schema.get(\"properties\", {})\n    )\n    variables.update(values)\n\n    populated_configuration = apply_values(template=job_config, values=variables)\n    populated_configuration = await resolve_block_document_references(\n        template=populated_configuration, client=client\n    )\n    populated_configuration = await resolve_variables(\n        template=populated_configuration, client=client\n    )\n    return cls(**populated_configuration)\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseJobConfiguration.json_template","title":"json_template classmethod","text":"

    Returns a dict with job configuration as keys and the corresponding templates as values

    Defaults to using the job configuration parameter name as the template variable name.

    e.g. { key1: '{{ key1 }}', # default variable template key2: '{{ template2 }}', # template2 specifically provide as template }

    Source code in prefect/workers/base.py
    @classmethod\ndef json_template(cls) -> dict:\n    \"\"\"Returns a dict with job configuration as keys and the corresponding templates as values\n\n    Defaults to using the job configuration parameter name as the template variable name.\n\n    e.g.\n    {\n        key1: '{{ key1 }}',     # default variable template\n        key2: '{{ template2 }}', # `template2` specifically provide as template\n    }\n    \"\"\"\n    configuration = {}\n    properties = cls.schema()[\"properties\"]\n    for k, v in properties.items():\n        if v.get(\"template\"):\n            template = v[\"template\"]\n        else:\n            template = \"{{ \" + k + \" }}\"\n        configuration[k] = template\n\n    return configuration\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseJobConfiguration.prepare_for_flow_run","title":"prepare_for_flow_run","text":"

    Prepare the job configuration for a flow run.

    This method is called by the worker before starting a flow run. It should be used to set any configuration values that are dependent on the flow run.

    Parameters:

    Name Type Description Default flow_run FlowRun

    The flow run to be executed.

    required deployment Optional[DeploymentResponse]

    The deployment that the flow run is associated with.

    None flow Optional[Flow]

    The flow that the flow run is associated with.

    None Source code in prefect/workers/base.py
    def prepare_for_flow_run(\n    self,\n    flow_run: \"FlowRun\",\n    deployment: Optional[\"DeploymentResponse\"] = None,\n    flow: Optional[\"Flow\"] = None,\n):\n    \"\"\"\n    Prepare the job configuration for a flow run.\n\n    This method is called by the worker before starting a flow run. It\n    should be used to set any configuration values that are dependent on\n    the flow run.\n\n    Args:\n        flow_run: The flow run to be executed.\n        deployment: The deployment that the flow run is associated with.\n        flow: The flow that the flow run is associated with.\n    \"\"\"\n\n    self._related_objects = {\n        \"deployment\": deployment,\n        \"flow\": flow,\n        \"flow-run\": flow_run,\n    }\n    if deployment is not None:\n        deployment_labels = self._base_deployment_labels(deployment)\n    else:\n        deployment_labels = {}\n\n    if flow is not None:\n        flow_labels = self._base_flow_labels(flow)\n    else:\n        flow_labels = {}\n\n    env = {\n        **self._base_environment(),\n        **self._base_flow_run_environment(flow_run),\n        **self.env,\n    }\n    self.env = {key: value for key, value in env.items() if value is not None}\n    self.labels = {\n        **self._base_flow_run_labels(flow_run),\n        **deployment_labels,\n        **flow_labels,\n        **self.labels,\n    }\n    self.name = self.name or flow_run.name\n    self.command = self.command or self._base_flow_run_command()\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker","title":"BaseWorker","text":"

    Bases: ABC

    Source code in prefect/workers/base.py
    @register_base_type\nclass BaseWorker(abc.ABC):\n    type: str\n    job_configuration: Type[BaseJobConfiguration] = BaseJobConfiguration\n    job_configuration_variables: Optional[Type[BaseVariables]] = None\n\n    _documentation_url = \"\"\n    _logo_url = \"\"\n    _description = \"\"\n\n    def __init__(\n        self,\n        work_pool_name: str,\n        work_queues: Optional[List[str]] = None,\n        name: Optional[str] = None,\n        prefetch_seconds: Optional[float] = None,\n        create_pool_if_not_found: bool = True,\n        limit: Optional[int] = None,\n        heartbeat_interval_seconds: Optional[int] = None,\n        *,\n        base_job_template: Optional[Dict[str, Any]] = None,\n    ):\n        \"\"\"\n        Base class for all Prefect workers.\n\n        Args:\n            name: The name of the worker. If not provided, a random one\n                will be generated. If provided, it cannot contain '/' or '%'.\n                The name is used to identify the worker in the UI; if two\n                processes have the same name, they will be treated as the same\n                worker.\n            work_pool_name: The name of the work pool to poll.\n            work_queues: A list of work queues to poll. If not provided, all\n                work queue in the work pool will be polled.\n            prefetch_seconds: The number of seconds to prefetch flow runs for.\n            create_pool_if_not_found: Whether to create the work pool\n                if it is not found. Defaults to `True`, but can be set to `False` to\n                ensure that work pools are not created accidentally.\n            limit: The maximum number of flow runs this worker should be running at\n                a given time.\n            base_job_template: If creating the work pool, provide the base job\n                template to use. Logs a warning if the pool already exists.\n        \"\"\"\n        if name and (\"/\" in name or \"%\" in name):\n            raise ValueError(\"Worker name cannot contain '/' or '%'\")\n        self.name = name or f\"{self.__class__.__name__} {uuid4()}\"\n        self._logger = get_logger(f\"worker.{self.__class__.type}.{self.name.lower()}\")\n\n        self.is_setup = False\n        self._create_pool_if_not_found = create_pool_if_not_found\n        self._base_job_template = base_job_template\n        self._work_pool_name = work_pool_name\n        self._work_queues: Set[str] = set(work_queues) if work_queues else set()\n\n        self._prefetch_seconds: float = (\n            prefetch_seconds or PREFECT_WORKER_PREFETCH_SECONDS.value()\n        )\n        self.heartbeat_interval_seconds = (\n            heartbeat_interval_seconds or PREFECT_WORKER_HEARTBEAT_SECONDS.value()\n        )\n\n        self._work_pool: Optional[WorkPool] = None\n        self._runs_task_group: Optional[anyio.abc.TaskGroup] = None\n        self._client: Optional[PrefectClient] = None\n        self._last_polled_time: pendulum.DateTime = pendulum.now(\"utc\")\n        self._limit = limit\n        self._limiter: Optional[anyio.CapacityLimiter] = None\n        self._submitting_flow_run_ids = set()\n        self._cancelling_flow_run_ids = set()\n        self._scheduled_task_scopes = set()\n\n    @classmethod\n    def get_documentation_url(cls) -> str:\n        return cls._documentation_url\n\n    @classmethod\n    def get_logo_url(cls) -> str:\n        return cls._logo_url\n\n    @classmethod\n    def get_description(cls) -> str:\n        return cls._description\n\n    @classmethod\n    def get_default_base_job_template(cls) -> Dict:\n        if cls.job_configuration_variables is None:\n            schema = cls.job_configuration.schema()\n            # remove \"template\" key from all dicts in schema['properties'] because it is not a\n            # relevant field\n            for key, value in schema[\"properties\"].items():\n                if isinstance(value, dict):\n                    schema[\"properties\"][key].pop(\"template\", None)\n            variables_schema = schema\n        else:\n            variables_schema = cls.job_configuration_variables.schema()\n        variables_schema.pop(\"title\", None)\n        return {\n            \"job_configuration\": cls.job_configuration.json_template(),\n            \"variables\": variables_schema,\n        }\n\n    @staticmethod\n    def get_worker_class_from_type(type: str) -> Optional[Type[\"BaseWorker\"]]:\n        \"\"\"\n        Returns the worker class for a given worker type. If the worker type\n        is not recognized, returns None.\n        \"\"\"\n        load_prefect_collections()\n        worker_registry = get_registry_for_type(BaseWorker)\n        if worker_registry is not None:\n            return worker_registry.get(type)\n\n    @staticmethod\n    def get_all_available_worker_types() -> List[str]:\n        \"\"\"\n        Returns all worker types available in the local registry.\n        \"\"\"\n        load_prefect_collections()\n        worker_registry = get_registry_for_type(BaseWorker)\n        if worker_registry is not None:\n            return list(worker_registry.keys())\n        return []\n\n    def get_name_slug(self):\n        return slugify(self.name)\n\n    def get_flow_run_logger(self, flow_run: \"FlowRun\") -> PrefectLogAdapter:\n        return flow_run_logger(flow_run=flow_run).getChild(\n            \"worker\",\n            extra={\n                \"worker_name\": self.name,\n                \"work_pool_name\": (\n                    self._work_pool_name if self._work_pool else \"<unknown>\"\n                ),\n                \"work_pool_id\": str(getattr(self._work_pool, \"id\", \"unknown\")),\n            },\n        )\n\n    @abc.abstractmethod\n    async def run(\n        self,\n        flow_run: \"FlowRun\",\n        configuration: BaseJobConfiguration,\n        task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> BaseWorkerResult:\n        \"\"\"\n        Runs a given flow run on the current worker.\n        \"\"\"\n        raise NotImplementedError(\n            \"Workers must implement a method for running submitted flow runs\"\n        )\n\n    async def kill_infrastructure(\n        self,\n        infrastructure_pid: str,\n        configuration: BaseJobConfiguration,\n        grace_seconds: int = 30,\n    ):\n        \"\"\"\n        Method for killing infrastructure created by a worker. Should be implemented by\n        individual workers if they support killing infrastructure.\n        \"\"\"\n        raise NotImplementedError(\n            \"This worker does not support killing infrastructure.\"\n        )\n\n    @classmethod\n    def __dispatch_key__(cls):\n        if cls.__name__ == \"BaseWorker\":\n            return None  # The base class is abstract\n        return cls.type\n\n    async def setup(self):\n        \"\"\"Prepares the worker to run.\"\"\"\n        self._logger.debug(\"Setting up worker...\")\n        self._runs_task_group = anyio.create_task_group()\n        self._limiter = (\n            anyio.CapacityLimiter(self._limit) if self._limit is not None else None\n        )\n        self._client = get_client()\n        await self._client.__aenter__()\n        await self._runs_task_group.__aenter__()\n\n        self.is_setup = True\n\n    async def teardown(self, *exc_info):\n        \"\"\"Cleans up resources after the worker is stopped.\"\"\"\n        self._logger.debug(\"Tearing down worker...\")\n        self.is_setup = False\n        for scope in self._scheduled_task_scopes:\n            scope.cancel()\n        if self._runs_task_group:\n            await self._runs_task_group.__aexit__(*exc_info)\n        if self._client:\n            await self._client.__aexit__(*exc_info)\n        self._runs_task_group = None\n        self._client = None\n\n    def is_worker_still_polling(self, query_interval_seconds: int) -> bool:\n        \"\"\"\n        This method is invoked by a webserver healthcheck handler\n        and returns a boolean indicating if the worker has recorded a\n        scheduled flow run poll within a variable amount of time.\n\n        The `query_interval_seconds` is the same value that is used by\n        the loop services - we will evaluate if the _last_polled_time\n        was within that interval x 30 (so 10s -> 5m)\n\n        The instance property `self._last_polled_time`\n        is currently set/updated in `get_and_submit_flow_runs()`\n        \"\"\"\n        threshold_seconds = query_interval_seconds * 30\n\n        seconds_since_last_poll = (\n            pendulum.now(\"utc\") - self._last_polled_time\n        ).in_seconds()\n\n        is_still_polling = seconds_since_last_poll <= threshold_seconds\n\n        if not is_still_polling:\n            self._logger.error(\n                f\"Worker has not polled in the last {seconds_since_last_poll} seconds \"\n                \"and should be restarted\"\n            )\n\n        return is_still_polling\n\n    async def get_and_submit_flow_runs(self):\n        runs_response = await self._get_scheduled_flow_runs()\n\n        self._last_polled_time = pendulum.now(\"utc\")\n\n        return await self._submit_scheduled_flow_runs(flow_run_response=runs_response)\n\n    async def check_for_cancelled_flow_runs(self):\n        if not self.is_setup:\n            raise RuntimeError(\n                \"Worker is not set up. Please make sure you are running this worker \"\n                \"as an async context manager.\"\n            )\n\n        self._logger.debug(\"Checking for cancelled flow runs...\")\n\n        work_queue_filter = (\n            WorkQueueFilter(name=WorkQueueFilterName(any_=list(self._work_queues)))\n            if self._work_queues\n            else None\n        )\n\n        named_cancelling_flow_runs = await self._client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLED]),\n                    name=FlowRunFilterStateName(any_=[\"Cancelling\"]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(not_any_=list(self._cancelling_flow_run_ids)),\n            ),\n            work_pool_filter=WorkPoolFilter(\n                name=WorkPoolFilterName(any_=[self._work_pool_name])\n            ),\n            work_queue_filter=work_queue_filter,\n        )\n\n        typed_cancelling_flow_runs = await self._client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=FlowRunFilterState(\n                    type=FlowRunFilterStateType(any_=[StateType.CANCELLING]),\n                ),\n                # Avoid duplicate cancellation calls\n                id=FlowRunFilterId(not_any_=list(self._cancelling_flow_run_ids)),\n            ),\n            work_pool_filter=WorkPoolFilter(\n                name=WorkPoolFilterName(any_=[self._work_pool_name])\n            ),\n            work_queue_filter=work_queue_filter,\n        )\n\n        cancelling_flow_runs = named_cancelling_flow_runs + typed_cancelling_flow_runs\n\n        if cancelling_flow_runs:\n            self._logger.info(\n                f\"Found {len(cancelling_flow_runs)} flow runs awaiting cancellation.\"\n            )\n\n        for flow_run in cancelling_flow_runs:\n            self._cancelling_flow_run_ids.add(flow_run.id)\n            self._runs_task_group.start_soon(self.cancel_run, flow_run)\n\n        return cancelling_flow_runs\n\n    async def cancel_run(self, flow_run: \"FlowRun\"):\n        run_logger = self.get_flow_run_logger(flow_run)\n\n        try:\n            configuration = await self._get_configuration(flow_run)\n            if configuration.is_using_a_runner:\n                self._logger.info(\n                    f\"Skipping cancellation because flow run {str(flow_run.id)!r} is\"\n                    \" using enhanced cancellation. A dedicated runner will handle\"\n                    \" cancellation.\"\n                )\n                return\n        except ObjectNotFound:\n            self._logger.warning(\n                f\"Flow run {flow_run.id!r} cannot be cancelled by this worker:\"\n                f\" associated deployment {flow_run.deployment_id!r} does not exist.\"\n            )\n\n        if not flow_run.infrastructure_pid:\n            run_logger.error(\n                f\"Flow run '{flow_run.id}' does not have an infrastructure pid\"\n                \" attached. Cancellation cannot be guaranteed.\"\n            )\n            await self._mark_flow_run_as_cancelled(\n                flow_run,\n                state_updates={\n                    \"message\": (\n                        \"This flow run is missing infrastructure tracking information\"\n                        \" and cancellation cannot be guaranteed.\"\n                    )\n                },\n            )\n            return\n\n        try:\n            await self.kill_infrastructure(\n                infrastructure_pid=flow_run.infrastructure_pid,\n                configuration=configuration,\n            )\n        except NotImplementedError:\n            self._logger.error(\n                f\"Worker type {self.type!r} does not support killing created \"\n                \"infrastructure. Cancellation cannot be guaranteed.\"\n            )\n        except InfrastructureNotFound as exc:\n            self._logger.warning(f\"{exc} Marking flow run as cancelled.\")\n            await self._mark_flow_run_as_cancelled(flow_run)\n        except InfrastructureNotAvailable as exc:\n            self._logger.warning(f\"{exc} Flow run cannot be cancelled by this worker.\")\n        except Exception:\n            run_logger.exception(\n                \"Encountered exception while killing infrastructure for flow run \"\n                f\"'{flow_run.id}'. Flow run may not be cancelled.\"\n            )\n            # We will try again on generic exceptions\n            self._cancelling_flow_run_ids.remove(flow_run.id)\n            return\n        else:\n            self._emit_flow_run_cancelled_event(\n                flow_run=flow_run, configuration=configuration\n            )\n            await self._mark_flow_run_as_cancelled(flow_run)\n            run_logger.info(f\"Cancelled flow run '{flow_run.id}'!\")\n\n    async def _update_local_work_pool_info(self):\n        try:\n            work_pool = await self._client.read_work_pool(\n                work_pool_name=self._work_pool_name\n            )\n        except ObjectNotFound:\n            if self._create_pool_if_not_found:\n                wp = WorkPoolCreate(\n                    name=self._work_pool_name,\n                    type=self.type,\n                )\n                if self._base_job_template is not None:\n                    wp.base_job_template = self._base_job_template\n\n                work_pool = await self._client.create_work_pool(work_pool=wp)\n                self._logger.info(f\"Work pool {self._work_pool_name!r} created.\")\n            else:\n                self._logger.warning(f\"Work pool {self._work_pool_name!r} not found!\")\n                if self._base_job_template is not None:\n                    self._logger.warning(\n                        \"Ignoring supplied base job template because the work pool\"\n                        \" already exists\"\n                    )\n                return\n\n        # if the remote config type changes (or if it's being loaded for the\n        # first time), check if it matches the local type and warn if not\n        if getattr(self._work_pool, \"type\", 0) != work_pool.type:\n            if work_pool.type != self.__class__.type:\n                self._logger.warning(\n                    \"Worker type mismatch! This worker process expects type \"\n                    f\"{self.type!r} but received {work_pool.type!r}\"\n                    \" from the server. Unexpected behavior may occur.\"\n                )\n\n        # once the work pool is loaded, verify that it has a `base_job_template` and\n        # set it if not\n        if not work_pool.base_job_template:\n            job_template = self.__class__.get_default_base_job_template()\n            await self._set_work_pool_template(work_pool, job_template)\n            work_pool.base_job_template = job_template\n\n        self._work_pool = work_pool\n\n    async def _send_worker_heartbeat(self):\n        if self._work_pool:\n            await self._client.send_worker_heartbeat(\n                work_pool_name=self._work_pool_name,\n                worker_name=self.name,\n                heartbeat_interval_seconds=self.heartbeat_interval_seconds,\n            )\n\n    async def sync_with_backend(self):\n        \"\"\"\n        Updates the worker's local information about it's current work pool and\n        queues. Sends a worker heartbeat to the API.\n        \"\"\"\n        await self._update_local_work_pool_info()\n\n        await self._send_worker_heartbeat()\n\n        self._logger.debug(\"Worker synchronized with the Prefect API server.\")\n\n    async def _get_scheduled_flow_runs(\n        self,\n    ) -> List[\"WorkerFlowRunResponse\"]:\n        \"\"\"\n        Retrieve scheduled flow runs from the work pool's queues.\n        \"\"\"\n        scheduled_before = pendulum.now(\"utc\").add(seconds=int(self._prefetch_seconds))\n        self._logger.debug(\n            f\"Querying for flow runs scheduled before {scheduled_before}\"\n        )\n        try:\n            scheduled_flow_runs = (\n                await self._client.get_scheduled_flow_runs_for_work_pool(\n                    work_pool_name=self._work_pool_name,\n                    scheduled_before=scheduled_before,\n                    work_queue_names=list(self._work_queues),\n                )\n            )\n            self._logger.debug(\n                f\"Discovered {len(scheduled_flow_runs)} scheduled_flow_runs\"\n            )\n            return scheduled_flow_runs\n        except ObjectNotFound:\n            # the pool doesn't exist; it will be created on the next\n            # heartbeat (or an appropriate warning will be logged)\n            return []\n\n    async def _submit_scheduled_flow_runs(\n        self, flow_run_response: List[\"WorkerFlowRunResponse\"]\n    ) -> List[\"FlowRun\"]:\n        \"\"\"\n        Takes a list of WorkerFlowRunResponses and submits the referenced flow runs\n        for execution by the worker.\n        \"\"\"\n        submittable_flow_runs = [entry.flow_run for entry in flow_run_response]\n        submittable_flow_runs.sort(key=lambda run: run.next_scheduled_start_time)\n        for flow_run in submittable_flow_runs:\n            if flow_run.id in self._submitting_flow_run_ids:\n                continue\n\n            try:\n                if self._limiter:\n                    self._limiter.acquire_on_behalf_of_nowait(flow_run.id)\n            except anyio.WouldBlock:\n                self._logger.info(\n                    f\"Flow run limit reached; {self._limiter.borrowed_tokens} flow runs\"\n                    \" in progress.\"\n                )\n                break\n            else:\n                run_logger = self.get_flow_run_logger(flow_run)\n                run_logger.info(\n                    f\"Worker '{self.name}' submitting flow run '{flow_run.id}'\"\n                )\n                self._submitting_flow_run_ids.add(flow_run.id)\n                self._runs_task_group.start_soon(\n                    self._submit_run,\n                    flow_run,\n                )\n\n        return list(\n            filter(\n                lambda run: run.id in self._submitting_flow_run_ids,\n                submittable_flow_runs,\n            )\n        )\n\n    async def _check_flow_run(self, flow_run: \"FlowRun\") -> None:\n        \"\"\"\n        Performs a check on a submitted flow run to warn the user if the flow run\n        was created from a deployment with a storage block.\n        \"\"\"\n        if flow_run.deployment_id:\n            deployment = await self._client.read_deployment(flow_run.deployment_id)\n            if deployment.storage_document_id:\n                raise ValueError(\n                    f\"Flow run {flow_run.id!r} was created from deployment\"\n                    f\" {deployment.name!r} which is configured with a storage block.\"\n                    \" Please use an\"\n                    \" agent to execute this flow run.\"\n                )\n\n    async def _submit_run(self, flow_run: \"FlowRun\") -> None:\n        \"\"\"\n        Submits a given flow run for execution by the worker.\n        \"\"\"\n        run_logger = self.get_flow_run_logger(flow_run)\n\n        try:\n            await self._check_flow_run(flow_run)\n        except (ValueError, ObjectNotFound):\n            self._logger.exception(\n                (\n                    \"Flow run %s did not pass checks and will not be submitted for\"\n                    \" execution\"\n                ),\n                flow_run.id,\n            )\n            self._submitting_flow_run_ids.remove(flow_run.id)\n            return\n\n        ready_to_submit = await self._propose_pending_state(flow_run)\n\n        if ready_to_submit:\n            readiness_result = await self._runs_task_group.start(\n                self._submit_run_and_capture_errors, flow_run\n            )\n\n            if readiness_result and not isinstance(readiness_result, Exception):\n                try:\n                    await self._client.update_flow_run(\n                        flow_run_id=flow_run.id,\n                        infrastructure_pid=str(readiness_result),\n                    )\n                except Exception:\n                    run_logger.exception(\n                        \"An error occurred while setting the `infrastructure_pid` on \"\n                        f\"flow run {flow_run.id!r}. The flow run will \"\n                        \"not be cancellable.\"\n                    )\n\n            run_logger.info(f\"Completed submission of flow run '{flow_run.id}'\")\n\n        else:\n            # If the run is not ready to submit, release the concurrency slot\n            if self._limiter:\n                self._limiter.release_on_behalf_of(flow_run.id)\n\n        self._submitting_flow_run_ids.remove(flow_run.id)\n\n    async def _submit_run_and_capture_errors(\n        self, flow_run: \"FlowRun\", task_status: anyio.abc.TaskStatus = None\n    ) -> Union[BaseWorkerResult, Exception]:\n        run_logger = self.get_flow_run_logger(flow_run)\n\n        try:\n            configuration = await self._get_configuration(flow_run)\n            submitted_event = self._emit_flow_run_submitted_event(configuration)\n            result = await self.run(\n                flow_run=flow_run,\n                task_status=task_status,\n                configuration=configuration,\n            )\n        except Exception as exc:\n            if not task_status._future.done():\n                # This flow run was being submitted and did not start successfully\n                run_logger.exception(\n                    f\"Failed to submit flow run '{flow_run.id}' to infrastructure.\"\n                )\n                # Mark the task as started to prevent agent crash\n                task_status.started(exc)\n                await self._propose_crashed_state(\n                    flow_run, \"Flow run could not be submitted to infrastructure\"\n                )\n            else:\n                run_logger.exception(\n                    f\"An error occurred while monitoring flow run '{flow_run.id}'. \"\n                    \"The flow run will not be marked as failed, but an issue may have \"\n                    \"occurred.\"\n                )\n            return exc\n        finally:\n            if self._limiter:\n                self._limiter.release_on_behalf_of(flow_run.id)\n\n        if not task_status._future.done():\n            run_logger.error(\n                f\"Infrastructure returned without reporting flow run '{flow_run.id}' \"\n                \"as started or raising an error. This behavior is not expected and \"\n                \"generally indicates improper implementation of infrastructure. The \"\n                \"flow run will not be marked as failed, but an issue may have occurred.\"\n            )\n            # Mark the task as started to prevent agent crash\n            task_status.started()\n\n        if result.status_code != 0:\n            await self._propose_crashed_state(\n                flow_run,\n                (\n                    \"Flow run infrastructure exited with non-zero status code\"\n                    f\" {result.status_code}.\"\n                ),\n            )\n\n        self._emit_flow_run_executed_event(result, configuration, submitted_event)\n\n        return result\n\n    def get_status(self):\n        \"\"\"\n        Retrieves the status of the current worker including its name, current worker\n        pool, the work pool queues it is polling, and its local settings.\n        \"\"\"\n        return {\n            \"name\": self.name,\n            \"work_pool\": (\n                self._work_pool.dict(json_compatible=True)\n                if self._work_pool is not None\n                else None\n            ),\n            \"settings\": {\n                \"prefetch_seconds\": self._prefetch_seconds,\n            },\n        }\n\n    async def _get_configuration(\n        self,\n        flow_run: \"FlowRun\",\n    ) -> BaseJobConfiguration:\n        deployment = await self._client.read_deployment(flow_run.deployment_id)\n        flow = await self._client.read_flow(flow_run.flow_id)\n        configuration = await self.job_configuration.from_template_and_values(\n            base_job_template=self._work_pool.base_job_template,\n            values=deployment.infra_overrides or {},\n            client=self._client,\n        )\n        configuration.prepare_for_flow_run(\n            flow_run=flow_run, deployment=deployment, flow=flow\n        )\n        return configuration\n\n    async def _propose_pending_state(self, flow_run: \"FlowRun\") -> bool:\n        run_logger = self.get_flow_run_logger(flow_run)\n        state = flow_run.state\n        try:\n            state = await propose_state(\n                self._client, Pending(), flow_run_id=flow_run.id\n            )\n        except Abort as exc:\n            run_logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}'. \"\n                    f\"Server sent an abort signal: {exc}\"\n                ),\n            )\n            return False\n        except Exception:\n            run_logger.exception(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n            )\n            return False\n\n        if not state.is_pending():\n            run_logger.info(\n                (\n                    f\"Aborted submission of flow run '{flow_run.id}': \"\n                    f\"Server returned a non-pending state {state.type.value!r}\"\n                ),\n            )\n            return False\n\n        return True\n\n    async def _propose_failed_state(self, flow_run: \"FlowRun\", exc: Exception) -> None:\n        run_logger = self.get_flow_run_logger(flow_run)\n        try:\n            await propose_state(\n                self._client,\n                await exception_to_failed_state(message=\"Submission failed.\", exc=exc),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # We've already failed, no need to note the abort but we don't want it to\n            # raise in the agent process\n            pass\n        except Exception:\n            run_logger.error(\n                f\"Failed to update state of flow run '{flow_run.id}'\",\n                exc_info=True,\n            )\n\n    async def _propose_crashed_state(self, flow_run: \"FlowRun\", message: str) -> None:\n        run_logger = self.get_flow_run_logger(flow_run)\n        try:\n            state = await propose_state(\n                self._client,\n                Crashed(message=message),\n                flow_run_id=flow_run.id,\n            )\n        except Abort:\n            # Flow run already marked as failed\n            pass\n        except Exception:\n            run_logger.exception(f\"Failed to update state of flow run '{flow_run.id}'\")\n        else:\n            if state.is_crashed():\n                run_logger.info(\n                    f\"Reported flow run '{flow_run.id}' as crashed: {message}\"\n                )\n\n    async def _mark_flow_run_as_cancelled(\n        self, flow_run: \"FlowRun\", state_updates: Optional[dict] = None\n    ) -> None:\n        state_updates = state_updates or {}\n        state_updates.setdefault(\"name\", \"Cancelled\")\n        state_updates.setdefault(\"type\", StateType.CANCELLED)\n        state = flow_run.state.copy(update=state_updates)\n\n        await self._client.set_flow_run_state(flow_run.id, state, force=True)\n\n        # Do not remove the flow run from the cancelling set immediately because\n        # the API caches responses for the `read_flow_runs` and we do not want to\n        # duplicate cancellations.\n        await self._schedule_task(\n            60 * 10, self._cancelling_flow_run_ids.remove, flow_run.id\n        )\n\n    async def _set_work_pool_template(self, work_pool, job_template):\n        \"\"\"Updates the `base_job_template` for the worker's work pool server side.\"\"\"\n        await self._client.update_work_pool(\n            work_pool_name=work_pool.name,\n            work_pool=WorkPoolUpdate(\n                base_job_template=job_template,\n            ),\n        )\n\n    async def _schedule_task(self, __in_seconds: int, fn, *args, **kwargs):\n        \"\"\"\n        Schedule a background task to start after some time.\n\n        These tasks will be run immediately when the worker exits instead of waiting.\n\n        The function may be async or sync. Async functions will be awaited.\n        \"\"\"\n\n        async def wrapper(task_status):\n            # If we are shutting down, do not sleep; otherwise sleep until the scheduled\n            # time or shutdown\n            if self.is_setup:\n                with anyio.CancelScope() as scope:\n                    self._scheduled_task_scopes.add(scope)\n                    task_status.started()\n                    await anyio.sleep(__in_seconds)\n\n                self._scheduled_task_scopes.remove(scope)\n            else:\n                task_status.started()\n\n            result = fn(*args, **kwargs)\n            if inspect.iscoroutine(result):\n                await result\n\n        await self._runs_task_group.start(wrapper)\n\n    async def __aenter__(self):\n        self._logger.debug(\"Entering worker context...\")\n        await self.setup()\n        return self\n\n    async def __aexit__(self, *exc_info):\n        self._logger.debug(\"Exiting worker context...\")\n        await self.teardown(*exc_info)\n\n    def __repr__(self):\n        return f\"Worker(pool={self._work_pool_name!r}, name={self.name!r})\"\n\n    def _event_resource(self):\n        return {\n            \"prefect.resource.id\": f\"prefect.worker.{self.type}.{self.get_name_slug()}\",\n            \"prefect.resource.name\": self.name,\n            \"prefect.version\": prefect.__version__,\n            \"prefect.worker-type\": self.type,\n        }\n\n    def _event_related_resources(\n        self,\n        configuration: Optional[BaseJobConfiguration] = None,\n        include_self: bool = False,\n    ) -> List[RelatedResource]:\n        related = []\n        if configuration:\n            related += configuration._related_resources()\n\n        if self._work_pool:\n            related.append(\n                object_as_related_resource(\n                    kind=\"work-pool\", role=\"work-pool\", object=self._work_pool\n                )\n            )\n\n        if include_self:\n            worker_resource = self._event_resource()\n            worker_resource[\"prefect.resource.role\"] = \"worker\"\n            related.append(RelatedResource(__root__=worker_resource))\n\n        return related\n\n    def _emit_flow_run_submitted_event(\n        self, configuration: BaseJobConfiguration\n    ) -> Event:\n        return emit_event(\n            event=\"prefect.worker.submitted-flow-run\",\n            resource=self._event_resource(),\n            related=self._event_related_resources(configuration=configuration),\n        )\n\n    def _emit_flow_run_executed_event(\n        self,\n        result: BaseWorkerResult,\n        configuration: BaseJobConfiguration,\n        submitted_event: Event,\n    ):\n        related = self._event_related_resources(configuration=configuration)\n\n        for resource in related:\n            if resource.role == \"flow-run\":\n                resource[\"prefect.infrastructure.identifier\"] = str(result.identifier)\n                resource[\"prefect.infrastructure.status-code\"] = str(result.status_code)\n\n        emit_event(\n            event=\"prefect.worker.executed-flow-run\",\n            resource=self._event_resource(),\n            related=related,\n            follows=submitted_event,\n        )\n\n    async def _emit_worker_started_event(self) -> Event:\n        return emit_event(\n            \"prefect.worker.started\",\n            resource=self._event_resource(),\n            related=self._event_related_resources(),\n        )\n\n    async def _emit_worker_stopped_event(self, started_event: Event):\n        emit_event(\n            \"prefect.worker.stopped\",\n            resource=self._event_resource(),\n            related=self._event_related_resources(),\n            follows=started_event,\n        )\n\n    def _emit_flow_run_cancelled_event(\n        self, flow_run: \"FlowRun\", configuration: BaseJobConfiguration\n    ):\n        related = self._event_related_resources(configuration=configuration)\n\n        for resource in related:\n            if resource.role == \"flow-run\":\n                resource[\"prefect.infrastructure.identifier\"] = str(\n                    flow_run.infrastructure_pid\n                )\n\n        emit_event(\n            event=\"prefect.worker.cancelled-flow-run\",\n            resource=self._event_resource(),\n            related=related,\n        )\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.get_all_available_worker_types","title":"get_all_available_worker_types staticmethod","text":"

    Returns all worker types available in the local registry.

    Source code in prefect/workers/base.py
    @staticmethod\ndef get_all_available_worker_types() -> List[str]:\n    \"\"\"\n    Returns all worker types available in the local registry.\n    \"\"\"\n    load_prefect_collections()\n    worker_registry = get_registry_for_type(BaseWorker)\n    if worker_registry is not None:\n        return list(worker_registry.keys())\n    return []\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.get_status","title":"get_status","text":"

    Retrieves the status of the current worker including its name, current worker pool, the work pool queues it is polling, and its local settings.

    Source code in prefect/workers/base.py
    def get_status(self):\n    \"\"\"\n    Retrieves the status of the current worker including its name, current worker\n    pool, the work pool queues it is polling, and its local settings.\n    \"\"\"\n    return {\n        \"name\": self.name,\n        \"work_pool\": (\n            self._work_pool.dict(json_compatible=True)\n            if self._work_pool is not None\n            else None\n        ),\n        \"settings\": {\n            \"prefetch_seconds\": self._prefetch_seconds,\n        },\n    }\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.get_worker_class_from_type","title":"get_worker_class_from_type staticmethod","text":"

    Returns the worker class for a given worker type. If the worker type is not recognized, returns None.

    Source code in prefect/workers/base.py
    @staticmethod\ndef get_worker_class_from_type(type: str) -> Optional[Type[\"BaseWorker\"]]:\n    \"\"\"\n    Returns the worker class for a given worker type. If the worker type\n    is not recognized, returns None.\n    \"\"\"\n    load_prefect_collections()\n    worker_registry = get_registry_for_type(BaseWorker)\n    if worker_registry is not None:\n        return worker_registry.get(type)\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.is_worker_still_polling","title":"is_worker_still_polling","text":"

    This method is invoked by a webserver healthcheck handler and returns a boolean indicating if the worker has recorded a scheduled flow run poll within a variable amount of time.

    The query_interval_seconds is the same value that is used by the loop services - we will evaluate if the _last_polled_time was within that interval x 30 (so 10s -> 5m)

    The instance property self._last_polled_time is currently set/updated in get_and_submit_flow_runs()

    Source code in prefect/workers/base.py
    def is_worker_still_polling(self, query_interval_seconds: int) -> bool:\n    \"\"\"\n    This method is invoked by a webserver healthcheck handler\n    and returns a boolean indicating if the worker has recorded a\n    scheduled flow run poll within a variable amount of time.\n\n    The `query_interval_seconds` is the same value that is used by\n    the loop services - we will evaluate if the _last_polled_time\n    was within that interval x 30 (so 10s -> 5m)\n\n    The instance property `self._last_polled_time`\n    is currently set/updated in `get_and_submit_flow_runs()`\n    \"\"\"\n    threshold_seconds = query_interval_seconds * 30\n\n    seconds_since_last_poll = (\n        pendulum.now(\"utc\") - self._last_polled_time\n    ).in_seconds()\n\n    is_still_polling = seconds_since_last_poll <= threshold_seconds\n\n    if not is_still_polling:\n        self._logger.error(\n            f\"Worker has not polled in the last {seconds_since_last_poll} seconds \"\n            \"and should be restarted\"\n        )\n\n    return is_still_polling\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.kill_infrastructure","title":"kill_infrastructure async","text":"

    Method for killing infrastructure created by a worker. Should be implemented by individual workers if they support killing infrastructure.

    Source code in prefect/workers/base.py
    async def kill_infrastructure(\n    self,\n    infrastructure_pid: str,\n    configuration: BaseJobConfiguration,\n    grace_seconds: int = 30,\n):\n    \"\"\"\n    Method for killing infrastructure created by a worker. Should be implemented by\n    individual workers if they support killing infrastructure.\n    \"\"\"\n    raise NotImplementedError(\n        \"This worker does not support killing infrastructure.\"\n    )\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.run","title":"run abstractmethod async","text":"

    Runs a given flow run on the current worker.

    Source code in prefect/workers/base.py
    @abc.abstractmethod\nasync def run(\n    self,\n    flow_run: \"FlowRun\",\n    configuration: BaseJobConfiguration,\n    task_status: Optional[anyio.abc.TaskStatus] = None,\n) -> BaseWorkerResult:\n    \"\"\"\n    Runs a given flow run on the current worker.\n    \"\"\"\n    raise NotImplementedError(\n        \"Workers must implement a method for running submitted flow runs\"\n    )\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.setup","title":"setup async","text":"

    Prepares the worker to run.

    Source code in prefect/workers/base.py
    async def setup(self):\n    \"\"\"Prepares the worker to run.\"\"\"\n    self._logger.debug(\"Setting up worker...\")\n    self._runs_task_group = anyio.create_task_group()\n    self._limiter = (\n        anyio.CapacityLimiter(self._limit) if self._limit is not None else None\n    )\n    self._client = get_client()\n    await self._client.__aenter__()\n    await self._runs_task_group.__aenter__()\n\n    self.is_setup = True\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.sync_with_backend","title":"sync_with_backend async","text":"

    Updates the worker's local information about it's current work pool and queues. Sends a worker heartbeat to the API.

    Source code in prefect/workers/base.py
    async def sync_with_backend(self):\n    \"\"\"\n    Updates the worker's local information about it's current work pool and\n    queues. Sends a worker heartbeat to the API.\n    \"\"\"\n    await self._update_local_work_pool_info()\n\n    await self._send_worker_heartbeat()\n\n    self._logger.debug(\"Worker synchronized with the Prefect API server.\")\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/base/#prefect.workers.base.BaseWorker.teardown","title":"teardown async","text":"

    Cleans up resources after the worker is stopped.

    Source code in prefect/workers/base.py
    async def teardown(self, *exc_info):\n    \"\"\"Cleans up resources after the worker is stopped.\"\"\"\n    self._logger.debug(\"Tearing down worker...\")\n    self.is_setup = False\n    for scope in self._scheduled_task_scopes:\n        scope.cancel()\n    if self._runs_task_group:\n        await self._runs_task_group.__aexit__(*exc_info)\n    if self._client:\n        await self._client.__aexit__(*exc_info)\n    self._runs_task_group = None\n    self._client = None\n
    ","tags":["Python API","workers"]},{"location":"api-ref/prefect/workers/block/","title":"block","text":"","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block","title":"prefect.workers.block","text":"","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block.BlockWorkerJobConfiguration","title":"BlockWorkerJobConfiguration","text":"

    Bases: BaseModel

    Source code in prefect/workers/block.py
    class BlockWorkerJobConfiguration(BaseModel):\n    block: Block = Field(\n        default=..., description=\"The infrastructure block to use for job creation.\"\n    )\n\n    @validator(\"block\")\n    def _validate_block_is_infrastructure(cls, v):\n        print(\"v: \", v)\n        if not isinstance(v, Infrastructure):\n            raise TypeError(\"Provided block is not a valid infrastructure block.\")\n\n        return v\n\n    _related_objects: Dict[str, Any] = PrivateAttr(default_factory=dict)\n\n    @property\n    def is_using_a_runner(self):\n        return (\n            self.block.command is not None\n            and \"prefect flow-run execute\" in shlex.join(self.block.command)\n        )\n\n    @staticmethod\n    def _get_base_config_defaults(variables: dict) -> dict:\n        \"\"\"Get default values from base config for all variables that have them.\"\"\"\n        defaults = dict()\n        for variable_name, attrs in variables.items():\n            if \"default\" in attrs:\n                defaults[variable_name] = attrs[\"default\"]\n\n        return defaults\n\n    @classmethod\n    @inject_client\n    async def from_template_and_values(\n        cls, base_job_template: dict, values: dict, client: \"PrefectClient\" = None\n    ):\n        \"\"\"Creates a valid worker configuration object from the provided base\n        configuration and overrides.\n\n        Important: this method expects that the base_job_template was already\n        validated server-side.\n        \"\"\"\n        job_config: Dict[str, Any] = base_job_template[\"job_configuration\"]\n        variables_schema = base_job_template[\"variables\"]\n        variables = cls._get_base_config_defaults(\n            variables_schema.get(\"properties\", {})\n        )\n        variables.update(values)\n\n        populated_configuration = apply_values(template=job_config, values=variables)\n\n        block_document_id = get_from_dict(\n            populated_configuration, \"block.$ref.block_document_id\"\n        )\n        if not block_document_id:\n            raise ValueError(\n                \"Base job template is invalid for this worker type because it does not\"\n                \" contain a block_document_id after variable resolution.\"\n            )\n\n        block_document = await client.read_block_document(\n            block_document_id=block_document_id\n        )\n        infrastructure_block = Block._from_block_document(block_document)\n\n        populated_configuration[\"block\"] = infrastructure_block\n\n        return cls(**populated_configuration)\n\n    @classmethod\n    def json_template(cls) -> dict:\n        \"\"\"Returns a dict with job configuration as keys and the corresponding templates as values\n\n        Defaults to using the job configuration parameter name as the template variable name.\n\n        e.g.\n        {\n            key1: '{{ key1 }}',     # default variable template\n            key2: '{{ template2 }}', # `template2` specifically provide as template\n        }\n        \"\"\"\n        configuration = {}\n        properties = cls.schema()[\"properties\"]\n        for k, v in properties.items():\n            if v.get(\"template\"):\n                template = v[\"template\"]\n            else:\n                template = \"{{ \" + k + \" }}\"\n            configuration[k] = template\n\n        return configuration\n\n    def _related_resources(self) -> List[RelatedResource]:\n        tags = set()\n        related = []\n\n        for kind, obj in self._related_objects.items():\n            if obj is None:\n                continue\n            if hasattr(obj, \"tags\"):\n                tags.update(obj.tags)\n            related.append(object_as_related_resource(kind=kind, role=kind, object=obj))\n\n        return related + tags_as_related_resources(tags)\n\n    def prepare_for_flow_run(\n        self,\n        flow_run: \"FlowRun\",\n        deployment: Optional[\"DeploymentResponse\"] = None,\n        flow: Optional[\"Flow\"] = None,\n    ):\n        self.block = self.block.prepare_for_flow_run(\n            flow_run=flow_run, deployment=deployment, flow=flow\n        )\n
    ","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block.BlockWorkerJobConfiguration.from_template_and_values","title":"from_template_and_values async classmethod","text":"

    Creates a valid worker configuration object from the provided base configuration and overrides.

    Important: this method expects that the base_job_template was already validated server-side.

    Source code in prefect/workers/block.py
    @classmethod\n@inject_client\nasync def from_template_and_values(\n    cls, base_job_template: dict, values: dict, client: \"PrefectClient\" = None\n):\n    \"\"\"Creates a valid worker configuration object from the provided base\n    configuration and overrides.\n\n    Important: this method expects that the base_job_template was already\n    validated server-side.\n    \"\"\"\n    job_config: Dict[str, Any] = base_job_template[\"job_configuration\"]\n    variables_schema = base_job_template[\"variables\"]\n    variables = cls._get_base_config_defaults(\n        variables_schema.get(\"properties\", {})\n    )\n    variables.update(values)\n\n    populated_configuration = apply_values(template=job_config, values=variables)\n\n    block_document_id = get_from_dict(\n        populated_configuration, \"block.$ref.block_document_id\"\n    )\n    if not block_document_id:\n        raise ValueError(\n            \"Base job template is invalid for this worker type because it does not\"\n            \" contain a block_document_id after variable resolution.\"\n        )\n\n    block_document = await client.read_block_document(\n        block_document_id=block_document_id\n    )\n    infrastructure_block = Block._from_block_document(block_document)\n\n    populated_configuration[\"block\"] = infrastructure_block\n\n    return cls(**populated_configuration)\n
    ","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block.BlockWorkerJobConfiguration.json_template","title":"json_template classmethod","text":"

    Returns a dict with job configuration as keys and the corresponding templates as values

    Defaults to using the job configuration parameter name as the template variable name.

    e.g. { key1: '{{ key1 }}', # default variable template key2: '{{ template2 }}', # template2 specifically provide as template }

    Source code in prefect/workers/block.py
    @classmethod\ndef json_template(cls) -> dict:\n    \"\"\"Returns a dict with job configuration as keys and the corresponding templates as values\n\n    Defaults to using the job configuration parameter name as the template variable name.\n\n    e.g.\n    {\n        key1: '{{ key1 }}',     # default variable template\n        key2: '{{ template2 }}', # `template2` specifically provide as template\n    }\n    \"\"\"\n    configuration = {}\n    properties = cls.schema()[\"properties\"]\n    for k, v in properties.items():\n        if v.get(\"template\"):\n            template = v[\"template\"]\n        else:\n            template = \"{{ \" + k + \" }}\"\n        configuration[k] = template\n\n    return configuration\n
    ","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/block/#prefect.workers.block.BlockWorkerResult","title":"BlockWorkerResult","text":"

    Bases: BaseWorkerResult

    Result of a block worker job

    Source code in prefect/workers/block.py
    class BlockWorkerResult(BaseWorkerResult):\n    \"\"\"Result of a block worker job\"\"\"\n
    ","tags":["Python API","workers","block"]},{"location":"api-ref/prefect/workers/process/","title":"process","text":"","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/process/#prefect.workers.process","title":"prefect.workers.process","text":"

    Module containing the Process worker used for executing flow runs as subprocesses.

    To start a Process worker, run the following command:

    prefect worker start --pool 'my-work-pool' --type process\n

    Replace my-work-pool with the name of the work pool you want the worker to poll for flow runs.

    For more information about work pools and workers, checkout out the Prefect docs.

    ","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/process/#prefect.workers.process.ProcessJobConfiguration","title":"ProcessJobConfiguration","text":"

    Bases: BaseJobConfiguration

    Source code in prefect/workers/process.py
    class ProcessJobConfiguration(BaseJobConfiguration):\n    stream_output: bool = Field(default=True)\n    working_dir: Optional[Path] = Field(default=None)\n\n    @validator(\"working_dir\")\n    def validate_command(cls, v):\n        \"\"\"Make sure that the working directory is formatted for the current platform.\"\"\"\n        if v:\n            return relative_path_to_current_platform(v)\n        return v\n\n    def prepare_for_flow_run(\n        self,\n        flow_run: \"FlowRun\",\n        deployment: Optional[\"DeploymentResponse\"] = None,\n        flow: Optional[\"Flow\"] = None,\n    ):\n        super().prepare_for_flow_run(flow_run, deployment, flow)\n\n        self.env = {**os.environ, **self.env}\n        self.command = (\n            f\"{get_sys_executable()} -m prefect.engine\"\n            if self.command == self._base_flow_run_command()\n            else self.command\n        )\n\n    def _base_flow_run_command(self) -> str:\n        \"\"\"\n        Override the base flow run command because enhanced cancellation doesn't\n        work with the process worker.\n        \"\"\"\n        return \"python -m prefect.engine\"\n
    ","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/process/#prefect.workers.process.ProcessJobConfiguration.validate_command","title":"validate_command","text":"

    Make sure that the working directory is formatted for the current platform.

    Source code in prefect/workers/process.py
    @validator(\"working_dir\")\ndef validate_command(cls, v):\n    \"\"\"Make sure that the working directory is formatted for the current platform.\"\"\"\n    if v:\n        return relative_path_to_current_platform(v)\n    return v\n
    ","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/process/#prefect.workers.process.ProcessWorkerResult","title":"ProcessWorkerResult","text":"

    Bases: BaseWorkerResult

    Contains information about the final state of a completed process

    Source code in prefect/workers/process.py
    class ProcessWorkerResult(BaseWorkerResult):\n    \"\"\"Contains information about the final state of a completed process\"\"\"\n
    ","tags":["Python API","workers","process"]},{"location":"api-ref/prefect/workers/server/","title":"server","text":"","tags":["Python API","workers","server"]},{"location":"api-ref/prefect/workers/server/#prefect.workers.server","title":"prefect.workers.server","text":"","tags":["Python API","workers","server"]},{"location":"api-ref/prefect/workers/server/#prefect.workers.server.start_healthcheck_server","title":"start_healthcheck_server","text":"

    Run a healthcheck FastAPI server for a worker.

    Parameters:

    Name Type Description Default worker BaseWorker | ProcessWorker

    the worker whose health we will check

    required log_level str

    the log level to use for the server

    'error' Source code in prefect/workers/server.py
    def start_healthcheck_server(\n    worker: Union[BaseWorker, ProcessWorker],\n    query_interval_seconds: int,\n    log_level: str = \"error\",\n) -> None:\n    \"\"\"\n    Run a healthcheck FastAPI server for a worker.\n\n    Args:\n        worker (BaseWorker | ProcessWorker): the worker whose health we will check\n        log_level (str): the log level to use for the server\n    \"\"\"\n    webserver = FastAPI()\n    router = APIRouter()\n\n    def perform_health_check():\n        did_recently_poll = worker.is_worker_still_polling(\n            query_interval_seconds=query_interval_seconds\n        )\n\n        if not did_recently_poll:\n            return JSONResponse(\n                status_code=status.HTTP_503_SERVICE_UNAVAILABLE,\n                content={\"message\": \"Worker may be unresponsive at this time\"},\n            )\n        return JSONResponse(status_code=status.HTTP_200_OK, content={\"message\": \"OK\"})\n\n    router.add_api_route(\"/health\", perform_health_check, methods=[\"GET\"])\n\n    webserver.include_router(router)\n\n    uvicorn.run(\n        webserver,\n        host=PREFECT_WORKER_WEBSERVER_HOST.value(),\n        port=PREFECT_WORKER_WEBSERVER_PORT.value(),\n        log_level=log_level,\n    )\n
    ","tags":["Python API","workers","server"]},{"location":"api-ref/prefect/workers/utilities/","title":"utilities","text":"","tags":["Python API","workers","utilities"]},{"location":"api-ref/prefect/workers/utilities/#prefect.workers.utilities","title":"prefect.workers.utilities","text":"","tags":["Python API","workers","utilities"]},{"location":"api-ref/python/","title":"Python SDK","text":"

    The Prefect Python SDK is used to build, test, and execute workflows against the Prefect API.

    Explore the modules in the navigation bar to the left to learn more.

    ","tags":["API","Python SDK"]},{"location":"api-ref/rest-api/","title":"REST API","text":"

    The Prefect REST API is used for communicating data from clients to a Prefect server so that orchestration can be performed. This API is consumed by clients such as the Prefect Python SDK or the server dashboard.

    Prefect Cloud and a locally hosted Prefect server each provide a REST API.

    • Prefect Cloud:
      • Interactive Prefect Cloud REST API documentation
      • Finding your Prefect Cloud details
    • A Locally hosted open-source Prefect server:
      • Interactive REST API documentation for a locally hosted open-source Prefect server is available at http://localhost:4200/docs or the /docs endpoint of the PREFECT_API_URL you have configured to access the server. You must have the server running with prefect server start to access the interactive documentation.
      • Prefect REST API documentation
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#interacting-with-the-rest-api","title":"Interacting with the REST API","text":"

    You have many options to interact with the Prefect REST API:

    • Create an instance of PrefectClient
    • Use your favorite Python HTTP library such as Requests or HTTPX
    • Use an HTTP library in your language of choice
    • Use curl from the command line
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#prefectclient-with-a-prefect-server","title":"PrefectClient with a Prefect server","text":"

    This example uses PrefectClient with a locally hosted Prefect server:

    import asyncio\nfrom prefect.client import get_client\n\nasync def get_flows():\n    client = get_client()\n    r = await client.read_flows(limit=5)\n    return r\n\nr = asyncio.run(get_flows())\n\nfor flow in r:\n    print(flow.name, flow.id)\n\nif __name__ == \"__main__\":\n    asyncio.run(get_flows())\n

    Output:

    cat-facts 58ed68b1-0201-4f37-adef-0ea24bd2a022\ndog-facts e7c0403d-44e7-45cf-a6c8-79117b7f3766\nsloth-facts 771c0574-f5bf-4f59-a69d-3be3e061a62d\ncapybara-facts fbadaf8b-584f-48b9-b092-07d351edd424\nlemur-facts 53f710e7-3b0f-4b2f-ab6b-44934111818c\n
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#requests-with-prefect","title":"Requests with Prefect","text":"

    This example uses the Requests library with Prefect Cloud to return the five newest artifacts.

    import requests\n\nPREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/abc-my-cloud-account-id-is-here/workspaces/123-my-workspace-id-is-here\"\nPREFECT_API_KEY=\"123abc_my_api_key_goes_here\"\ndata = {\n    \"sort\": \"CREATED_DESC\",\n    \"limit\": 5,\n    \"artifacts\": {\n        \"key\": {\n            \"exists_\": True\n        }\n    }\n}\n\nheaders = {\"Authorization\": f\"Bearer {PREFECT_API_KEY}\"}\nendpoint = f\"{PREFECT_API_URL}/artifacts/filter\"\n\nresponse = requests.post(endpoint, headers=headers, json=data)\nassert response.status_code == 200\nfor artifact in response.json():\n    print(artifact)\n
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#curl-with-prefect-cloud","title":"curl with Prefect Cloud","text":"

    This example uses curl with Prefect Cloud to create a flow run:

    ACCOUNT_ID=\"abc-my-cloud-account-id-goes-here\"\nWORKSPACE_ID=\"123-my-workspace-id-goes-here\"\nPREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/$ACCOUNT_ID/workspaces/$WORKSPACE_ID\"\nPREFECT_API_KEY=\"123abc_my_api_key_goes_here\"\nDEPLOYMENT_ID=\"my_deployment_id\"\n\ncurl --location --request POST \"$PREFECT_API_URL/deployments/$DEPLOYMENT_ID/create_flow_run\" \\\n  --header \"Content-Type: application/json\" \\\n  --header \"Authorization: Bearer $PREFECT_API_KEY\" \\\n  --header \"X-PREFECT-API-VERSION: 0.8.4\" \\\n  --data-raw \"{}\"\n

    Note that in this example --data-raw \"{}\" is required and is where you can specify other aspects of the flow run such as the state. Windows users substitute ^ for \\ for line multi-line commands.

    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#finding-your-prefect-cloud-details","title":"Finding your Prefect Cloud details","text":"

    When working with the Prefect Cloud REST API you will need your Account ID and often the Workspace ID for the workspace you want to interact with. You can find both IDs for a Prefect profile in the CLI with prefect profile inspect my_profile. This command will also display your Prefect API key, as shown below:

    PREFECT_API_URL='https://api.prefect.cloud/api/accounts/abc-my-account-id-is-here/workspaces/123-my-workspace-id-is-here'\nPREFECT_API_KEY='123abc_my_api_key_is_here'\n

    Alternatively, view your Account ID and Workspace ID in your browser URL. For example: https://app.prefect.cloud/account/abc-my-account-id-is-here/workspaces/123-my-workspace-id-is-here.

    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#rest-guidelines","title":"REST Guidelines","text":"

    The REST APIs adhere to the following guidelines:

    • Collection names are pluralized (for example, /flows or /runs).
    • We indicate variable placeholders with colons: GET /flows/:id.
    • We use snake case for route names: GET /task_runs.
    • We avoid nested resources unless there is no possibility of accessing the child resource outside the parent context. For example, we query /task_runs with a flow run filter instead of accessing /flow_runs/:id/task_runs.
    • The API is hosted with an /api/:version prefix that (optionally) allows versioning in the future. By convention, we treat that as part of the base URL and do not include that in API examples.
    • Filtering, sorting, and pagination parameters are provided in the request body of POST requests where applicable.
      • Pagination parameters are limit and offset.
      • Sorting is specified with a single sort parameter.
      • See more information on filtering below.
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#http-verbs","title":"HTTP verbs","text":"
    • GET, PUT and DELETE requests are always idempotent. POST and PATCH are not guaranteed to be idempotent.
    • GET requests cannot receive information from the request body.
    • POST requests can receive information from the request body.
    • POST /collection creates a new member of the collection.
    • GET /collection lists all members of the collection.
    • GET /collection/:id gets a specific member of the collection by ID.
    • DELETE /collection/:id deletes a specific member of the collection.
    • PUT /collection/:id creates or replaces a specific member of the collection.
    • PATCH /collection/:id partially updates a specific member of the collection.
    • POST /collection/action is how we implement non-CRUD actions. For example, to set a flow run's state, we use POST /flow_runs/:id/set_state.
    • POST /collection/action may also be used for read-only queries. This is to allow us to send complex arguments as body arguments (which often cannot be done via GET). Examples include POST /flow_runs/filter, POST /flow_runs/count, and POST /flow_runs/history.
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#filtering","title":"Filtering","text":"

    Objects can be filtered by providing filter criteria in the body of a POST request. When multiple criteria are specified, logical AND will be applied to the criteria.

    Filter criteria are structured as follows:

    {\n    \"objects\": {\n        \"object_field\": {\n            \"field_operator_\": <field_value>\n        }\n    }\n}\n

    In this example, objects is the name of the collection to filter over (for example, flows). The collection can be either the object being queried for (flows for POST /flows/filter) or a related object (flow_runs for POST /flows/filter).

    object_field is the name of the field over which to filter (name for flows). Note that some objects may have nested object fields, such as {flow_run: {state: {type: {any_: []}}}}.

    field_operator_ is the operator to apply to a field when filtering. Common examples include:

    • any_: return objects where this field matches any of the following values.
    • is_null_: return objects where this field is or is not null.
    • eq_: return objects where this field is equal to the following value.
    • all_: return objects where this field matches all of the following values.
    • before_: return objects where this datetime field is less than or equal to the following value.
    • after_: return objects where this datetime field is greater than or equal to the following value.

    For example, to query for flows with the tag \"database\" and failed flow runs, POST /flows/filter with the following request body:

    {\n    \"flows\": {\n        \"tags\": {\n            \"all_\": [\"database\"]\n        }\n    },\n    \"flow_runs\": {\n        \"state\": {\n            \"type\": {\n              \"any_\": [\"FAILED\"]\n            }\n        }\n    }\n}\n
    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/rest-api/#openapi","title":"OpenAPI","text":"

    The Prefect REST API can be fully described with an OpenAPI 3.0 compliant document. OpenAPI is a standard specification for describing REST APIs.

    To generate the Prefect server's complete OpenAPI document, run the following commands in an interactive Python session:

    from prefect.server.api.server import create_app\n\napp = create_app()\nopenapi_doc = app.openapi()\n

    This document allows you to generate your own API client, explore the API using an API inspection tool, or write tests to ensure API compliance.

    ","tags":["REST API","Prefect Cloud","Prefect server","curl","PrefectClient","Requests","API reference"]},{"location":"api-ref/server/","title":"Server API","text":"

    The Prefect server API is used by the server to work with workflow metadata and enforce orchestration logic. This API is primarily used by Prefect developers.

    Select links in the left navigation menu to explore.

    ","tags":["API","Server API"]},{"location":"api-ref/server/api/admin/","title":"server.api.admin","text":"","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin","title":"prefect.server.api.admin","text":"

    Routes for admin-level interactions with the Prefect REST API.

    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.clear_database","title":"clear_database async","text":"

    Clear all database tables without dropping them.

    Source code in prefect/server/api/admin.py
    @router.post(\"/database/clear\", status_code=status.HTTP_204_NO_CONTENT)\nasync def clear_database(\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    confirm: bool = Body(\n        False,\n        embed=True,\n        description=\"Pass confirm=True to confirm you want to modify the database.\",\n    ),\n    response: Response = None,\n):\n    \"\"\"Clear all database tables without dropping them.\"\"\"\n    if not confirm:\n        response.status_code = status.HTTP_400_BAD_REQUEST\n        return\n    async with db.session_context(begin_transaction=True) as session:\n        # work pool has a circular dependency on pool queue; delete it first\n        await session.execute(db.WorkPool.__table__.delete())\n        for table in reversed(db.Base.metadata.sorted_tables):\n            await session.execute(table.delete())\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.create_database","title":"create_database async","text":"

    Create all database objects.

    Source code in prefect/server/api/admin.py
    @router.post(\"/database/create\", status_code=status.HTTP_204_NO_CONTENT)\nasync def create_database(\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    confirm: bool = Body(\n        False,\n        embed=True,\n        description=\"Pass confirm=True to confirm you want to modify the database.\",\n    ),\n    response: Response = None,\n):\n    \"\"\"Create all database objects.\"\"\"\n    if not confirm:\n        response.status_code = status.HTTP_400_BAD_REQUEST\n        return\n\n    await db.create_db()\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.drop_database","title":"drop_database async","text":"

    Drop all database objects.

    Source code in prefect/server/api/admin.py
    @router.post(\"/database/drop\", status_code=status.HTTP_204_NO_CONTENT)\nasync def drop_database(\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    confirm: bool = Body(\n        False,\n        embed=True,\n        description=\"Pass confirm=True to confirm you want to modify the database.\",\n    ),\n    response: Response = None,\n):\n    \"\"\"Drop all database objects.\"\"\"\n    if not confirm:\n        response.status_code = status.HTTP_400_BAD_REQUEST\n        return\n\n    await db.drop_db()\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.read_settings","title":"read_settings async","text":"

    Get the current Prefect REST API settings.

    Secret setting values will be obfuscated.

    Source code in prefect/server/api/admin.py
    @router.get(\"/settings\")\nasync def read_settings() -> prefect.settings.Settings:\n    \"\"\"\n    Get the current Prefect REST API settings.\n\n    Secret setting values will be obfuscated.\n    \"\"\"\n    return prefect.settings.get_current_settings().with_obfuscated_secrets()\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/admin/#prefect.server.api.admin.read_version","title":"read_version async","text":"

    Returns the Prefect version number

    Source code in prefect/server/api/admin.py
    @router.get(\"/version\")\nasync def read_version() -> str:\n    \"\"\"Returns the Prefect version number\"\"\"\n    return prefect.__version__\n
    ","tags":["Prefect API","administration"]},{"location":"api-ref/server/api/dependencies/","title":"server.api.dependencies","text":"","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/dependencies/#prefect.server.api.dependencies","title":"prefect.server.api.dependencies","text":"

    Utilities for injecting FastAPI dependencies.

    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/dependencies/#prefect.server.api.dependencies.EnforceMinimumAPIVersion","title":"EnforceMinimumAPIVersion","text":"

    FastAPI Dependency used to check compatibility between the version of the api and a given request.

    Looks for the header 'X-PREFECT-API-VERSION' in the request and compares it to the api's version. Rejects requests that are lower than the minimum version.

    Source code in prefect/server/api/dependencies.py
    class EnforceMinimumAPIVersion:\n    \"\"\"\n    FastAPI Dependency used to check compatibility between the version of the api\n    and a given request.\n\n    Looks for the header 'X-PREFECT-API-VERSION' in the request and compares it\n    to the api's version. Rejects requests that are lower than the minimum version.\n    \"\"\"\n\n    def __init__(self, minimum_api_version: str, logger: logging.Logger):\n        self.minimum_api_version = minimum_api_version\n        versions = [int(v) for v in minimum_api_version.split(\".\")]\n        self.api_major = versions[0]\n        self.api_minor = versions[1]\n        self.api_patch = versions[2]\n        self.logger = logger\n\n    async def __call__(\n        self,\n        x_prefect_api_version: str = Header(None),\n    ):\n        request_version = x_prefect_api_version\n\n        # if no version header, assume latest and continue\n        if not request_version:\n            return\n\n        # parse version\n        try:\n            major, minor, patch = [int(v) for v in request_version.split(\".\")]\n        except ValueError:\n            await self._notify_of_invalid_value(request_version)\n            raise HTTPException(\n                status_code=status.HTTP_400_BAD_REQUEST,\n                detail=(\n                    \"Invalid X-PREFECT-API-VERSION header format.\"\n                    f\"Expected header in format 'x.y.z' but received {request_version}\"\n                ),\n            )\n\n        if (major, minor, patch) < (self.api_major, self.api_minor, self.api_patch):\n            await self._notify_of_outdated_version(request_version)\n            raise HTTPException(\n                status_code=status.HTTP_400_BAD_REQUEST,\n                detail=(\n                    f\"The request specified API version {request_version} but this \"\n                    f\"server requires version {self.minimum_api_version} or higher.\"\n                ),\n            )\n\n    async def _notify_of_invalid_value(self, request_version: str):\n        self.logger.error(\n            f\"Invalid X-PREFECT-API-VERSION header format: '{request_version}'\"\n        )\n\n    async def _notify_of_outdated_version(self, request_version: str):\n        self.logger.error(\n            f\"X-PREFECT-API-VERSION header specifies version '{request_version}' \"\n            f\"but minimum allowed version is '{self.minimum_api_version}'\"\n        )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/dependencies/#prefect.server.api.dependencies.LimitBody","title":"LimitBody","text":"

    A fastapi.Depends factory for pulling a limit: int parameter from the request body while determining the default from the current settings.

    Source code in prefect/server/api/dependencies.py
    def LimitBody() -> Depends:\n    \"\"\"\n    A `fastapi.Depends` factory for pulling a `limit: int` parameter from the\n    request body while determining the default from the current settings.\n    \"\"\"\n\n    def get_limit(\n        limit: int = Body(\n            None,\n            description=\"Defaults to PREFECT_API_DEFAULT_LIMIT if not provided.\",\n        )\n    ):\n        default_limit = PREFECT_API_DEFAULT_LIMIT.value()\n        limit = limit if limit is not None else default_limit\n        if not limit >= 0:\n            raise HTTPException(\n                status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,\n                detail=\"Invalid limit: must be greater than or equal to 0.\",\n            )\n        if limit > default_limit:\n            raise HTTPException(\n                status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,\n                detail=f\"Invalid limit: must be less than or equal to {default_limit}.\",\n            )\n        return limit\n\n    return Depends(get_limit)\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/deployments/","title":"server.api.deployments","text":"","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments","title":"prefect.server.api.deployments","text":"

    Routes for interacting with Deployment objects.

    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.count_deployments","title":"count_deployments async","text":"

    Count deployments.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/count\")\nasync def count_deployments(\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_pool_queues: schemas.filters.WorkQueueFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> int:\n    \"\"\"\n    Count deployments.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.deployments.count_deployments(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n        )\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.create_deployment","title":"create_deployment async","text":"

    Gracefully creates a new deployment from the provided schema. If a deployment with the same name and flow_id already exists, the deployment is updated.

    If the deployment has an active schedule, flow runs will be scheduled. When upserting, any scheduled runs from the existing deployment will be deleted.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/\")\nasync def create_deployment(\n    deployment: schemas.actions.DeploymentCreate,\n    response: Response,\n    worker_lookups: WorkerLookups = Depends(WorkerLookups),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.responses.DeploymentResponse:\n    \"\"\"\n    Gracefully creates a new deployment from the provided schema. If a deployment with\n    the same name and flow_id already exists, the deployment is updated.\n\n    If the deployment has an active schedule, flow runs will be scheduled.\n    When upserting, any scheduled runs from the existing deployment will be deleted.\n    \"\"\"\n\n    async with db.session_context(begin_transaction=True) as session:\n        if (\n            deployment.work_pool_name\n            and deployment.work_pool_name != DEFAULT_AGENT_WORK_POOL_NAME\n        ):\n            # Make sure that deployment is valid before beginning creation process\n            work_pool = await models.workers.read_work_pool_by_name(\n                session=session, work_pool_name=deployment.work_pool_name\n            )\n            if work_pool is None:\n                raise HTTPException(\n                    status_code=status.HTTP_404_NOT_FOUND,\n                    detail=f'Work pool \"{deployment.work_pool_name}\" not found.',\n                )\n            try:\n                deployment.check_valid_configuration(work_pool.base_job_template)\n            except (MissingVariableError, jsonschema.exceptions.ValidationError) as exc:\n                raise HTTPException(\n                    status_code=status.HTTP_409_CONFLICT,\n                    detail=f\"Error creating deployment: {exc!r}\",\n                )\n\n        # hydrate the input model into a full model\n        deployment_dict = deployment.dict(exclude={\"work_pool_name\"})\n        if deployment.work_pool_name and deployment.work_queue_name:\n            # If a specific pool name/queue name combination was provided, get the\n            # ID for that work pool queue.\n            deployment_dict[\"work_queue_id\"] = (\n                await worker_lookups._get_work_queue_id_from_name(\n                    session=session,\n                    work_pool_name=deployment.work_pool_name,\n                    work_queue_name=deployment.work_queue_name,\n                    create_queue_if_not_found=True,\n                )\n            )\n        elif deployment.work_pool_name:\n            # If just a pool name was provided, get the ID for its default\n            # work pool queue.\n            deployment_dict[\"work_queue_id\"] = (\n                await worker_lookups._get_default_work_queue_id_from_work_pool_name(\n                    session=session,\n                    work_pool_name=deployment.work_pool_name,\n                )\n            )\n        elif deployment.work_queue_name:\n            # If just a queue name was provided, ensure that the queue exists and\n            # get its ID.\n            work_queue = await models.work_queues._ensure_work_queue_exists(\n                session=session, name=deployment.work_queue_name\n            )\n            deployment_dict[\"work_queue_id\"] = work_queue.id\n\n        deployment = schemas.core.Deployment(**deployment_dict)\n        # check to see if relevant blocks exist, allowing us throw a useful error message\n        # for debugging\n        if deployment.infrastructure_document_id is not None:\n            infrastructure_block = (\n                await models.block_documents.read_block_document_by_id(\n                    session=session,\n                    block_document_id=deployment.infrastructure_document_id,\n                )\n            )\n            if not infrastructure_block:\n                raise HTTPException(\n                    status_code=status.HTTP_409_CONFLICT,\n                    detail=(\n                        \"Error creating deployment. Could not find infrastructure\"\n                        f\" block with id: {deployment.infrastructure_document_id}. This\"\n                        \" usually occurs when applying a deployment specification that\"\n                        \" was built against a different Prefect database / workspace.\"\n                    ),\n                )\n\n        if deployment.storage_document_id is not None:\n            infrastructure_block = (\n                await models.block_documents.read_block_document_by_id(\n                    session=session,\n                    block_document_id=deployment.storage_document_id,\n                )\n            )\n            if not infrastructure_block:\n                raise HTTPException(\n                    status_code=status.HTTP_409_CONFLICT,\n                    detail=(\n                        \"Error creating deployment. Could not find storage block with\"\n                        f\" id: {deployment.storage_document_id}. This usually occurs\"\n                        \" when applying a deployment specification that was built\"\n                        \" against a different Prefect database / workspace.\"\n                    ),\n                )\n\n        now = pendulum.now(\"UTC\")\n        model = await models.deployments.create_deployment(\n            session=session, deployment=deployment\n        )\n\n        if model.created >= now:\n            response.status_code = status.HTTP_201_CREATED\n\n        return schemas.responses.DeploymentResponse.from_orm(model)\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.create_flow_run_from_deployment","title":"create_flow_run_from_deployment async","text":"

    Create a flow run from a deployment.

    Any parameters not provided will be inferred from the deployment's parameters. If tags are not provided, the deployment's tags will be used.

    If no state is provided, the flow run will be created in a SCHEDULED state.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/{id}/create_flow_run\")\nasync def create_flow_run_from_deployment(\n    flow_run: schemas.actions.DeploymentFlowRunCreate,\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    worker_lookups: WorkerLookups = Depends(WorkerLookups),\n    response: Response = None,\n) -> schemas.responses.FlowRunResponse:\n    \"\"\"\n    Create a flow run from a deployment.\n\n    Any parameters not provided will be inferred from the deployment's parameters.\n    If tags are not provided, the deployment's tags will be used.\n\n    If no state is provided, the flow run will be created in a SCHEDULED state.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        # get relevant info from the deployment\n        deployment = await models.deployments.read_deployment(\n            session=session, deployment_id=deployment_id\n        )\n\n        if not deployment:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n\n        parameters = deployment.parameters\n        parameters.update(flow_run.parameters or {})\n\n        if deployment.enforce_parameter_schema:\n            if not isinstance(deployment.parameter_openapi_schema, dict):\n                raise HTTPException(\n                    status.HTTP_409_CONFLICT,\n                    detail=(\n                        \"Error updating deployment: Cannot update parameters because\"\n                        \" parameter schema enforcement is enabled and the deployment\"\n                        \" does not have a valid parameter schema.\"\n                    ),\n                )\n            try:\n                validate_values_conform_to_schema(\n                    parameters, deployment.parameter_openapi_schema\n                )\n            except ValueError as exc:\n                raise HTTPException(\n                    status.HTTP_409_CONFLICT, detail=f\"Error creating flow run: {exc}\"\n                )\n\n        work_queue_name = deployment.work_queue_name\n        work_queue_id = deployment.work_queue_id\n\n        if flow_run.work_queue_name:\n            # can't mutate the ORM model or else it will commit the changes back\n            work_queue_id = await worker_lookups._get_work_queue_id_from_name(\n                session=session,\n                work_pool_name=deployment.work_queue.work_pool.name,\n                work_queue_name=flow_run.work_queue_name,\n                create_queue_if_not_found=True,\n            )\n            work_queue_name = flow_run.work_queue_name\n\n        # hydrate the input model into a full flow run / state model\n        flow_run = schemas.core.FlowRun(\n            **flow_run.dict(\n                exclude={\n                    \"parameters\",\n                    \"tags\",\n                    \"infrastructure_document_id\",\n                    \"work_queue_name\",\n                }\n            ),\n            flow_id=deployment.flow_id,\n            deployment_id=deployment.id,\n            parameters=parameters,\n            tags=set(deployment.tags).union(flow_run.tags),\n            infrastructure_document_id=(\n                flow_run.infrastructure_document_id\n                or deployment.infrastructure_document_id\n            ),\n            work_queue_name=work_queue_name,\n            work_queue_id=work_queue_id,\n        )\n\n        if not flow_run.state:\n            flow_run.state = schemas.states.Scheduled()\n\n        now = pendulum.now(\"UTC\")\n        model = await models.flow_runs.create_flow_run(\n            session=session, flow_run=flow_run\n        )\n        if model.created >= now:\n            response.status_code = status.HTTP_201_CREATED\n        return schemas.responses.FlowRunResponse.from_orm(model)\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.delete_deployment","title":"delete_deployment async","text":"

    Delete a deployment by id.

    Source code in prefect/server/api/deployments.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_deployment(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a deployment by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.deployments.delete_deployment(\n            session=session, deployment_id=deployment_id\n        )\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n        )\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.get_scheduled_flow_runs_for_deployments","title":"get_scheduled_flow_runs_for_deployments async","text":"

    Get scheduled runs for a set of deployments. Used by a runner to poll for work.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/get_scheduled_flow_runs\")\nasync def get_scheduled_flow_runs_for_deployments(\n    deployment_ids: List[UUID] = Body(\n        default=..., description=\"The deployment IDs to get scheduled runs for\"\n    ),\n    scheduled_before: DateTimeTZ = Body(\n        None, description=\"The maximum time to look for scheduled flow runs\"\n    ),\n    limit: int = dependencies.LimitBody(),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.FlowRunResponse]:\n    \"\"\"\n    Get scheduled runs for a set of deployments. Used by a runner to poll for work.\n    \"\"\"\n    async with db.session_context() as session:\n        orm_flow_runs = await models.flow_runs.read_flow_runs(\n            session=session,\n            limit=limit,\n            deployment_filter=schemas.filters.DeploymentFilter(\n                id=schemas.filters.DeploymentFilterId(any_=deployment_ids),\n            ),\n            flow_run_filter=schemas.filters.FlowRunFilter(\n                next_scheduled_start_time=schemas.filters.FlowRunFilterNextScheduledStartTime(\n                    before_=scheduled_before\n                ),\n                state=schemas.filters.FlowRunFilterState(\n                    type=schemas.filters.FlowRunFilterStateType(\n                        any_=[schemas.states.StateType.SCHEDULED]\n                    )\n                ),\n            ),\n            sort=schemas.sorting.FlowRunSort.NEXT_SCHEDULED_START_TIME_ASC,\n        )\n\n        flow_run_responses = [\n            schemas.responses.FlowRunResponse.from_orm(orm_flow_run=orm_flow_run)\n            for orm_flow_run in orm_flow_runs\n        ]\n\n    async with db.session_context(\n        begin_transaction=True, with_for_update=True\n    ) as session:\n        await models.deployments._update_deployment_last_polled(\n            session=session, deployment_ids=deployment_ids\n        )\n\n    return flow_run_responses\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.read_deployment","title":"read_deployment async","text":"

    Get a deployment by id.

    Source code in prefect/server/api/deployments.py
    @router.get(\"/{id}\")\nasync def read_deployment(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.responses.DeploymentResponse:\n    \"\"\"\n    Get a deployment by id.\n    \"\"\"\n    async with db.session_context() as session:\n        deployment = await models.deployments.read_deployment(\n            session=session, deployment_id=deployment_id\n        )\n        if not deployment:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n        return schemas.responses.DeploymentResponse.from_orm(deployment)\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.read_deployment_by_name","title":"read_deployment_by_name async","text":"

    Get a deployment using the name of the flow and the deployment.

    Source code in prefect/server/api/deployments.py
    @router.get(\"/name/{flow_name}/{deployment_name}\")\nasync def read_deployment_by_name(\n    flow_name: str = Path(..., description=\"The name of the flow\"),\n    deployment_name: str = Path(..., description=\"The name of the deployment\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.responses.DeploymentResponse:\n    \"\"\"\n    Get a deployment using the name of the flow and the deployment.\n    \"\"\"\n    async with db.session_context() as session:\n        deployment = await models.deployments.read_deployment_by_name(\n            session=session, name=deployment_name, flow_name=flow_name\n        )\n        if not deployment:\n            raise HTTPException(\n                status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n        return schemas.responses.DeploymentResponse.from_orm(deployment)\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.read_deployments","title":"read_deployments async","text":"

    Query for deployments.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/filter\")\nasync def read_deployments(\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_pool_queues: schemas.filters.WorkQueueFilter = None,\n    sort: schemas.sorting.DeploymentSort = Body(\n        schemas.sorting.DeploymentSort.NAME_ASC\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.DeploymentResponse]:\n    \"\"\"\n    Query for deployments.\n    \"\"\"\n    async with db.session_context() as session:\n        response = await models.deployments.read_deployments(\n            session=session,\n            offset=offset,\n            sort=sort,\n            limit=limit,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n        )\n        return [\n            schemas.responses.DeploymentResponse.from_orm(orm_deployment=deployment)\n            for deployment in response\n        ]\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.schedule_deployment","title":"schedule_deployment async","text":"

    Schedule runs for a deployment. For backfills, provide start/end times in the past.

    This function will generate the minimum number of runs that satisfy the min and max times, and the min and max counts. Specifically, the following order will be respected.

    - Runs will be generated starting on or after the `start_time`\n- No more than `max_runs` runs will be generated\n- No runs will be generated after `end_time` is reached\n- At least `min_runs` runs will be generated\n- Runs will be generated until at least `start_time + min_time` is reached\n
    Source code in prefect/server/api/deployments.py
    @router.post(\"/{id}/schedule\")\nasync def schedule_deployment(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    start_time: DateTimeTZ = Body(None, description=\"The earliest date to schedule\"),\n    end_time: DateTimeTZ = Body(None, description=\"The latest date to schedule\"),\n    min_time: datetime.timedelta = Body(\n        None,\n        description=(\n            \"Runs will be scheduled until at least this long after the `start_time`\"\n        ),\n    ),\n    min_runs: int = Body(None, description=\"The minimum number of runs to schedule\"),\n    max_runs: int = Body(None, description=\"The maximum number of runs to schedule\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> None:\n    \"\"\"\n    Schedule runs for a deployment. For backfills, provide start/end times in the past.\n\n    This function will generate the minimum number of runs that satisfy the min\n    and max times, and the min and max counts. Specifically, the following order\n    will be respected.\n\n        - Runs will be generated starting on or after the `start_time`\n        - No more than `max_runs` runs will be generated\n        - No runs will be generated after `end_time` is reached\n        - At least `min_runs` runs will be generated\n        - Runs will be generated until at least `start_time + min_time` is reached\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        await models.deployments.schedule_runs(\n            session=session,\n            deployment_id=deployment_id,\n            start_time=start_time,\n            min_time=min_time,\n            end_time=end_time,\n            min_runs=min_runs,\n            max_runs=max_runs,\n        )\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.set_schedule_active","title":"set_schedule_active async","text":"

    Set a deployment schedule to active. Runs will be scheduled immediately.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/{id}/set_schedule_active\")\nasync def set_schedule_active(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> None:\n    \"\"\"\n    Set a deployment schedule to active. Runs will be scheduled immediately.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        deployment = await models.deployments.read_deployment(\n            session=session, deployment_id=deployment_id\n        )\n        if not deployment:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n        deployment.is_schedule_active = True\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.set_schedule_inactive","title":"set_schedule_inactive async","text":"

    Set a deployment schedule to inactive. Any auto-scheduled runs still in a Scheduled state will be deleted.

    Source code in prefect/server/api/deployments.py
    @router.post(\"/{id}/set_schedule_inactive\")\nasync def set_schedule_inactive(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> None:\n    \"\"\"\n    Set a deployment schedule to inactive. Any auto-scheduled runs still in a Scheduled\n    state will be deleted.\n    \"\"\"\n    async with db.session_context(begin_transaction=False) as session:\n        deployment = await models.deployments.read_deployment(\n            session=session, deployment_id=deployment_id\n        )\n        if not deployment:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n            )\n        deployment.is_schedule_active = False\n        # commit here to make the inactive schedule \"visible\" to the scheduler service\n        await session.commit()\n\n        # delete any auto scheduled runs\n        await models.deployments._delete_scheduled_runs(\n            session=session,\n            deployment_id=deployment_id,\n            db=db,\n            auto_scheduled_only=True,\n        )\n\n        await session.commit()\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/deployments/#prefect.server.api.deployments.work_queue_check_for_deployment","title":"work_queue_check_for_deployment async","text":"

    Get list of work-queues that are able to pick up the specified deployment.

    This endpoint is intended to be used by the UI to provide users warnings about deployments that are unable to be executed because there are no work queues that will pick up their runs, based on existing filter criteria. It may be deprecated in the future because there is not a strict relationship between work queues and deployments.

    Source code in prefect/server/api/deployments.py
    @router.get(\"/{id}/work_queue_check\", deprecated=True)\nasync def work_queue_check_for_deployment(\n    deployment_id: UUID = Path(..., description=\"The deployment id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.WorkQueue]:\n    \"\"\"\n    Get list of work-queues that are able to pick up the specified deployment.\n\n    This endpoint is intended to be used by the UI to provide users warnings\n    about deployments that are unable to be executed because there are no work\n    queues that will pick up their runs, based on existing filter criteria. It\n    may be deprecated in the future because there is not a strict relationship\n    between work queues and deployments.\n    \"\"\"\n    try:\n        async with db.session_context() as session:\n            work_queues = await models.deployments.check_work_queues_for_deployment(\n                session=session, deployment_id=deployment_id\n            )\n    except ObjectNotFoundError:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Deployment not found\"\n        )\n    return work_queues\n
    ","tags":["Prefect API","deployments"]},{"location":"api-ref/server/api/flow_run_states/","title":"server.api.flow_run_states","text":"","tags":["Prefect API","flow runs","states"]},{"location":"api-ref/server/api/flow_run_states/#prefect.server.api.flow_run_states","title":"prefect.server.api.flow_run_states","text":"

    Routes for interacting with flow run state objects.

    ","tags":["Prefect API","flow runs","states"]},{"location":"api-ref/server/api/flow_run_states/#prefect.server.api.flow_run_states.read_flow_run_state","title":"read_flow_run_state async","text":"

    Get a flow run state by id.

    Source code in prefect/server/api/flow_run_states.py
    @router.get(\"/{id}\")\nasync def read_flow_run_state(\n    flow_run_state_id: UUID = Path(\n        ..., description=\"The flow run state id\", alias=\"id\"\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.states.State:\n    \"\"\"\n    Get a flow run state by id.\n    \"\"\"\n    async with db.session_context() as session:\n        flow_run_state = await models.flow_run_states.read_flow_run_state(\n            session=session, flow_run_state_id=flow_run_state_id\n        )\n    if not flow_run_state:\n        raise HTTPException(\n            status.HTTP_404_NOT_FOUND, detail=\"Flow run state not found\"\n        )\n    return flow_run_state\n
    ","tags":["Prefect API","flow runs","states"]},{"location":"api-ref/server/api/flow_run_states/#prefect.server.api.flow_run_states.read_flow_run_states","title":"read_flow_run_states async","text":"

    Get states associated with a flow run.

    Source code in prefect/server/api/flow_run_states.py
    @router.get(\"/\")\nasync def read_flow_run_states(\n    flow_run_id: UUID,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.states.State]:\n    \"\"\"\n    Get states associated with a flow run.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flow_run_states.read_flow_run_states(\n            session=session, flow_run_id=flow_run_id\n        )\n
    ","tags":["Prefect API","flow runs","states"]},{"location":"api-ref/server/api/flow_runs/","title":"server.api.flow_runs","text":"","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs","title":"prefect.server.api.flow_runs","text":"

    Routes for interacting with flow run objects.

    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.average_flow_run_lateness","title":"average_flow_run_lateness async","text":"

    Query for average flow-run lateness in seconds.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/lateness\")\nasync def average_flow_run_lateness(\n    flows: Optional[schemas.filters.FlowFilter] = None,\n    flow_runs: Optional[schemas.filters.FlowRunFilter] = None,\n    task_runs: Optional[schemas.filters.TaskRunFilter] = None,\n    deployments: Optional[schemas.filters.DeploymentFilter] = None,\n    work_pools: Optional[schemas.filters.WorkPoolFilter] = None,\n    work_pool_queues: Optional[schemas.filters.WorkQueueFilter] = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> Optional[float]:\n    \"\"\"\n    Query for average flow-run lateness in seconds.\n    \"\"\"\n    async with db.session_context() as session:\n        if db.dialect.name == \"sqlite\":\n            # Since we want an _average_ of the lateness we're unable to use\n            # the existing FlowRun.expected_start_time_delta property as it\n            # returns a timedelta and SQLite is unable to properly deal with it\n            # and always returns 1970.0 as the average. This copies the same\n            # logic but ensures that it returns the number of seconds instead\n            # so it's compatible with SQLite.\n            base_query = sa.case(\n                (\n                    db.FlowRun.start_time > db.FlowRun.expected_start_time,\n                    sa.func.strftime(\"%s\", db.FlowRun.start_time)\n                    - sa.func.strftime(\"%s\", db.FlowRun.expected_start_time),\n                ),\n                (\n                    db.FlowRun.start_time.is_(None)\n                    & db.FlowRun.state_type.notin_(schemas.states.TERMINAL_STATES)\n                    & (db.FlowRun.expected_start_time < sa.func.datetime(\"now\")),\n                    sa.func.strftime(\"%s\", sa.func.datetime(\"now\"))\n                    - sa.func.strftime(\"%s\", db.FlowRun.expected_start_time),\n                ),\n                else_=0,\n            )\n        else:\n            base_query = db.FlowRun.estimated_start_time_delta\n\n        query = await models.flow_runs._apply_flow_run_filters(\n            sa.select(sa.func.avg(base_query)),\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n        )\n        result = await session.execute(query)\n\n        avg_lateness = result.scalar()\n\n        if avg_lateness is None:\n            return None\n        elif isinstance(avg_lateness, datetime.timedelta):\n            return avg_lateness.total_seconds()\n        else:\n            return avg_lateness\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.count_flow_runs","title":"count_flow_runs async","text":"

    Query for flow runs.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/count\")\nasync def count_flow_runs(\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_pool_queues: schemas.filters.WorkQueueFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> int:\n    \"\"\"\n    Query for flow runs.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flow_runs.count_flow_runs(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.create_flow_run","title":"create_flow_run async","text":"

    Create a flow run. If a flow run with the same flow_id and idempotency key already exists, the existing flow run will be returned.

    If no state is provided, the flow run will be created in a PENDING state.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/\")\nasync def create_flow_run(\n    flow_run: schemas.actions.FlowRunCreate,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    response: Response = None,\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_flow_orchestration_parameters\n    ),\n    api_version=Depends(dependencies.provide_request_api_version),\n) -> schemas.responses.FlowRunResponse:\n    \"\"\"\n    Create a flow run. If a flow run with the same flow_id and\n    idempotency key already exists, the existing flow run will be returned.\n\n    If no state is provided, the flow run will be created in a PENDING state.\n    \"\"\"\n    # hydrate the input model into a full flow run / state model\n    flow_run = schemas.core.FlowRun(**flow_run.dict())\n\n    # pass the request version to the orchestration engine to support compatibility code\n    orchestration_parameters.update({\"api-version\": api_version})\n\n    if not flow_run.state:\n        flow_run.state = schemas.states.Pending()\n\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        model = await models.flow_runs.create_flow_run(\n            session=session,\n            flow_run=flow_run,\n            orchestration_parameters=orchestration_parameters,\n        )\n        if model.created >= now:\n            response.status_code = status.HTTP_201_CREATED\n\n        return schemas.responses.FlowRunResponse.from_orm(model)\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.create_flow_run_input","title":"create_flow_run_input async","text":"

    Create a key/value input for a flow run.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/{id}/input\", status_code=status.HTTP_201_CREATED)\nasync def create_flow_run_input(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    key: str = Body(..., description=\"The input key\"),\n    value: bytes = Body(..., description=\"The value of the input\"),\n    sender: Optional[str] = Body(None, description=\"The sender of the input\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Create a key/value input for a flow run.\n    \"\"\"\n    async with db.session_context() as session:\n        try:\n            await models.flow_run_input.create_flow_run_input(\n                session=session,\n                flow_run_input=schemas.core.FlowRunInput(\n                    flow_run_id=flow_run_id,\n                    key=key,\n                    sender=sender,\n                    value=value.decode(),\n                ),\n            )\n            await session.commit()\n\n        except IntegrityError as exc:\n            if \"unique constraint\" in str(exc).lower():\n                raise HTTPException(\n                    status_code=status.HTTP_409_CONFLICT,\n                    detail=\"A flow run input with this key already exists.\",\n                )\n            else:\n                raise HTTPException(\n                    status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run not found\"\n                )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.delete_flow_run","title":"delete_flow_run async","text":"

    Delete a flow run by id.

    Source code in prefect/server/api/flow_runs.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_flow_run(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a flow run by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.flow_runs.delete_flow_run(\n            session=session, flow_run_id=flow_run_id\n        )\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run not found\"\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.delete_flow_run_input","title":"delete_flow_run_input async","text":"

    Delete a flow run input

    Source code in prefect/server/api/flow_runs.py
    @router.delete(\"/{id}/input/{key}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_flow_run_input(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    key: str = Path(..., description=\"The input key\", alias=\"key\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a flow run input\n    \"\"\"\n\n    async with db.session_context() as session:\n        deleted = await models.flow_run_input.delete_flow_run_input(\n            session=session, flow_run_id=flow_run_id, key=key\n        )\n        await session.commit()\n\n        if not deleted:\n            raise HTTPException(\n                status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run input not found\"\n            )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.filter_flow_run_input","title":"filter_flow_run_input async","text":"

    Filter flow run inputs by key prefix

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/{id}/input/filter\")\nasync def filter_flow_run_input(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    prefix: str = Body(..., description=\"The input key prefix\", embed=True),\n    limit: int = Body(\n        1, description=\"The maximum number of results to return\", embed=True\n    ),\n    exclude_keys: List[str] = Body(\n        [], description=\"Exclude inputs with these keys\", embed=True\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.FlowRunInput]:\n    \"\"\"\n    Filter flow run inputs by key prefix\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flow_run_input.filter_flow_run_input(\n            session=session,\n            flow_run_id=flow_run_id,\n            prefix=prefix,\n            limit=limit,\n            exclude_keys=exclude_keys,\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.flow_run_history","title":"flow_run_history async","text":"

    Query for flow run history data across a given range and interval.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/history\")\nasync def flow_run_history(\n    history_start: DateTimeTZ = Body(..., description=\"The history's start time.\"),\n    history_end: DateTimeTZ = Body(..., description=\"The history's end time.\"),\n    history_interval: datetime.timedelta = Body(\n        ...,\n        description=(\n            \"The size of each history interval, in seconds. Must be at least 1 second.\"\n        ),\n        alias=\"history_interval_seconds\",\n    ),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_queues: schemas.filters.WorkQueueFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.HistoryResponse]:\n    \"\"\"\n    Query for flow run history data across a given range and interval.\n    \"\"\"\n    if history_interval < datetime.timedelta(seconds=1):\n        raise HTTPException(\n            status.HTTP_422_UNPROCESSABLE_ENTITY,\n            detail=\"History interval must not be less than 1 second.\",\n        )\n\n    async with db.session_context() as session:\n        return await run_history(\n            session=session,\n            run_type=\"flow_run\",\n            history_start=history_start,\n            history_end=history_end,\n            history_interval=history_interval,\n            flows=flows,\n            flow_runs=flow_runs,\n            task_runs=task_runs,\n            deployments=deployments,\n            work_pools=work_pools,\n            work_queues=work_queues,\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_run","title":"read_flow_run async","text":"

    Get a flow run by id.

    Source code in prefect/server/api/flow_runs.py
    @router.get(\"/{id}\")\nasync def read_flow_run(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.responses.FlowRunResponse:\n    \"\"\"\n    Get a flow run by id.\n    \"\"\"\n    async with db.session_context() as session:\n        flow_run = await models.flow_runs.read_flow_run(\n            session=session, flow_run_id=flow_run_id\n        )\n        if not flow_run:\n            raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Flow run not found\")\n        return schemas.responses.FlowRunResponse.from_orm(flow_run)\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_run_graph_v1","title":"read_flow_run_graph_v1 async","text":"

    Get a task run dependency map for a given flow run.

    Source code in prefect/server/api/flow_runs.py
    @router.get(\"/{id}/graph\")\nasync def read_flow_run_graph_v1(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[DependencyResult]:\n    \"\"\"\n    Get a task run dependency map for a given flow run.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flow_runs.read_task_run_dependencies(\n            session=session, flow_run_id=flow_run_id\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_run_graph_v2","title":"read_flow_run_graph_v2 async","text":"

    Get a graph of the tasks and subflow runs for the given flow run

    Source code in prefect/server/api/flow_runs.py
    @router.get(\"/{id:uuid}/graph-v2\")\nasync def read_flow_run_graph_v2(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    since: datetime.datetime = Query(\n        datetime.datetime.min,\n        description=\"Only include runs that start or end after this time.\",\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> Graph:\n    \"\"\"\n    Get a graph of the tasks and subflow runs for the given flow run\n    \"\"\"\n    async with db.session_context() as session:\n        try:\n            return await read_flow_run_graph(\n                session=session,\n                flow_run_id=flow_run_id,\n                since=since,\n            )\n        except FlowRunGraphTooLarge as e:\n            raise HTTPException(\n                status_code=status.HTTP_400_BAD_REQUEST,\n                detail=str(e),\n            )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_run_input","title":"read_flow_run_input async","text":"

    Create a value from a flow run input

    Source code in prefect/server/api/flow_runs.py
    @router.get(\"/{id}/input/{key}\")\nasync def read_flow_run_input(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    key: str = Path(..., description=\"The input key\", alias=\"key\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> PlainTextResponse:\n    \"\"\"\n    Create a value from a flow run input\n    \"\"\"\n\n    async with db.session_context() as session:\n        flow_run_input = await models.flow_run_input.read_flow_run_input(\n            session=session, flow_run_id=flow_run_id, key=key\n        )\n\n    if flow_run_input:\n        return PlainTextResponse(flow_run_input.value)\n    else:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run input not found\"\n        )\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.read_flow_runs","title":"read_flow_runs async","text":"

    Query for flow runs.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/filter\", response_class=ORJSONResponse)\nasync def read_flow_runs(\n    sort: schemas.sorting.FlowRunSort = Body(schemas.sorting.FlowRunSort.ID_DESC),\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_pool_queues: schemas.filters.WorkQueueFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.FlowRunResponse]:\n    \"\"\"\n    Query for flow runs.\n    \"\"\"\n    async with db.session_context() as session:\n        db_flow_runs = await models.flow_runs.read_flow_runs(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_pool_queues,\n            offset=offset,\n            limit=limit,\n            sort=sort,\n        )\n\n        # Instead of relying on fastapi.encoders.jsonable_encoder to convert the\n        # response to JSON, we do so more efficiently ourselves.\n        # In particular, the FastAPI encoder is very slow for large, nested objects.\n        # See: https://github.com/tiangolo/fastapi/issues/1224\n        encoded = [\n            schemas.responses.FlowRunResponse.from_orm(fr).dict(json_compatible=True)\n            for fr in db_flow_runs\n        ]\n        return ORJSONResponse(content=encoded)\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.resume_flow_run","title":"resume_flow_run async","text":"

    Resume a paused flow run.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/{id}/resume\")\nasync def resume_flow_run(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    run_input: Optional[Dict] = Body(default=None, embed=True),\n    response: Response = None,\n    flow_policy: BaseOrchestrationPolicy = Depends(\n        orchestration_dependencies.provide_flow_policy\n    ),\n    task_policy: BaseOrchestrationPolicy = Depends(\n        orchestration_dependencies.provide_task_policy\n    ),\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_flow_orchestration_parameters\n    ),\n    api_version=Depends(dependencies.provide_request_api_version),\n) -> OrchestrationResult:\n    \"\"\"\n    Resume a paused flow run.\n    \"\"\"\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        flow_run = await models.flow_runs.read_flow_run(session, flow_run_id)\n        state = flow_run.state\n\n        if state is None or state.type != schemas.states.StateType.PAUSED:\n            result = OrchestrationResult(\n                state=None,\n                status=schemas.responses.SetStateStatus.ABORT,\n                details=schemas.responses.StateAbortDetails(\n                    reason=\"Cannot resume a flow run that is not paused.\"\n                ),\n            )\n            return result\n\n        orchestration_parameters.update({\"api-version\": api_version})\n\n        keyset = state.state_details.run_input_keyset\n        if keyset and not run_input:\n            return OrchestrationResult(\n                state=state,\n                status=schemas.responses.SetStateStatus.REJECT,\n                details=schemas.responses.StateAbortDetails(\n                    reason=\"Flow run was expecting input but none was provided.\"\n                ),\n            )\n        elif keyset and run_input:\n            schema_json = await models.flow_run_input.read_flow_run_input(\n                session=session, flow_run_id=flow_run.id, key=keyset[\"schema\"]\n            )\n\n            if schema_json is None:\n                return OrchestrationResult(\n                    state=state,\n                    status=schemas.responses.SetStateStatus.REJECT,\n                    details=schemas.responses.StateAbortDetails(\n                        reason=\"Run input schema not found.\"\n                    ),\n                )\n\n            try:\n                schema = orjson.loads(schema_json.value)\n            except orjson.JSONDecodeError:\n                return OrchestrationResult(\n                    state=state,\n                    status=schemas.responses.SetStateStatus.REJECT,\n                    details=schemas.responses.StateAbortDetails(\n                        reason=\"Run input schema is not valid JSON.\"\n                    ),\n                )\n\n            try:\n                jsonschema.validate(run_input, schema)\n            except (jsonschema.ValidationError, jsonschema.SchemaError) as exc:\n                return OrchestrationResult(\n                    state=state,\n                    status=schemas.responses.SetStateStatus.REJECT,\n                    details=schemas.responses.StateAbortDetails(\n                        reason=f\"Run input validation failed: {exc.message}\"\n                    ),\n                )\n\n        if state.state_details.pause_reschedule:\n            orchestration_result = await models.flow_runs.set_flow_run_state(\n                session=session,\n                flow_run_id=flow_run_id,\n                state=schemas.states.Scheduled(\n                    name=\"Resuming\", scheduled_time=pendulum.now(\"UTC\")\n                ),\n                flow_policy=flow_policy,\n                orchestration_parameters=orchestration_parameters,\n            )\n        else:\n            orchestration_result = await models.flow_runs.set_flow_run_state(\n                session=session,\n                flow_run_id=flow_run_id,\n                state=schemas.states.Running(),\n                flow_policy=flow_policy,\n                orchestration_parameters=orchestration_parameters,\n            )\n\n        if (\n            keyset\n            and run_input\n            and orchestration_result.status == schemas.responses.SetStateStatus.ACCEPT\n        ):\n            # The state change is accepted, go ahead and store the validated\n            # run input.\n            await models.flow_run_input.create_flow_run_input(\n                session=session,\n                flow_run_input=schemas.core.FlowRunInput(\n                    flow_run_id=flow_run_id,\n                    key=keyset[\"response\"],\n                    value=orjson.dumps(run_input).decode(\"utf-8\"),\n                ),\n            )\n\n        # set the 201 if a new state was created\n        if orchestration_result.state and orchestration_result.state.timestamp >= now:\n            response.status_code = status.HTTP_201_CREATED\n        else:\n            response.status_code = status.HTTP_200_OK\n\n        return orchestration_result\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.set_flow_run_state","title":"set_flow_run_state async","text":"

    Set a flow run state, invoking any orchestration rules.

    Source code in prefect/server/api/flow_runs.py
    @router.post(\"/{id}/set_state\")\nasync def set_flow_run_state(\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    state: schemas.actions.StateCreate = Body(..., description=\"The intended state.\"),\n    force: bool = Body(\n        False,\n        description=(\n            \"If false, orchestration rules will be applied that may alter or prevent\"\n            \" the state transition. If True, orchestration rules are not applied.\"\n        ),\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    response: Response = None,\n    flow_policy: BaseOrchestrationPolicy = Depends(\n        orchestration_dependencies.provide_flow_policy\n    ),\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_flow_orchestration_parameters\n    ),\n    api_version=Depends(dependencies.provide_request_api_version),\n) -> OrchestrationResult:\n    \"\"\"Set a flow run state, invoking any orchestration rules.\"\"\"\n\n    # pass the request version to the orchestration engine to support compatibility code\n    orchestration_parameters.update({\"api-version\": api_version})\n\n    now = pendulum.now(\"UTC\")\n\n    # create the state\n    async with db.session_context(\n        begin_transaction=True, with_for_update=True\n    ) as session:\n        orchestration_result = await models.flow_runs.set_flow_run_state(\n            session=session,\n            flow_run_id=flow_run_id,\n            # convert to a full State object\n            state=schemas.states.State.parse_obj(state),\n            force=force,\n            flow_policy=flow_policy,\n            orchestration_parameters=orchestration_parameters,\n        )\n\n    # set the 201 if a new state was created\n    if orchestration_result.state and orchestration_result.state.timestamp >= now:\n        response.status_code = status.HTTP_201_CREATED\n    else:\n        response.status_code = status.HTTP_200_OK\n\n    return orchestration_result\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flow_runs/#prefect.server.api.flow_runs.update_flow_run","title":"update_flow_run async","text":"

    Updates a flow run.

    Source code in prefect/server/api/flow_runs.py
    @router.patch(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def update_flow_run(\n    flow_run: schemas.actions.FlowRunUpdate,\n    flow_run_id: UUID = Path(..., description=\"The flow run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Updates a flow run.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.flow_runs.update_flow_run(\n            session=session, flow_run=flow_run, flow_run_id=flow_run_id\n        )\n    if not result:\n        raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Flow run not found\")\n
    ","tags":["Prefect API","flow runs"]},{"location":"api-ref/server/api/flows/","title":"server.api.flows","text":"","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows","title":"prefect.server.api.flows","text":"

    Routes for interacting with flow objects.

    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.count_flows","title":"count_flows async","text":"

    Count flows.

    Source code in prefect/server/api/flows.py
    @router.post(\"/count\")\nasync def count_flows(\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> int:\n    \"\"\"\n    Count flows.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flows.count_flows(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n        )\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.create_flow","title":"create_flow async","text":"

    Gracefully creates a new flow from the provided schema. If a flow with the same name already exists, the existing flow is returned.

    Source code in prefect/server/api/flows.py
    @router.post(\"/\")\nasync def create_flow(\n    flow: schemas.actions.FlowCreate,\n    response: Response,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.Flow:\n    \"\"\"Gracefully creates a new flow from the provided schema. If a flow with the\n    same name already exists, the existing flow is returned.\n    \"\"\"\n    # hydrate the input model into a full flow model\n    flow = schemas.core.Flow(**flow.dict())\n\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        model = await models.flows.create_flow(session=session, flow=flow)\n\n    if model.created >= now:\n        response.status_code = status.HTTP_201_CREATED\n    return model\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.delete_flow","title":"delete_flow async","text":"

    Delete a flow by id.

    Source code in prefect/server/api/flows.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_flow(\n    flow_id: UUID = Path(..., description=\"The flow id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a flow by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.flows.delete_flow(session=session, flow_id=flow_id)\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow not found\"\n        )\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.read_flow","title":"read_flow async","text":"

    Get a flow by id.

    Source code in prefect/server/api/flows.py
    @router.get(\"/{id}\")\nasync def read_flow(\n    flow_id: UUID = Path(..., description=\"The flow id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.Flow:\n    \"\"\"\n    Get a flow by id.\n    \"\"\"\n    async with db.session_context() as session:\n        flow = await models.flows.read_flow(session=session, flow_id=flow_id)\n    if not flow:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow not found\"\n        )\n    return flow\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.read_flow_by_name","title":"read_flow_by_name async","text":"

    Get a flow by name.

    Source code in prefect/server/api/flows.py
    @router.get(\"/name/{name}\")\nasync def read_flow_by_name(\n    name: str = Path(..., description=\"The name of the flow\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.Flow:\n    \"\"\"\n    Get a flow by name.\n    \"\"\"\n    async with db.session_context() as session:\n        flow = await models.flows.read_flow_by_name(session=session, name=name)\n    if not flow:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow not found\"\n        )\n    return flow\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.read_flows","title":"read_flows async","text":"

    Query for flows.

    Source code in prefect/server/api/flows.py
    @router.post(\"/filter\")\nasync def read_flows(\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    sort: schemas.sorting.FlowSort = Body(schemas.sorting.FlowSort.NAME_ASC),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.Flow]:\n    \"\"\"\n    Query for flows.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.flows.read_flows(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            sort=sort,\n            offset=offset,\n            limit=limit,\n        )\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/flows/#prefect.server.api.flows.update_flow","title":"update_flow async","text":"

    Updates a flow.

    Source code in prefect/server/api/flows.py
    @router.patch(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def update_flow(\n    flow: schemas.actions.FlowUpdate,\n    flow_id: UUID = Path(..., description=\"The flow id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Updates a flow.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.flows.update_flow(\n            session=session, flow=flow, flow_id=flow_id\n        )\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow not found\"\n        )\n
    ","tags":["Prefect API","flows"]},{"location":"api-ref/server/api/run_history/","title":"server.api.run_history","text":"","tags":["Prefect API","flow runs","task runs","observability"]},{"location":"api-ref/server/api/run_history/#prefect.server.api.run_history","title":"prefect.server.api.run_history","text":"

    Utilities for querying flow and task run history.

    ","tags":["Prefect API","flow runs","task runs","observability"]},{"location":"api-ref/server/api/run_history/#prefect.server.api.run_history.run_history","title":"run_history async","text":"

    Produce a history of runs aggregated by interval and state

    Source code in prefect/server/api/run_history.py
    @inject_db\nasync def run_history(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    run_type: Literal[\"flow_run\", \"task_run\"],\n    history_start: DateTimeTZ,\n    history_end: DateTimeTZ,\n    history_interval: datetime.timedelta,\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    work_pools: schemas.filters.WorkPoolFilter = None,\n    work_queues: schemas.filters.WorkQueueFilter = None,\n) -> List[schemas.responses.HistoryResponse]:\n    \"\"\"\n    Produce a history of runs aggregated by interval and state\n    \"\"\"\n\n    # SQLite has issues with very small intervals\n    # (by 0.001 seconds it stops incrementing the interval)\n    if history_interval < datetime.timedelta(seconds=1):\n        raise ValueError(\"History interval must not be less than 1 second.\")\n\n    # prepare run-specific models\n    if run_type == \"flow_run\":\n        run_model = db.FlowRun\n        run_filter_function = models.flow_runs._apply_flow_run_filters\n    elif run_type == \"task_run\":\n        run_model = db.TaskRun\n        run_filter_function = models.task_runs._apply_task_run_filters\n    else:\n        raise ValueError(\n            f\"Unknown run type {run_type!r}. Expected 'flow_run' or 'task_run'.\"\n        )\n\n    # create a CTE for timestamp intervals\n    intervals = db.make_timestamp_intervals(\n        history_start,\n        history_end,\n        history_interval,\n    ).cte(\"intervals\")\n\n    # apply filters to the flow runs (and related states)\n    runs = (\n        await run_filter_function(\n            sa.select(\n                run_model.id,\n                run_model.expected_start_time,\n                run_model.estimated_run_time,\n                run_model.estimated_start_time_delta,\n                run_model.state_type,\n                run_model.state_name,\n            ).select_from(run_model),\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            work_pool_filter=work_pools,\n            work_queue_filter=work_queues,\n        )\n    ).alias(\"runs\")\n    # outer join intervals to the filtered runs to create a dataset composed of\n    # every interval and the aggregate of all its runs. The runs aggregate is represented\n    # by a descriptive JSON object\n    counts = (\n        sa.select(\n            intervals.c.interval_start,\n            intervals.c.interval_end,\n            # build a JSON object, ignoring the case where the count of runs is 0\n            sa.case(\n                (sa.func.count(runs.c.id) == 0, None),\n                else_=db.build_json_object(\n                    \"state_type\",\n                    runs.c.state_type,\n                    \"state_name\",\n                    runs.c.state_name,\n                    \"count_runs\",\n                    sa.func.count(runs.c.id),\n                    # estimated run times only includes positive run times (to avoid any unexpected corner cases)\n                    \"sum_estimated_run_time\",\n                    sa.func.sum(\n                        db.greatest(0, sa.extract(\"epoch\", runs.c.estimated_run_time))\n                    ),\n                    # estimated lateness is the sum of any positive start time deltas\n                    \"sum_estimated_lateness\",\n                    sa.func.sum(\n                        db.greatest(\n                            0, sa.extract(\"epoch\", runs.c.estimated_start_time_delta)\n                        )\n                    ),\n                ),\n            ).label(\"state_agg\"),\n        )\n        .select_from(intervals)\n        .join(\n            runs,\n            sa.and_(\n                runs.c.expected_start_time >= intervals.c.interval_start,\n                runs.c.expected_start_time < intervals.c.interval_end,\n            ),\n            isouter=True,\n        )\n        .group_by(\n            intervals.c.interval_start,\n            intervals.c.interval_end,\n            runs.c.state_type,\n            runs.c.state_name,\n        )\n    ).alias(\"counts\")\n\n    # aggregate all state-aggregate objects into a single array for each interval,\n    # ensuring that intervals with no runs have an empty array\n    query = (\n        sa.select(\n            counts.c.interval_start,\n            counts.c.interval_end,\n            sa.func.coalesce(\n                db.json_arr_agg(db.cast_to_json(counts.c.state_agg)).filter(\n                    counts.c.state_agg.is_not(None)\n                ),\n                sa.text(\"'[]'\"),\n            ).label(\"states\"),\n        )\n        .group_by(counts.c.interval_start, counts.c.interval_end)\n        .order_by(counts.c.interval_start)\n        # return no more than 500 bars\n        .limit(500)\n    )\n\n    # issue the query\n    result = await session.execute(query)\n    records = result.mappings()\n\n    # load and parse the record if the database returns JSON as strings\n    if db.uses_json_strings:\n        records = [dict(r) for r in records]\n        for r in records:\n            r[\"states\"] = json.loads(r[\"states\"])\n\n    return pydantic.parse_obj_as(List[schemas.responses.HistoryResponse], list(records))\n
    ","tags":["Prefect API","flow runs","task runs","observability"]},{"location":"api-ref/server/api/saved_searches/","title":"server.api.saved_searches","text":"","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches","title":"prefect.server.api.saved_searches","text":"

    Routes for interacting with saved search objects.

    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches.create_saved_search","title":"create_saved_search async","text":"

    Gracefully creates a new saved search from the provided schema.

    If a saved search with the same name already exists, the saved search's fields are replaced.

    Source code in prefect/server/api/saved_searches.py
    @router.put(\"/\")\nasync def create_saved_search(\n    saved_search: schemas.actions.SavedSearchCreate,\n    response: Response,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.SavedSearch:\n    \"\"\"Gracefully creates a new saved search from the provided schema.\n\n    If a saved search with the same name already exists, the saved search's fields are\n    replaced.\n    \"\"\"\n\n    # hydrate the input model into a full model\n    saved_search = schemas.core.SavedSearch(**saved_search.dict())\n\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        model = await models.saved_searches.create_saved_search(\n            session=session, saved_search=saved_search\n        )\n\n    if model.created >= now:\n        response.status_code = status.HTTP_201_CREATED\n\n    return model\n
    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches.delete_saved_search","title":"delete_saved_search async","text":"

    Delete a saved search by id.

    Source code in prefect/server/api/saved_searches.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_saved_search(\n    saved_search_id: UUID = Path(..., description=\"The saved search id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a saved search by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.saved_searches.delete_saved_search(\n            session=session, saved_search_id=saved_search_id\n        )\n    if not result:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Saved search not found\"\n        )\n
    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches.read_saved_search","title":"read_saved_search async","text":"

    Get a saved search by id.

    Source code in prefect/server/api/saved_searches.py
    @router.get(\"/{id}\")\nasync def read_saved_search(\n    saved_search_id: UUID = Path(..., description=\"The saved search id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.SavedSearch:\n    \"\"\"\n    Get a saved search by id.\n    \"\"\"\n    async with db.session_context() as session:\n        saved_search = await models.saved_searches.read_saved_search(\n            session=session, saved_search_id=saved_search_id\n        )\n    if not saved_search:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Saved search not found\"\n        )\n    return saved_search\n
    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/saved_searches/#prefect.server.api.saved_searches.read_saved_searches","title":"read_saved_searches async","text":"

    Query for saved searches.

    Source code in prefect/server/api/saved_searches.py
    @router.post(\"/filter\")\nasync def read_saved_searches(\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.SavedSearch]:\n    \"\"\"\n    Query for saved searches.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.saved_searches.read_saved_searches(\n            session=session,\n            offset=offset,\n            limit=limit,\n        )\n
    ","tags":["Prefect API","search","saved search"]},{"location":"api-ref/server/api/server/","title":"server.api.server","text":"","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server","title":"prefect.server.api.server","text":"

    Defines the Prefect REST API FastAPI app.

    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.RequestLimitMiddleware","title":"RequestLimitMiddleware","text":"

    A middleware that limits the number of concurrent requests handled by the API.

    This is a blunt tool for limiting SQLite concurrent writes which will cause failures at high volume. Ideally, we would only apply the limit to routes that perform writes.

    Source code in prefect/server/api/server.py
    class RequestLimitMiddleware:\n    \"\"\"\n    A middleware that limits the number of concurrent requests handled by the API.\n\n    This is a blunt tool for limiting SQLite concurrent writes which will cause failures\n    at high volume. Ideally, we would only apply the limit to routes that perform\n    writes.\n    \"\"\"\n\n    def __init__(self, app, limit: float):\n        self.app = app\n        self._limiter = anyio.CapacityLimiter(limit)\n\n    async def __call__(self, scope, receive, send) -> None:\n        async with self._limiter:\n            await self.app(scope, receive, send)\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.SPAStaticFiles","title":"SPAStaticFiles","text":"

    Bases: StaticFiles

    Implementation of StaticFiles for serving single page applications.

    Adds get_response handling to ensure that when a resource isn't found the application still returns the index.

    Source code in prefect/server/api/server.py
    class SPAStaticFiles(StaticFiles):\n    \"\"\"\n    Implementation of `StaticFiles` for serving single page applications.\n\n    Adds `get_response` handling to ensure that when a resource isn't found the\n    application still returns the index.\n    \"\"\"\n\n    async def get_response(self, path: str, scope):\n        try:\n            return await super().get_response(path, scope)\n        except HTTPException:\n            return await super().get_response(\"./index.html\", scope)\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.create_api_app","title":"create_api_app","text":"

    Create a FastAPI app that includes the Prefect REST API

    Parameters:

    Name Type Description Default router_prefix Optional[str]

    a prefix to apply to all included routers

    '' dependencies Optional[List[Depends]]

    a list of global dependencies to add to each Prefect REST API router

    None health_check_path str

    the health check route path

    '/health' fast_api_app_kwargs dict

    kwargs to pass to the FastAPI constructor

    None router_overrides Mapping[str, Optional[APIRouter]]

    a mapping of route prefixes (i.e. \"/admin\") to new routers allowing the caller to override the default routers. If None is provided as a value, the default router will be dropped from the application.

    None

    Returns:

    Type Description FastAPI

    a FastAPI app that serves the Prefect REST API

    Source code in prefect/server/api/server.py
    def create_api_app(\n    router_prefix: Optional[str] = \"\",\n    dependencies: Optional[List[Depends]] = None,\n    health_check_path: str = \"/health\",\n    version_check_path: str = \"/version\",\n    fast_api_app_kwargs: dict = None,\n    router_overrides: Mapping[str, Optional[APIRouter]] = None,\n) -> FastAPI:\n    \"\"\"\n    Create a FastAPI app that includes the Prefect REST API\n\n    Args:\n        router_prefix: a prefix to apply to all included routers\n        dependencies: a list of global dependencies to add to each Prefect REST API router\n        health_check_path: the health check route path\n        fast_api_app_kwargs: kwargs to pass to the FastAPI constructor\n        router_overrides: a mapping of route prefixes (i.e. \"/admin\") to new routers\n            allowing the caller to override the default routers. If `None` is provided\n            as a value, the default router will be dropped from the application.\n\n    Returns:\n        a FastAPI app that serves the Prefect REST API\n    \"\"\"\n    fast_api_app_kwargs = fast_api_app_kwargs or {}\n    api_app = FastAPI(title=API_TITLE, **fast_api_app_kwargs)\n    api_app.add_middleware(GZipMiddleware)\n\n    @api_app.get(health_check_path, tags=[\"Root\"])\n    async def health_check():\n        return True\n\n    @api_app.get(version_check_path, tags=[\"Root\"])\n    async def orion_info():\n        return SERVER_API_VERSION\n\n    # always include version checking\n    if dependencies is None:\n        dependencies = [Depends(enforce_minimum_version)]\n    else:\n        dependencies.append(Depends(enforce_minimum_version))\n\n    routers = {router.prefix: router for router in API_ROUTERS}\n\n    if router_overrides:\n        for prefix, router in router_overrides.items():\n            # We may want to allow this behavior in the future to inject new routes, but\n            # for now this will be treated an as an exception\n            if prefix not in routers:\n                raise KeyError(\n                    \"Router override provided for prefix that does not exist:\"\n                    f\" {prefix!r}\"\n                )\n\n            # Drop the existing router\n            existing_router = routers.pop(prefix)\n\n            # Replace it with a new router if provided\n            if router is not None:\n                if prefix != router.prefix:\n                    # We may want to allow this behavior in the future, but it will\n                    # break expectations without additional routing and is banned for\n                    # now\n                    raise ValueError(\n                        f\"Router override for {prefix!r} defines a different prefix \"\n                        f\"{router.prefix!r}.\"\n                    )\n\n                existing_paths = method_paths_from_routes(existing_router.routes)\n                new_paths = method_paths_from_routes(router.routes)\n                if not existing_paths.issubset(new_paths):\n                    raise ValueError(\n                        f\"Router override for {prefix!r} is missing paths defined by \"\n                        f\"the original router: {existing_paths.difference(new_paths)}\"\n                    )\n\n                routers[prefix] = router\n\n    for router in routers.values():\n        api_app.include_router(router, prefix=router_prefix, dependencies=dependencies)\n\n    return api_app\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.create_app","title":"create_app","text":"

    Create an FastAPI app that includes the Prefect REST API and UI

    Parameters:

    Name Type Description Default settings Settings

    The settings to use to create the app. If not set, settings are pulled from the context.

    None ignore_cache bool

    If set, a new application will be created even if the settings match. Otherwise, an application is returned from the cache.

    False ephemeral bool

    If set, the application will be treated as ephemeral. The UI and services will be disabled.

    False Source code in prefect/server/api/server.py
    def create_app(\n    settings: prefect.settings.Settings = None,\n    ephemeral: bool = False,\n    ignore_cache: bool = False,\n) -> FastAPI:\n    \"\"\"\n    Create an FastAPI app that includes the Prefect REST API and UI\n\n    Args:\n        settings: The settings to use to create the app. If not set, settings are pulled\n            from the context.\n        ignore_cache: If set, a new application will be created even if the settings\n            match. Otherwise, an application is returned from the cache.\n        ephemeral: If set, the application will be treated as ephemeral. The UI\n            and services will be disabled.\n    \"\"\"\n    settings = settings or prefect.settings.get_current_settings()\n    cache_key = (settings, ephemeral)\n\n    if cache_key in APP_CACHE and not ignore_cache:\n        return APP_CACHE[cache_key]\n\n    # TODO: Move these startup functions out of this closure into the top-level or\n    #       another dedicated location\n    async def run_migrations():\n        \"\"\"Ensure the database is created and up to date with the current migrations\"\"\"\n        if prefect.settings.PREFECT_API_DATABASE_MIGRATE_ON_START:\n            from prefect.server.database.dependencies import provide_database_interface\n\n            db = provide_database_interface()\n            await db.create_db()\n\n    @_memoize_block_auto_registration\n    async def add_block_types():\n        \"\"\"Add all registered blocks to the database\"\"\"\n        if not prefect.settings.PREFECT_API_BLOCKS_REGISTER_ON_START:\n            return\n\n        from prefect.server.database.dependencies import provide_database_interface\n        from prefect.server.models.block_registration import run_block_auto_registration\n\n        db = provide_database_interface()\n        session = await db.session()\n\n        async with session:\n            await run_block_auto_registration(session=session)\n\n    async def start_services():\n        \"\"\"Start additional services when the Prefect REST API starts up.\"\"\"\n\n        if ephemeral:\n            app.state.services = None\n            return\n\n        service_instances = []\n\n        if prefect.settings.PREFECT_API_SERVICES_SCHEDULER_ENABLED.value():\n            service_instances.append(services.scheduler.Scheduler())\n            service_instances.append(services.scheduler.RecentDeploymentsScheduler())\n\n        if prefect.settings.PREFECT_API_SERVICES_LATE_RUNS_ENABLED.value():\n            service_instances.append(services.late_runs.MarkLateRuns())\n\n        if prefect.settings.PREFECT_API_SERVICES_PAUSE_EXPIRATIONS_ENABLED.value():\n            service_instances.append(services.pause_expirations.FailExpiredPauses())\n\n        if prefect.settings.PREFECT_API_SERVICES_CANCELLATION_CLEANUP_ENABLED.value():\n            service_instances.append(\n                services.cancellation_cleanup.CancellationCleanup()\n            )\n\n        if prefect.settings.PREFECT_SERVER_ANALYTICS_ENABLED.value():\n            service_instances.append(services.telemetry.Telemetry())\n\n        if prefect.settings.PREFECT_API_SERVICES_FLOW_RUN_NOTIFICATIONS_ENABLED.value():\n            service_instances.append(\n                services.flow_run_notifications.FlowRunNotifications()\n            )\n\n        loop = asyncio.get_running_loop()\n\n        app.state.services = {\n            service: loop.create_task(service.start()) for service in service_instances\n        }\n\n        for service, task in app.state.services.items():\n            logger.info(f\"{service.name} service scheduled to start in-app\")\n            task.add_done_callback(partial(on_service_exit, service))\n\n    async def stop_services():\n        \"\"\"Ensure services are stopped before the Prefect REST API shuts down.\"\"\"\n        if hasattr(app.state, \"services\") and app.state.services:\n            await asyncio.gather(*[service.stop() for service in app.state.services])\n            try:\n                await asyncio.gather(\n                    *[task.stop() for task in app.state.services.values()]\n                )\n            except Exception:\n                # `on_service_exit` should handle logging exceptions on exit\n                pass\n\n    @asynccontextmanager\n    async def lifespan(app):\n        try:\n            await run_migrations()\n            await add_block_types()\n            await start_services()\n            yield\n        finally:\n            await stop_services()\n\n    def on_service_exit(service, task):\n        \"\"\"\n        Added as a callback for completion of services to log exit\n        \"\"\"\n        try:\n            # Retrieving the result will raise the exception\n            task.result()\n        except Exception:\n            logger.error(f\"{service.name} service failed!\", exc_info=True)\n        else:\n            logger.info(f\"{service.name} service stopped!\")\n\n    app = FastAPI(\n        title=TITLE,\n        version=API_VERSION,\n        lifespan=lifespan,\n    )\n    api_app = create_api_app(\n        fast_api_app_kwargs={\n            \"exception_handlers\": {\n                # NOTE: FastAPI special cases the generic `Exception` handler and\n                #       registers it as a separate middleware from the others\n                Exception: custom_internal_exception_handler,\n                RequestValidationError: validation_exception_handler,\n                sa.exc.IntegrityError: integrity_exception_handler,\n                ObjectNotFoundError: prefect_object_not_found_exception_handler,\n            }\n        },\n    )\n    ui_app = create_ui_app(ephemeral)\n\n    # middleware\n    app.add_middleware(\n        CORSMiddleware,\n        allow_origins=[\"*\"],\n        allow_methods=[\"*\"],\n        allow_headers=[\"*\"],\n    )\n\n    # Limit the number of concurrent requests when using a SQLite database to reduce\n    # chance of errors where the database cannot be opened due to a high number of\n    # concurrent writes\n    if (\n        get_dialect(prefect.settings.PREFECT_API_DATABASE_CONNECTION_URL.value()).name\n        == \"sqlite\"\n    ):\n        app.add_middleware(RequestLimitMiddleware, limit=100)\n\n    api_app.mount(\n        \"/static\",\n        StaticFiles(\n            directory=os.path.join(\n                os.path.dirname(os.path.realpath(__file__)), \"static\"\n            )\n        ),\n        name=\"static\",\n    )\n    app.api_app = api_app\n    app.mount(\"/api\", app=api_app, name=\"api\")\n    app.mount(\"/\", app=ui_app, name=\"ui\")\n\n    def openapi():\n        \"\"\"\n        Convenience method for extracting the user facing OpenAPI schema from the API app.\n\n        This method is attached to the global public app for easy access.\n        \"\"\"\n        partial_schema = get_openapi(\n            title=API_TITLE,\n            version=API_VERSION,\n            routes=api_app.routes,\n        )\n        new_schema = partial_schema.copy()\n        new_schema[\"paths\"] = {}\n        for path, value in partial_schema[\"paths\"].items():\n            new_schema[\"paths\"][f\"/api{path}\"] = value\n\n        new_schema[\"info\"][\"x-logo\"] = {\"url\": \"static/prefect-logo-mark-gradient.png\"}\n        return new_schema\n\n    app.openapi = openapi\n\n    APP_CACHE[cache_key] = app\n    return app\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.custom_internal_exception_handler","title":"custom_internal_exception_handler async","text":"

    Log a detailed exception for internal server errors before returning.

    Send 503 for errors clients can retry on.

    Source code in prefect/server/api/server.py
    async def custom_internal_exception_handler(request: Request, exc: Exception):\n    \"\"\"\n    Log a detailed exception for internal server errors before returning.\n\n    Send 503 for errors clients can retry on.\n    \"\"\"\n    logger.error(\"Encountered exception in request:\", exc_info=True)\n\n    if is_client_retryable_exception(exc):\n        return JSONResponse(\n            content={\"exception_message\": \"Service Unavailable\"},\n            status_code=status.HTTP_503_SERVICE_UNAVAILABLE,\n        )\n\n    return JSONResponse(\n        content={\"exception_message\": \"Internal Server Error\"},\n        status_code=status.HTTP_500_INTERNAL_SERVER_ERROR,\n    )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.integrity_exception_handler","title":"integrity_exception_handler async","text":"

    Capture database integrity errors.

    Source code in prefect/server/api/server.py
    async def integrity_exception_handler(request: Request, exc: Exception):\n    \"\"\"Capture database integrity errors.\"\"\"\n    logger.error(\"Encountered exception in request:\", exc_info=True)\n    return JSONResponse(\n        content={\n            \"detail\": (\n                \"Data integrity conflict. This usually means a \"\n                \"unique or foreign key constraint was violated. \"\n                \"See server logs for details.\"\n            )\n        },\n        status_code=status.HTTP_409_CONFLICT,\n    )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.prefect_object_not_found_exception_handler","title":"prefect_object_not_found_exception_handler async","text":"

    Return 404 status code on object not found exceptions.

    Source code in prefect/server/api/server.py
    async def prefect_object_not_found_exception_handler(\n    request: Request, exc: ObjectNotFoundError\n):\n    \"\"\"Return 404 status code on object not found exceptions.\"\"\"\n    return JSONResponse(\n        content={\"exception_message\": str(exc)}, status_code=status.HTTP_404_NOT_FOUND\n    )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/server/#prefect.server.api.server.validation_exception_handler","title":"validation_exception_handler async","text":"

    Provide a detailed message for request validation errors.

    Source code in prefect/server/api/server.py
    async def validation_exception_handler(request: Request, exc: RequestValidationError):\n    \"\"\"Provide a detailed message for request validation errors.\"\"\"\n    return JSONResponse(\n        status_code=status.HTTP_422_UNPROCESSABLE_ENTITY,\n        content=jsonable_encoder(\n            {\n                \"exception_message\": \"Invalid request received.\",\n                \"exception_detail\": exc.errors(),\n                \"request_body\": exc.body,\n            }\n        ),\n    )\n
    ","tags":["Prefect API","FastAPI"]},{"location":"api-ref/server/api/task_run_states/","title":"server.api.task_run_states","text":"","tags":["Prefect API","task runs","states"]},{"location":"api-ref/server/api/task_run_states/#prefect.server.api.task_run_states","title":"prefect.server.api.task_run_states","text":"

    Routes for interacting with task run state objects.

    ","tags":["Prefect API","task runs","states"]},{"location":"api-ref/server/api/task_run_states/#prefect.server.api.task_run_states.read_task_run_state","title":"read_task_run_state async","text":"

    Get a task run state by id.

    Source code in prefect/server/api/task_run_states.py
    @router.get(\"/{id}\")\nasync def read_task_run_state(\n    task_run_state_id: UUID = Path(\n        ..., description=\"The task run state id\", alias=\"id\"\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.states.State:\n    \"\"\"\n    Get a task run state by id.\n    \"\"\"\n    async with db.session_context() as session:\n        task_run_state = await models.task_run_states.read_task_run_state(\n            session=session, task_run_state_id=task_run_state_id\n        )\n    if not task_run_state:\n        raise HTTPException(\n            status_code=status.HTTP_404_NOT_FOUND, detail=\"Flow run state not found\"\n        )\n    return task_run_state\n
    ","tags":["Prefect API","task runs","states"]},{"location":"api-ref/server/api/task_run_states/#prefect.server.api.task_run_states.read_task_run_states","title":"read_task_run_states async","text":"

    Get states associated with a task run.

    Source code in prefect/server/api/task_run_states.py
    @router.get(\"/\")\nasync def read_task_run_states(\n    task_run_id: UUID,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.states.State]:\n    \"\"\"\n    Get states associated with a task run.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.task_run_states.read_task_run_states(\n            session=session, task_run_id=task_run_id\n        )\n
    ","tags":["Prefect API","task runs","states"]},{"location":"api-ref/server/api/task_runs/","title":"server.api.task_runs","text":"","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs","title":"prefect.server.api.task_runs","text":"

    Routes for interacting with task run objects.

    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.count_task_runs","title":"count_task_runs async","text":"

    Count task runs.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/count\")\nasync def count_task_runs(\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n) -> int:\n    \"\"\"\n    Count task runs.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.task_runs.count_task_runs(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n        )\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.create_task_run","title":"create_task_run async","text":"

    Create a task run. If a task run with the same flow_run_id, task_key, and dynamic_key already exists, the existing task run will be returned.

    If no state is provided, the task run will be created in a PENDING state.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/\")\nasync def create_task_run(\n    task_run: schemas.actions.TaskRunCreate,\n    response: Response,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_task_orchestration_parameters\n    ),\n) -> schemas.core.TaskRun:\n    \"\"\"\n    Create a task run. If a task run with the same flow_run_id,\n    task_key, and dynamic_key already exists, the existing task\n    run will be returned.\n\n    If no state is provided, the task run will be created in a PENDING state.\n    \"\"\"\n    # hydrate the input model into a full task run / state model\n    task_run = schemas.core.TaskRun(**task_run.dict())\n\n    if not task_run.state:\n        task_run.state = schemas.states.Pending()\n\n    now = pendulum.now(\"UTC\")\n\n    async with db.session_context(begin_transaction=True) as session:\n        model = await models.task_runs.create_task_run(\n            session=session,\n            task_run=task_run,\n            orchestration_parameters=orchestration_parameters,\n        )\n\n    if model.created >= now:\n        response.status_code = status.HTTP_201_CREATED\n    return model\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.delete_task_run","title":"delete_task_run async","text":"

    Delete a task run by id.

    Source code in prefect/server/api/task_runs.py
    @router.delete(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def delete_task_run(\n    task_run_id: UUID = Path(..., description=\"The task run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Delete a task run by id.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.task_runs.delete_task_run(\n            session=session, task_run_id=task_run_id\n        )\n    if not result:\n        raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Task not found\")\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.read_task_run","title":"read_task_run async","text":"

    Get a task run by id.

    Source code in prefect/server/api/task_runs.py
    @router.get(\"/{id}\")\nasync def read_task_run(\n    task_run_id: UUID = Path(..., description=\"The task run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> schemas.core.TaskRun:\n    \"\"\"\n    Get a task run by id.\n    \"\"\"\n    async with db.session_context() as session:\n        task_run = await models.task_runs.read_task_run(\n            session=session, task_run_id=task_run_id\n        )\n    if not task_run:\n        raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Task not found\")\n    return task_run\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.read_task_runs","title":"read_task_runs async","text":"

    Query for task runs.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/filter\")\nasync def read_task_runs(\n    sort: schemas.sorting.TaskRunSort = Body(schemas.sorting.TaskRunSort.ID_DESC),\n    limit: int = dependencies.LimitBody(),\n    offset: int = Body(0, ge=0),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.core.TaskRun]:\n    \"\"\"\n    Query for task runs.\n    \"\"\"\n    async with db.session_context() as session:\n        return await models.task_runs.read_task_runs(\n            session=session,\n            flow_filter=flows,\n            flow_run_filter=flow_runs,\n            task_run_filter=task_runs,\n            deployment_filter=deployments,\n            offset=offset,\n            limit=limit,\n            sort=sort,\n        )\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.set_task_run_state","title":"set_task_run_state async","text":"

    Set a task run state, invoking any orchestration rules.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/{id}/set_state\")\nasync def set_task_run_state(\n    task_run_id: UUID = Path(..., description=\"The task run id\", alias=\"id\"),\n    state: schemas.actions.StateCreate = Body(..., description=\"The intended state.\"),\n    force: bool = Body(\n        False,\n        description=(\n            \"If false, orchestration rules will be applied that may alter or prevent\"\n            \" the state transition. If True, orchestration rules are not applied.\"\n        ),\n    ),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n    response: Response = None,\n    task_policy: BaseOrchestrationPolicy = Depends(\n        orchestration_dependencies.provide_task_policy\n    ),\n    orchestration_parameters: dict = Depends(\n        orchestration_dependencies.provide_task_orchestration_parameters\n    ),\n) -> OrchestrationResult:\n    \"\"\"Set a task run state, invoking any orchestration rules.\"\"\"\n\n    now = pendulum.now(\"UTC\")\n\n    # create the state\n    async with db.session_context(\n        begin_transaction=True, with_for_update=True\n    ) as session:\n        orchestration_result = await models.task_runs.set_task_run_state(\n            session=session,\n            task_run_id=task_run_id,\n            state=schemas.states.State.parse_obj(\n                state\n            ),  # convert to a full State object\n            force=force,\n            task_policy=task_policy,\n            orchestration_parameters=orchestration_parameters,\n        )\n\n    # set the 201 if a new state was created\n    if orchestration_result.state and orchestration_result.state.timestamp >= now:\n        response.status_code = status.HTTP_201_CREATED\n    else:\n        response.status_code = status.HTTP_200_OK\n\n    return orchestration_result\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.task_run_history","title":"task_run_history async","text":"

    Query for task run history data across a given range and interval.

    Source code in prefect/server/api/task_runs.py
    @router.post(\"/history\")\nasync def task_run_history(\n    history_start: DateTimeTZ = Body(..., description=\"The history's start time.\"),\n    history_end: DateTimeTZ = Body(..., description=\"The history's end time.\"),\n    history_interval: datetime.timedelta = Body(\n        ...,\n        description=(\n            \"The size of each history interval, in seconds. Must be at least 1 second.\"\n        ),\n        alias=\"history_interval_seconds\",\n    ),\n    flows: schemas.filters.FlowFilter = None,\n    flow_runs: schemas.filters.FlowRunFilter = None,\n    task_runs: schemas.filters.TaskRunFilter = None,\n    deployments: schemas.filters.DeploymentFilter = None,\n    db: PrefectDBInterface = Depends(provide_database_interface),\n) -> List[schemas.responses.HistoryResponse]:\n    \"\"\"\n    Query for task run history data across a given range and interval.\n    \"\"\"\n    if history_interval < datetime.timedelta(seconds=1):\n        raise HTTPException(\n            status.HTTP_422_UNPROCESSABLE_ENTITY,\n            detail=\"History interval must not be less than 1 second.\",\n        )\n\n    async with db.session_context() as session:\n        return await run_history(\n            session=session,\n            run_type=\"task_run\",\n            history_start=history_start,\n            history_end=history_end,\n            history_interval=history_interval,\n            flows=flows,\n            flow_runs=flow_runs,\n            task_runs=task_runs,\n            deployments=deployments,\n        )\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/api/task_runs/#prefect.server.api.task_runs.update_task_run","title":"update_task_run async","text":"

    Updates a task run.

    Source code in prefect/server/api/task_runs.py
    @router.patch(\"/{id}\", status_code=status.HTTP_204_NO_CONTENT)\nasync def update_task_run(\n    task_run: schemas.actions.TaskRunUpdate,\n    task_run_id: UUID = Path(..., description=\"The task run id\", alias=\"id\"),\n    db: PrefectDBInterface = Depends(provide_database_interface),\n):\n    \"\"\"\n    Updates a task run.\n    \"\"\"\n    async with db.session_context(begin_transaction=True) as session:\n        result = await models.task_runs.update_task_run(\n            session=session, task_run=task_run, task_run_id=task_run_id\n        )\n    if not result:\n        raise HTTPException(status.HTTP_404_NOT_FOUND, detail=\"Task run not found\")\n
    ","tags":["Prefect API","task runs"]},{"location":"api-ref/server/models/deployments/","title":"server.models.deployments","text":""},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments","title":"prefect.server.models.deployments","text":"

    Functions for interacting with deployment ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.check_work_queues_for_deployment","title":"check_work_queues_for_deployment async","text":"

    Get work queues that can pick up the specified deployment.

    Work queues will pick up a deployment when all of the following are met.

    • The deployment has ALL tags that the work queue has (i.e. the work queue's tags must be a subset of the deployment's tags).
    • The work queue's specified deployment IDs match the deployment's ID, or the work queue does NOT have specified deployment IDs.
    • The work queue's specified flow runners match the deployment's flow runner or the work queue does NOT have a specified flow runner.

    Notes on the query:

    • Our database currently allows either \"null\" and empty lists as null values in filters, so we need to catch both cases with \"or\".
    • json_contains(A, B) should be interpreted as \"True if A contains B\".

    Returns:

    Type Description List[WorkQueue]

    List[db.WorkQueue]: WorkQueues

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def check_work_queues_for_deployment(\n    db: PrefectDBInterface, session: sa.orm.Session, deployment_id: UUID\n) -> List[schemas.core.WorkQueue]:\n    \"\"\"\n    Get work queues that can pick up the specified deployment.\n\n    Work queues will pick up a deployment when all of the following are met.\n\n    - The deployment has ALL tags that the work queue has (i.e. the work\n    queue's tags must be a subset of the deployment's tags).\n    - The work queue's specified deployment IDs match the deployment's ID,\n    or the work queue does NOT have specified deployment IDs.\n    - The work queue's specified flow runners match the deployment's flow\n    runner or the work queue does NOT have a specified flow runner.\n\n    Notes on the query:\n\n    - Our database currently allows either \"null\" and empty lists as\n    null values in filters, so we need to catch both cases with \"or\".\n    - `json_contains(A, B)` should be interpreted as \"True if A\n    contains B\".\n\n    Returns:\n        List[db.WorkQueue]: WorkQueues\n    \"\"\"\n    deployment = await session.get(db.Deployment, deployment_id)\n    if not deployment:\n        raise ObjectNotFoundError(f\"Deployment with id {deployment_id} not found\")\n\n    query = (\n        select(db.WorkQueue)\n        # work queue tags are a subset of deployment tags\n        .filter(\n            or_(\n                json_contains(deployment.tags, db.WorkQueue.filter[\"tags\"]),\n                json_contains([], db.WorkQueue.filter[\"tags\"]),\n                json_contains(None, db.WorkQueue.filter[\"tags\"]),\n            )\n        )\n        # deployment_ids is null or contains the deployment's ID\n        .filter(\n            or_(\n                json_contains(\n                    db.WorkQueue.filter[\"deployment_ids\"],\n                    str(deployment.id),\n                ),\n                json_contains(None, db.WorkQueue.filter[\"deployment_ids\"]),\n                json_contains([], db.WorkQueue.filter[\"deployment_ids\"]),\n            )\n        )\n    )\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.count_deployments","title":"count_deployments async","text":"

    Count deployments.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_filter FlowFilter

    only count deployments whose flows match these criteria

    None flow_run_filter FlowRunFilter

    only count deployments whose flow runs match these criteria

    None task_run_filter TaskRunFilter

    only count deployments whose task runs match these criteria

    None deployment_filter DeploymentFilter

    only count deployment that match these filters

    None work_pool_filter WorkPoolFilter

    only count deployments that match these work pool filters

    None work_queue_filter WorkQueueFilter

    only count deployments that match these work pool queue filters

    None

    Returns:

    Name Type Description int int

    the number of deployments matching filters

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def count_deployments(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    work_queue_filter: schemas.filters.WorkQueueFilter = None,\n) -> int:\n    \"\"\"\n    Count deployments.\n\n    Args:\n        session: A database session\n        flow_filter: only count deployments whose flows match these criteria\n        flow_run_filter: only count deployments whose flow runs match these criteria\n        task_run_filter: only count deployments whose task runs match these criteria\n        deployment_filter: only count deployment that match these filters\n        work_pool_filter: only count deployments that match these work pool filters\n        work_queue_filter: only count deployments that match these work pool queue filters\n\n    Returns:\n        int: the number of deployments matching filters\n    \"\"\"\n\n    query = select(sa.func.count(sa.text(\"*\"))).select_from(db.Deployment)\n\n    query = await _apply_deployment_filters(\n        query=query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        work_queue_filter=work_queue_filter,\n        db=db,\n    )\n\n    result = await session.execute(query)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.create_deployment","title":"create_deployment async","text":"

    Upserts a deployment.

    Parameters:

    Name Type Description Default session Session

    a database session

    required deployment Deployment

    a deployment model

    required

    Returns:

    Type Description

    db.Deployment: the newly-created or updated deployment

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def create_deployment(\n    session: sa.orm.Session, deployment: schemas.core.Deployment, db: PrefectDBInterface\n):\n    \"\"\"Upserts a deployment.\n\n    Args:\n        session: a database session\n        deployment: a deployment model\n\n    Returns:\n        db.Deployment: the newly-created or updated deployment\n\n    \"\"\"\n\n    # set `updated` manually\n    # known limitation of `on_conflict_do_update`, will not use `Column.onupdate`\n    # https://docs.sqlalchemy.org/en/14/dialects/sqlite.html#the-set-clause\n    deployment.updated = pendulum.now(\"UTC\")\n\n    insert_values = deployment.dict(shallow=True, exclude_unset=True)\n\n    insert_stmt = (\n        (await db.insert(db.Deployment))\n        .values(**insert_values)\n        .on_conflict_do_update(\n            index_elements=db.deployment_unique_upsert_columns,\n            set_={\n                **deployment.dict(\n                    shallow=True,\n                    exclude_unset=True,\n                    exclude={\"id\", \"created\", \"created_by\"},\n                ),\n            },\n        )\n    )\n\n    await session.execute(insert_stmt)\n\n    query = (\n        sa.select(db.Deployment)\n        .where(\n            sa.and_(\n                db.Deployment.flow_id == deployment.flow_id,\n                db.Deployment.name == deployment.name,\n            )\n        )\n        .execution_options(populate_existing=True)\n    )\n    result = await session.execute(query)\n    model = result.scalar()\n\n    # because this could upsert a different schedule, delete any runs from the old\n    # deployment\n    await _delete_scheduled_runs(\n        session=session, deployment_id=model.id, db=db, auto_scheduled_only=True\n    )\n\n    return model\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.delete_deployment","title":"delete_deployment async","text":"

    Delete a deployment by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required deployment_id UUID

    a deployment id

    required

    Returns:

    Name Type Description bool bool

    whether or not the deployment was deleted

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def delete_deployment(\n    session: sa.orm.Session, deployment_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a deployment by id.\n\n    Args:\n        session: A database session\n        deployment_id: a deployment id\n\n    Returns:\n        bool: whether or not the deployment was deleted\n    \"\"\"\n\n    # delete scheduled runs, both auto- and user- created.\n    await _delete_scheduled_runs(\n        session=session, deployment_id=deployment_id, auto_scheduled_only=False\n    )\n\n    result = await session.execute(\n        delete(db.Deployment).where(db.Deployment.id == deployment_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.read_deployment","title":"read_deployment async","text":"

    Reads a deployment by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required deployment_id UUID

    a deployment id

    required

    Returns:

    Type Description

    db.Deployment: the deployment

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def read_deployment(\n    session: sa.orm.Session, deployment_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"Reads a deployment by id.\n\n    Args:\n        session: A database session\n        deployment_id: a deployment id\n\n    Returns:\n        db.Deployment: the deployment\n    \"\"\"\n\n    return await session.get(db.Deployment, deployment_id)\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.read_deployment_by_name","title":"read_deployment_by_name async","text":"

    Reads a deployment by name.

    Parameters:

    Name Type Description Default session Session

    A database session

    required name str

    a deployment name

    required flow_name str

    the name of the flow the deployment belongs to

    required

    Returns:

    Type Description

    db.Deployment: the deployment

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def read_deployment_by_name(\n    session: sa.orm.Session, name: str, flow_name: str, db: PrefectDBInterface\n):\n    \"\"\"Reads a deployment by name.\n\n    Args:\n        session: A database session\n        name: a deployment name\n        flow_name: the name of the flow the deployment belongs to\n\n    Returns:\n        db.Deployment: the deployment\n    \"\"\"\n\n    result = await session.execute(\n        select(db.Deployment)\n        .join(db.Flow, db.Deployment.flow_id == db.Flow.id)\n        .where(\n            sa.and_(\n                db.Flow.name == flow_name,\n                db.Deployment.name == name,\n            )\n        )\n        .limit(1)\n    )\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.read_deployments","title":"read_deployments async","text":"

    Read deployments.

    Parameters:

    Name Type Description Default session Session

    A database session

    required offset int

    Query offset

    None limit int

    Query limit

    None flow_filter FlowFilter

    only select deployments whose flows match these criteria

    None flow_run_filter FlowRunFilter

    only select deployments whose flow runs match these criteria

    None task_run_filter TaskRunFilter

    only select deployments whose task runs match these criteria

    None deployment_filter DeploymentFilter

    only select deployment that match these filters

    None work_pool_filter WorkPoolFilter

    only select deployments whose work pools match these criteria

    None work_queue_filter WorkQueueFilter

    only select deployments whose work pool queues match these criteria

    None sort DeploymentSort

    the sort criteria for selected deployments. Defaults to name ASC.

    NAME_ASC

    Returns:

    Type Description

    List[db.Deployment]: deployments

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def read_deployments(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    offset: int = None,\n    limit: int = None,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    work_queue_filter: schemas.filters.WorkQueueFilter = None,\n    sort: schemas.sorting.DeploymentSort = schemas.sorting.DeploymentSort.NAME_ASC,\n):\n    \"\"\"\n    Read deployments.\n\n    Args:\n        session: A database session\n        offset: Query offset\n        limit: Query limit\n        flow_filter: only select deployments whose flows match these criteria\n        flow_run_filter: only select deployments whose flow runs match these criteria\n        task_run_filter: only select deployments whose task runs match these criteria\n        deployment_filter: only select deployment that match these filters\n        work_pool_filter: only select deployments whose work pools match these criteria\n        work_queue_filter: only select deployments whose work pool queues match these criteria\n        sort: the sort criteria for selected deployments. Defaults to `name` ASC.\n\n    Returns:\n        List[db.Deployment]: deployments\n    \"\"\"\n\n    query = select(db.Deployment).order_by(sort.as_sql_sort(db=db))\n\n    query = await _apply_deployment_filters(\n        query=query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        work_queue_filter=work_queue_filter,\n        db=db,\n    )\n\n    if offset is not None:\n        query = query.offset(offset)\n    if limit is not None:\n        query = query.limit(limit)\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.schedule_runs","title":"schedule_runs async","text":"

    Schedule flow runs for a deployment

    Parameters:

    Name Type Description Default session Session

    a database session

    required deployment_id UUID

    the id of the deployment to schedule

    required start_time datetime

    the time from which to start scheduling runs

    None end_time datetime

    runs will be scheduled until at most this time

    None min_time timedelta

    runs will be scheduled until at least this far in the future

    None min_runs int

    a minimum amount of runs to schedule

    None max_runs int

    a maximum amount of runs to schedule

    None

    This function will generate the minimum number of runs that satisfy the min and max times, and the min and max counts. Specifically, the following order will be respected.

    - Runs will be generated starting on or after the `start_time`\n- No more than `max_runs` runs will be generated\n- No runs will be generated after `end_time` is reached\n- At least `min_runs` runs will be generated\n- Runs will be generated until at least `start_time` + `min_time` is reached\n

    Returns:

    Type Description List[UUID]

    a list of flow run ids scheduled for the deployment

    Source code in prefect/server/models/deployments.py
    async def schedule_runs(\n    session: sa.orm.Session,\n    deployment_id: UUID,\n    start_time: datetime.datetime = None,\n    end_time: datetime.datetime = None,\n    min_time: datetime.timedelta = None,\n    min_runs: int = None,\n    max_runs: int = None,\n    auto_scheduled: bool = True,\n) -> List[UUID]:\n    \"\"\"\n    Schedule flow runs for a deployment\n\n    Args:\n        session: a database session\n        deployment_id: the id of the deployment to schedule\n        start_time: the time from which to start scheduling runs\n        end_time: runs will be scheduled until at most this time\n        min_time: runs will be scheduled until at least this far in the future\n        min_runs: a minimum amount of runs to schedule\n        max_runs: a maximum amount of runs to schedule\n\n    This function will generate the minimum number of runs that satisfy the min\n    and max times, and the min and max counts. Specifically, the following order\n    will be respected.\n\n        - Runs will be generated starting on or after the `start_time`\n        - No more than `max_runs` runs will be generated\n        - No runs will be generated after `end_time` is reached\n        - At least `min_runs` runs will be generated\n        - Runs will be generated until at least `start_time` + `min_time` is reached\n\n    Returns:\n        a list of flow run ids scheduled for the deployment\n    \"\"\"\n    if min_runs is None:\n        min_runs = PREFECT_API_SERVICES_SCHEDULER_MIN_RUNS.value()\n    if max_runs is None:\n        max_runs = PREFECT_API_SERVICES_SCHEDULER_MAX_RUNS.value()\n    if start_time is None:\n        start_time = pendulum.now(\"UTC\")\n    if end_time is None:\n        end_time = start_time + (\n            PREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME.value()\n        )\n    if min_time is None:\n        min_time = PREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME.value()\n\n    start_time = pendulum.instance(start_time)\n    end_time = pendulum.instance(end_time)\n\n    runs = await _generate_scheduled_flow_runs(\n        session=session,\n        deployment_id=deployment_id,\n        start_time=start_time,\n        end_time=end_time,\n        min_time=min_time,\n        min_runs=min_runs,\n        max_runs=max_runs,\n        auto_scheduled=auto_scheduled,\n    )\n    return await _insert_scheduled_flow_runs(session=session, runs=runs)\n
    "},{"location":"api-ref/server/models/deployments/#prefect.server.models.deployments.update_deployment","title":"update_deployment async","text":"

    Updates a deployment.

    Parameters:

    Name Type Description Default session Session

    a database session

    required deployment_id UUID

    the ID of the deployment to modify

    required deployment DeploymentUpdate

    changes to a deployment model

    required

    Returns:

    Name Type Description bool bool

    whether the deployment was updated

    Source code in prefect/server/models/deployments.py
    @inject_db\nasync def update_deployment(\n    session: sa.orm.Session,\n    deployment_id: UUID,\n    deployment: schemas.actions.DeploymentUpdate,\n    db: PrefectDBInterface,\n) -> bool:\n    \"\"\"Updates a deployment.\n\n    Args:\n        session: a database session\n        deployment_id: the ID of the deployment to modify\n        deployment: changes to a deployment model\n\n    Returns:\n        bool: whether the deployment was updated\n\n    \"\"\"\n\n    # exclude_unset=True allows us to only update values provided by\n    # the user, ignoring any defaults on the model\n    update_data = deployment.dict(\n        shallow=True,\n        exclude_unset=True,\n        exclude={\"work_pool_name\"},\n    )\n    if deployment.work_pool_name and deployment.work_queue_name:\n        # If a specific pool name/queue name combination was provided, get the\n        # ID for that work pool queue.\n        update_data[\n            \"work_queue_id\"\n        ] = await WorkerLookups()._get_work_queue_id_from_name(\n            session=session,\n            work_pool_name=deployment.work_pool_name,\n            work_queue_name=deployment.work_queue_name,\n            create_queue_if_not_found=True,\n        )\n    elif deployment.work_pool_name:\n        # If just a pool name was provided, get the ID for its default\n        # work pool queue.\n        update_data[\n            \"work_queue_id\"\n        ] = await WorkerLookups()._get_default_work_queue_id_from_work_pool_name(\n            session=session,\n            work_pool_name=deployment.work_pool_name,\n        )\n    elif deployment.work_queue_name:\n        # If just a queue name was provided, ensure the queue exists and\n        # get its ID.\n        work_queue = await models.work_queues._ensure_work_queue_exists(\n            session=session, name=update_data[\"work_queue_name\"], db=db\n        )\n        update_data[\"work_queue_id\"] = work_queue.id\n\n    update_stmt = (\n        sa.update(db.Deployment)\n        .where(db.Deployment.id == deployment_id)\n        .values(**update_data)\n    )\n    result = await session.execute(update_stmt)\n\n    # delete any auto scheduled runs that would have reflected the old deployment config\n    await _delete_scheduled_runs(\n        session=session, deployment_id=deployment_id, db=db, auto_scheduled_only=True\n    )\n\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flow_run_states/","title":"server.models.flow_run_states","text":""},{"location":"api-ref/server/models/flow_run_states/#prefect.server.models.flow_run_states","title":"prefect.server.models.flow_run_states","text":"

    Functions for interacting with flow run state ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/flow_run_states/#prefect.server.models.flow_run_states.delete_flow_run_state","title":"delete_flow_run_state async","text":"

    Delete a flow run state by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_run_state_id UUID

    a flow run state id

    required

    Returns:

    Name Type Description bool bool

    whether or not the flow run state was deleted

    Source code in prefect/server/models/flow_run_states.py
    @inject_db\nasync def delete_flow_run_state(\n    session: sa.orm.Session, flow_run_state_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a flow run state by id.\n\n    Args:\n        session: A database session\n        flow_run_state_id: a flow run state id\n\n    Returns:\n        bool: whether or not the flow run state was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.FlowRunState).where(db.FlowRunState.id == flow_run_state_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flow_run_states/#prefect.server.models.flow_run_states.read_flow_run_state","title":"read_flow_run_state async","text":"

    Reads a flow run state by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_run_state_id UUID

    a flow run state id

    required

    Returns:

    Type Description

    db.FlowRunState: the flow state

    Source code in prefect/server/models/flow_run_states.py
    @inject_db\nasync def read_flow_run_state(\n    session: sa.orm.Session, flow_run_state_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads a flow run state by id.\n\n    Args:\n        session: A database session\n        flow_run_state_id: a flow run state id\n\n    Returns:\n        db.FlowRunState: the flow state\n    \"\"\"\n\n    return await session.get(db.FlowRunState, flow_run_state_id)\n
    "},{"location":"api-ref/server/models/flow_run_states/#prefect.server.models.flow_run_states.read_flow_run_states","title":"read_flow_run_states async","text":"

    Reads flow runs states for a flow run.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_run_id UUID

    the flow run id

    required

    Returns:

    Type Description

    List[db.FlowRunState]: the flow run states

    Source code in prefect/server/models/flow_run_states.py
    @inject_db\nasync def read_flow_run_states(\n    session: sa.orm.Session, flow_run_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads flow runs states for a flow run.\n\n    Args:\n        session: A database session\n        flow_run_id: the flow run id\n\n    Returns:\n        List[db.FlowRunState]: the flow run states\n    \"\"\"\n\n    query = (\n        select(db.FlowRunState)\n        .filter_by(flow_run_id=flow_run_id)\n        .order_by(db.FlowRunState.timestamp)\n    )\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/flow_runs/","title":"server.models.flow_runs","text":""},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs","title":"prefect.server.models.flow_runs","text":"

    Functions for interacting with flow run ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.count_flow_runs","title":"count_flow_runs async","text":"

    Count flow runs.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required flow_filter FlowFilter

    only count flow runs whose flows match these filters

    None flow_run_filter FlowRunFilter

    only count flow runs that match these filters

    None task_run_filter TaskRunFilter

    only count flow runs whose task runs match these filters

    None deployment_filter DeploymentFilter

    only count flow runs whose deployments match these filters

    None

    Returns:

    Name Type Description int int

    count of flow runs

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def count_flow_runs(\n    session: AsyncSession,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    work_queue_filter: schemas.filters.WorkQueueFilter = None,\n) -> int:\n    \"\"\"\n    Count flow runs.\n\n    Args:\n        session: a database session\n        flow_filter: only count flow runs whose flows match these filters\n        flow_run_filter: only count flow runs that match these filters\n        task_run_filter: only count flow runs whose task runs match these filters\n        deployment_filter: only count flow runs whose deployments match these filters\n\n    Returns:\n        int: count of flow runs\n    \"\"\"\n\n    query = select(sa.func.count(sa.text(\"*\"))).select_from(db.FlowRun)\n\n    query = await _apply_flow_run_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        work_queue_filter=work_queue_filter,\n        db=db,\n    )\n\n    result = await session.execute(query)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.create_flow_run","title":"create_flow_run async","text":"

    Creates a new flow run.

    If the provided flow run has a state attached, it will also be created.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required flow_run FlowRun

    a flow run model

    required

    Returns:

    Type Description

    db.FlowRun: the newly-created flow run

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def create_flow_run(\n    session: AsyncSession,\n    flow_run: schemas.core.FlowRun,\n    db: PrefectDBInterface,\n    orchestration_parameters: Optional[dict] = None,\n):\n    \"\"\"Creates a new flow run.\n\n    If the provided flow run has a state attached, it will also be created.\n\n    Args:\n        session: a database session\n        flow_run: a flow run model\n\n    Returns:\n        db.FlowRun: the newly-created flow run\n    \"\"\"\n    now = pendulum.now(\"UTC\")\n\n    flow_run_dict = dict(\n        **flow_run.dict(\n            shallow=True,\n            exclude={\n                \"created\",\n                \"state\",\n                \"estimated_run_time\",\n                \"estimated_start_time_delta\",\n            },\n            exclude_unset=True,\n        ),\n        created=now,\n    )\n\n    # if no idempotency key was provided, create the run directly\n    if not flow_run.idempotency_key:\n        model = db.FlowRun(**flow_run_dict)\n        session.add(model)\n        await session.flush()\n\n    # otherwise let the database take care of enforcing idempotency\n    else:\n        insert_stmt = (\n            (await db.insert(db.FlowRun))\n            .values(**flow_run_dict)\n            .on_conflict_do_nothing(\n                index_elements=db.flow_run_unique_upsert_columns,\n            )\n        )\n        await session.execute(insert_stmt)\n\n        # read the run to see if idempotency was applied or not\n        query = (\n            sa.select(db.FlowRun)\n            .where(\n                sa.and_(\n                    db.FlowRun.flow_id == flow_run.flow_id,\n                    db.FlowRun.idempotency_key == flow_run.idempotency_key,\n                )\n            )\n            .limit(1)\n            .execution_options(populate_existing=True)\n            .options(\n                selectinload(db.FlowRun.work_queue).selectinload(db.WorkQueue.work_pool)\n            )\n        )\n        result = await session.execute(query)\n        model = result.scalar()\n\n    # if the flow run was created in this function call then we need to set the\n    # state. If it was created idempotently, the created time won't match.\n    if model.created == now and flow_run.state:\n        await models.flow_runs.set_flow_run_state(\n            session=session,\n            flow_run_id=model.id,\n            state=flow_run.state,\n            force=True,\n            orchestration_parameters=orchestration_parameters,\n        )\n    return model\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.delete_flow_run","title":"delete_flow_run async","text":"

    Delete a flow run by flow_run_id.

    Parameters:

    Name Type Description Default session AsyncSession

    A database session

    required flow_run_id UUID

    a flow run id

    required

    Returns:

    Name Type Description bool bool

    whether or not the flow run was deleted

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def delete_flow_run(\n    session: AsyncSession, flow_run_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a flow run by flow_run_id.\n\n    Args:\n        session: A database session\n        flow_run_id: a flow run id\n\n    Returns:\n        bool: whether or not the flow run was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.FlowRun).where(db.FlowRun.id == flow_run_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.read_flow_run","title":"read_flow_run async","text":"

    Reads a flow run by id.

    Parameters:

    Name Type Description Default session AsyncSession

    A database session

    required flow_run_id UUID

    a flow run id

    required

    Returns:

    Type Description

    db.FlowRun: the flow run

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def read_flow_run(\n    session: AsyncSession,\n    flow_run_id: UUID,\n    db: PrefectDBInterface,\n    for_update: bool = False,\n):\n    \"\"\"\n    Reads a flow run by id.\n\n    Args:\n        session: A database session\n        flow_run_id: a flow run id\n\n    Returns:\n        db.FlowRun: the flow run\n    \"\"\"\n    select = (\n        sa.select(db.FlowRun)\n        .where(db.FlowRun.id == flow_run_id)\n        .options(\n            selectinload(db.FlowRun.work_queue).selectinload(db.WorkQueue.work_pool)\n        )\n    )\n\n    if for_update:\n        select = select.with_for_update()\n\n    result = await session.execute(select)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.read_flow_run_graph","title":"read_flow_run_graph async","text":"

    Given a flow run, return the graph of it's task and subflow runs. If a since datetime is provided, only return items that may have changed since that time.

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def read_flow_run_graph(\n    db: PrefectDBInterface,\n    session: AsyncSession,\n    flow_run_id: UUID,\n    since: datetime.datetime = datetime.datetime.min,\n) -> Graph:\n    \"\"\"Given a flow run, return the graph of it's task and subflow runs. If a `since`\n    datetime is provided, only return items that may have changed since that time.\"\"\"\n    return await db.queries.flow_run_graph_v2(\n        db=db,\n        session=session,\n        flow_run_id=flow_run_id,\n        since=since,\n        max_nodes=PREFECT_API_MAX_FLOW_RUN_GRAPH_NODES.value(),\n    )\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.read_flow_runs","title":"read_flow_runs async","text":"

    Read flow runs.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required columns List

    a list of the flow run ORM columns to load, for performance

    None flow_filter FlowFilter

    only select flow runs whose flows match these filters

    None flow_run_filter FlowRunFilter

    only select flow runs match these filters

    None task_run_filter TaskRunFilter

    only select flow runs whose task runs match these filters

    None deployment_filter DeploymentFilter

    only select flow runs whose deployments match these filters

    None offset int

    Query offset

    None limit int

    Query limit

    None sort FlowRunSort

    Query sort

    ID_DESC

    Returns:

    Type Description

    List[db.FlowRun]: flow runs

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def read_flow_runs(\n    session: AsyncSession,\n    db: PrefectDBInterface,\n    columns: List = None,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    work_queue_filter: schemas.filters.WorkQueueFilter = None,\n    offset: int = None,\n    limit: int = None,\n    sort: schemas.sorting.FlowRunSort = schemas.sorting.FlowRunSort.ID_DESC,\n):\n    \"\"\"\n    Read flow runs.\n\n    Args:\n        session: a database session\n        columns: a list of the flow run ORM columns to load, for performance\n        flow_filter: only select flow runs whose flows match these filters\n        flow_run_filter: only select flow runs match these filters\n        task_run_filter: only select flow runs whose task runs match these filters\n        deployment_filter: only select flow runs whose deployments match these filters\n        offset: Query offset\n        limit: Query limit\n        sort: Query sort\n\n    Returns:\n        List[db.FlowRun]: flow runs\n    \"\"\"\n    query = (\n        select(db.FlowRun)\n        .order_by(sort.as_sql_sort(db))\n        .options(\n            selectinload(db.FlowRun.work_queue).selectinload(db.WorkQueue.work_pool)\n        )\n    )\n\n    if columns:\n        query = query.options(load_only(*columns))\n\n    query = await _apply_flow_run_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        work_queue_filter=work_queue_filter,\n        db=db,\n    )\n\n    if offset is not None:\n        query = query.offset(offset)\n\n    if limit is not None:\n        query = query.limit(limit)\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.read_task_run_dependencies","title":"read_task_run_dependencies async","text":"

    Get a task run dependency map for a given flow run.

    Source code in prefect/server/models/flow_runs.py
    async def read_task_run_dependencies(\n    session: AsyncSession,\n    flow_run_id: UUID,\n) -> List[DependencyResult]:\n    \"\"\"\n    Get a task run dependency map for a given flow run.\n    \"\"\"\n    flow_run = await models.flow_runs.read_flow_run(\n        session=session, flow_run_id=flow_run_id\n    )\n    if not flow_run:\n        raise ObjectNotFoundError(f\"Flow run with id {flow_run_id} not found\")\n\n    task_runs = await models.task_runs.read_task_runs(\n        session=session,\n        flow_run_filter=schemas.filters.FlowRunFilter(\n            id=schemas.filters.FlowRunFilterId(any_=[flow_run_id])\n        ),\n    )\n\n    dependency_graph = []\n\n    for task_run in task_runs:\n        inputs = list(set(chain(*task_run.task_inputs.values())))\n        untrackable_result_status = (\n            False\n            if task_run.state is None\n            else task_run.state.state_details.untrackable_result\n        )\n        dependency_graph.append(\n            {\n                \"id\": task_run.id,\n                \"upstream_dependencies\": inputs,\n                \"state\": task_run.state,\n                \"expected_start_time\": task_run.expected_start_time,\n                \"name\": task_run.name,\n                \"start_time\": task_run.start_time,\n                \"end_time\": task_run.end_time,\n                \"total_run_time\": task_run.total_run_time,\n                \"estimated_run_time\": task_run.estimated_run_time,\n                \"untrackable_result\": untrackable_result_status,\n            }\n        )\n\n    return dependency_graph\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.set_flow_run_state","title":"set_flow_run_state async","text":"

    Creates a new orchestrated flow run state.

    Setting a new state on a run is the one of the principal actions that is governed by Prefect's orchestration logic. Setting a new run state will not guarantee creation, but instead trigger orchestration rules to govern the proposed state input. If the state is considered valid, it will be written to the database. Otherwise, a it's possible a different state, or no state, will be created. A force flag is supplied to bypass a subset of orchestration logic.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required flow_run_id UUID

    the flow run id

    required state State

    a flow run state model

    required force bool

    if False, orchestration rules will be applied that may alter or prevent the state transition. If True, orchestration rules are not applied.

    False

    Returns:

    Type Description OrchestrationResult

    OrchestrationResult object

    Source code in prefect/server/models/flow_runs.py
    async def set_flow_run_state(\n    session: AsyncSession,\n    flow_run_id: UUID,\n    state: schemas.states.State,\n    force: bool = False,\n    flow_policy: BaseOrchestrationPolicy = None,\n    orchestration_parameters: dict = None,\n) -> OrchestrationResult:\n    \"\"\"\n    Creates a new orchestrated flow run state.\n\n    Setting a new state on a run is the one of the principal actions that is governed by\n    Prefect's orchestration logic. Setting a new run state will not guarantee creation,\n    but instead trigger orchestration rules to govern the proposed `state` input. If\n    the state is considered valid, it will be written to the database. Otherwise, a\n    it's possible a different state, or no state, will be created. A `force` flag is\n    supplied to bypass a subset of orchestration logic.\n\n    Args:\n        session: a database session\n        flow_run_id: the flow run id\n        state: a flow run state model\n        force: if False, orchestration rules will be applied that may alter or prevent\n            the state transition. If True, orchestration rules are not applied.\n\n    Returns:\n        OrchestrationResult object\n    \"\"\"\n\n    # load the flow run\n    run = await models.flow_runs.read_flow_run(\n        session=session,\n        flow_run_id=flow_run_id,\n        # Lock the row to prevent orchestration race conditions\n        for_update=True,\n    )\n\n    if not run:\n        raise ObjectNotFoundError(f\"Flow run with id {flow_run_id} not found\")\n\n    initial_state = run.state.as_state() if run.state else None\n    initial_state_type = initial_state.type if initial_state else None\n    proposed_state_type = state.type if state else None\n    intended_transition = (initial_state_type, proposed_state_type)\n\n    if force or flow_policy is None:\n        flow_policy = MinimalFlowPolicy\n\n    orchestration_rules = flow_policy.compile_transition_rules(*intended_transition)\n    global_rules = GlobalFlowPolicy.compile_transition_rules(*intended_transition)\n\n    context = FlowOrchestrationContext(\n        session=session,\n        run=run,\n        initial_state=initial_state,\n        proposed_state=state,\n    )\n\n    if orchestration_parameters is not None:\n        context.parameters = orchestration_parameters\n\n    # apply orchestration rules and create the new flow run state\n    async with contextlib.AsyncExitStack() as stack:\n        for rule in orchestration_rules:\n            context = await stack.enter_async_context(\n                rule(context, *intended_transition)\n            )\n\n        for rule in global_rules:\n            context = await stack.enter_async_context(\n                rule(context, *intended_transition)\n            )\n\n        await context.validate_proposed_state()\n\n    if context.orchestration_error is not None:\n        raise context.orchestration_error\n\n    result = OrchestrationResult(\n        state=context.validated_state,\n        status=context.response_status,\n        details=context.response_details,\n    )\n\n    # if a new state is being set (either ACCEPTED from user or REJECTED\n    # and set by the server), check for any notification policies\n    if result.status in (SetStateStatus.ACCEPT, SetStateStatus.REJECT):\n        await models.flow_run_notification_policies.queue_flow_run_notifications(\n            session=session, flow_run=run\n        )\n\n    return result\n
    "},{"location":"api-ref/server/models/flow_runs/#prefect.server.models.flow_runs.update_flow_run","title":"update_flow_run async","text":"

    Updates a flow run.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required flow_run_id UUID

    the flow run id to update

    required flow_run FlowRunUpdate

    a flow run model

    required

    Returns:

    Name Type Description bool bool

    whether or not matching rows were found to update

    Source code in prefect/server/models/flow_runs.py
    @inject_db\nasync def update_flow_run(\n    session: AsyncSession,\n    flow_run_id: UUID,\n    flow_run: schemas.actions.FlowRunUpdate,\n    db: PrefectDBInterface,\n) -> bool:\n    \"\"\"\n    Updates a flow run.\n\n    Args:\n        session: a database session\n        flow_run_id: the flow run id to update\n        flow_run: a flow run model\n\n    Returns:\n        bool: whether or not matching rows were found to update\n    \"\"\"\n    update_stmt = (\n        sa.update(db.FlowRun).where(db.FlowRun.id == flow_run_id)\n        # exclude_unset=True allows us to only update values provided by\n        # the user, ignoring any defaults on the model\n        .values(**flow_run.dict(shallow=True, exclude_unset=True))\n    )\n    result = await session.execute(update_stmt)\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flows/","title":"server.models.flows","text":""},{"location":"api-ref/server/models/flows/#prefect.server.models.flows","title":"prefect.server.models.flows","text":"

    Functions for interacting with flow ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.count_flows","title":"count_flows async","text":"

    Count flows.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_filter FlowFilter

    only count flows that match these filters

    None flow_run_filter FlowRunFilter

    only count flows whose flow runs match these filters

    None task_run_filter TaskRunFilter

    only count flows whose task runs match these filters

    None deployment_filter DeploymentFilter

    only count flows whose deployments match these filters

    None work_pool_filter WorkPoolFilter

    only count flows whose work pools match these filters

    None

    Returns:

    Name Type Description int int

    count of flows

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def count_flows(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n) -> int:\n    \"\"\"\n    Count flows.\n\n    Args:\n        session: A database session\n        flow_filter: only count flows that match these filters\n        flow_run_filter: only count flows whose flow runs match these filters\n        task_run_filter: only count flows whose task runs match these filters\n        deployment_filter: only count flows whose deployments match these filters\n        work_pool_filter: only count flows whose work pools match these filters\n\n    Returns:\n        int: count of flows\n    \"\"\"\n\n    query = select(sa.func.count(sa.text(\"*\"))).select_from(db.Flow)\n\n    query = await _apply_flow_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        db=db,\n    )\n\n    result = await session.execute(query)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.create_flow","title":"create_flow async","text":"

    Creates a new flow.

    If a flow with the same name already exists, the existing flow is returned.

    Parameters:

    Name Type Description Default session Session

    a database session

    required flow Flow

    a flow model

    required

    Returns:

    Type Description

    db.Flow: the newly-created or existing flow

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def create_flow(\n    session: sa.orm.Session, flow: schemas.core.Flow, db: PrefectDBInterface\n):\n    \"\"\"\n    Creates a new flow.\n\n    If a flow with the same name already exists, the existing flow is returned.\n\n    Args:\n        session: a database session\n        flow: a flow model\n\n    Returns:\n        db.Flow: the newly-created or existing flow\n    \"\"\"\n\n    insert_stmt = (\n        (await db.insert(db.Flow))\n        .values(**flow.dict(shallow=True, exclude_unset=True))\n        .on_conflict_do_nothing(\n            index_elements=db.flow_unique_upsert_columns,\n        )\n    )\n    await session.execute(insert_stmt)\n\n    query = (\n        sa.select(db.Flow)\n        .where(\n            db.Flow.name == flow.name,\n        )\n        .limit(1)\n        .execution_options(populate_existing=True)\n    )\n    result = await session.execute(query)\n    model = result.scalar()\n    return model\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.delete_flow","title":"delete_flow async","text":"

    Delete a flow by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_id UUID

    a flow id

    required

    Returns:

    Name Type Description bool bool

    whether or not the flow was deleted

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def delete_flow(\n    session: sa.orm.Session, flow_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a flow by id.\n\n    Args:\n        session: A database session\n        flow_id: a flow id\n\n    Returns:\n        bool: whether or not the flow was deleted\n    \"\"\"\n\n    result = await session.execute(delete(db.Flow).where(db.Flow.id == flow_id))\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.read_flow","title":"read_flow async","text":"

    Reads a flow by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_id UUID

    a flow id

    required

    Returns:

    Type Description

    db.Flow: the flow

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def read_flow(session: sa.orm.Session, flow_id: UUID, db: PrefectDBInterface):\n    \"\"\"\n    Reads a flow by id.\n\n    Args:\n        session: A database session\n        flow_id: a flow id\n\n    Returns:\n        db.Flow: the flow\n    \"\"\"\n    return await session.get(db.Flow, flow_id)\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.read_flow_by_name","title":"read_flow_by_name async","text":"

    Reads a flow by name.

    Parameters:

    Name Type Description Default session Session

    A database session

    required name str

    a flow name

    required

    Returns:

    Type Description

    db.Flow: the flow

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def read_flow_by_name(session: sa.orm.Session, name: str, db: PrefectDBInterface):\n    \"\"\"\n    Reads a flow by name.\n\n    Args:\n        session: A database session\n        name: a flow name\n\n    Returns:\n        db.Flow: the flow\n    \"\"\"\n\n    result = await session.execute(select(db.Flow).filter_by(name=name))\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.read_flows","title":"read_flows async","text":"

    Read multiple flows.

    Parameters:

    Name Type Description Default session Session

    A database session

    required flow_filter FlowFilter

    only select flows that match these filters

    None flow_run_filter FlowRunFilter

    only select flows whose flow runs match these filters

    None task_run_filter TaskRunFilter

    only select flows whose task runs match these filters

    None deployment_filter DeploymentFilter

    only select flows whose deployments match these filters

    None work_pool_filter WorkPoolFilter

    only select flows whose work pools match these filters

    None offset int

    Query offset

    None limit int

    Query limit

    None

    Returns:

    Type Description

    List[db.Flow]: flows

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def read_flows(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    work_pool_filter: schemas.filters.WorkPoolFilter = None,\n    sort: schemas.sorting.FlowSort = schemas.sorting.FlowSort.NAME_ASC,\n    offset: int = None,\n    limit: int = None,\n):\n    \"\"\"\n    Read multiple flows.\n\n    Args:\n        session: A database session\n        flow_filter: only select flows that match these filters\n        flow_run_filter: only select flows whose flow runs match these filters\n        task_run_filter: only select flows whose task runs match these filters\n        deployment_filter: only select flows whose deployments match these filters\n        work_pool_filter: only select flows whose work pools match these filters\n        offset: Query offset\n        limit: Query limit\n\n    Returns:\n        List[db.Flow]: flows\n    \"\"\"\n\n    query = select(db.Flow).order_by(sort.as_sql_sort(db=db))\n\n    query = await _apply_flow_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        work_pool_filter=work_pool_filter,\n        db=db,\n    )\n\n    if offset is not None:\n        query = query.offset(offset)\n\n    if limit is not None:\n        query = query.limit(limit)\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/flows/#prefect.server.models.flows.update_flow","title":"update_flow async","text":"

    Updates a flow.

    Parameters:

    Name Type Description Default session Session

    a database session

    required flow_id UUID

    the flow id to update

    required flow FlowUpdate

    a flow update model

    required

    Returns:

    Name Type Description bool

    whether or not matching rows were found to update

    Source code in prefect/server/models/flows.py
    @inject_db\nasync def update_flow(\n    session: sa.orm.Session,\n    flow_id: UUID,\n    flow: schemas.actions.FlowUpdate,\n    db: PrefectDBInterface,\n):\n    \"\"\"\n    Updates a flow.\n\n    Args:\n        session: a database session\n        flow_id: the flow id to update\n        flow: a flow update model\n\n    Returns:\n        bool: whether or not matching rows were found to update\n    \"\"\"\n    update_stmt = (\n        sa.update(db.Flow).where(db.Flow.id == flow_id)\n        # exclude_unset=True allows us to only update values provided by\n        # the user, ignoring any defaults on the model\n        .values(**flow.dict(shallow=True, exclude_unset=True))\n    )\n    result = await session.execute(update_stmt)\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/saved_searches/","title":"server.models.saved_searches","text":""},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches","title":"prefect.server.models.saved_searches","text":"

    Functions for interacting with saved search ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.create_saved_search","title":"create_saved_search async","text":"

    Upserts a SavedSearch.

    If a SavedSearch with the same name exists, all properties will be updated.

    Parameters:

    Name Type Description Default session Session

    a database session

    required saved_search SavedSearch

    a SavedSearch model

    required

    Returns:

    Type Description

    db.SavedSearch: the newly-created or updated SavedSearch

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def create_saved_search(\n    session: sa.orm.Session,\n    saved_search: schemas.core.SavedSearch,\n    db: PrefectDBInterface,\n):\n    \"\"\"\n    Upserts a SavedSearch.\n\n    If a SavedSearch with the same name exists, all properties will be updated.\n\n    Args:\n        session (sa.orm.Session): a database session\n        saved_search (schemas.core.SavedSearch): a SavedSearch model\n\n    Returns:\n        db.SavedSearch: the newly-created or updated SavedSearch\n\n    \"\"\"\n\n    insert_stmt = (\n        (await db.insert(db.SavedSearch))\n        .values(**saved_search.dict(shallow=True, exclude_unset=True))\n        .on_conflict_do_update(\n            index_elements=db.saved_search_unique_upsert_columns,\n            set_=saved_search.dict(shallow=True, include={\"filters\"}),\n        )\n    )\n\n    await session.execute(insert_stmt)\n\n    query = (\n        sa.select(db.SavedSearch)\n        .where(\n            db.SavedSearch.name == saved_search.name,\n        )\n        .execution_options(populate_existing=True)\n    )\n    result = await session.execute(query)\n    model = result.scalar()\n\n    return model\n
    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.delete_saved_search","title":"delete_saved_search async","text":"

    Delete a SavedSearch by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required saved_search_id str

    a SavedSearch id

    required

    Returns:

    Name Type Description bool bool

    whether or not the SavedSearch was deleted

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def delete_saved_search(\n    session: sa.orm.Session, saved_search_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a SavedSearch by id.\n\n    Args:\n        session (sa.orm.Session): A database session\n        saved_search_id (str): a SavedSearch id\n\n    Returns:\n        bool: whether or not the SavedSearch was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.SavedSearch).where(db.SavedSearch.id == saved_search_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.read_saved_search","title":"read_saved_search async","text":"

    Reads a SavedSearch by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required saved_search_id str

    a SavedSearch id

    required

    Returns:

    Type Description

    db.SavedSearch: the SavedSearch

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def read_saved_search(\n    session: sa.orm.Session, saved_search_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads a SavedSearch by id.\n\n    Args:\n        session (sa.orm.Session): A database session\n        saved_search_id (str): a SavedSearch id\n\n    Returns:\n        db.SavedSearch: the SavedSearch\n    \"\"\"\n\n    return await session.get(db.SavedSearch, saved_search_id)\n
    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.read_saved_search_by_name","title":"read_saved_search_by_name async","text":"

    Reads a SavedSearch by name.

    Parameters:

    Name Type Description Default session Session

    A database session

    required name str

    a SavedSearch name

    required

    Returns:

    Type Description

    db.SavedSearch: the SavedSearch

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def read_saved_search_by_name(\n    session: sa.orm.Session, name: str, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads a SavedSearch by name.\n\n    Args:\n        session (sa.orm.Session): A database session\n        name (str): a SavedSearch name\n\n    Returns:\n        db.SavedSearch: the SavedSearch\n    \"\"\"\n    result = await session.execute(\n        select(db.SavedSearch).where(db.SavedSearch.name == name).limit(1)\n    )\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/saved_searches/#prefect.server.models.saved_searches.read_saved_searches","title":"read_saved_searches async","text":"

    Read SavedSearches.

    Parameters:

    Name Type Description Default session Session

    A database session

    required offset int

    Query offset

    None limit(int)

    Query limit

    required

    Returns:

    Type Description

    List[db.SavedSearch]: SavedSearches

    Source code in prefect/server/models/saved_searches.py
    @inject_db\nasync def read_saved_searches(\n    db: PrefectDBInterface,\n    session: sa.orm.Session,\n    offset: int = None,\n    limit: int = None,\n):\n    \"\"\"\n    Read SavedSearches.\n\n    Args:\n        session (sa.orm.Session): A database session\n        offset (int): Query offset\n        limit(int): Query limit\n\n    Returns:\n        List[db.SavedSearch]: SavedSearches\n    \"\"\"\n\n    query = select(db.SavedSearch).order_by(db.SavedSearch.name)\n\n    if offset is not None:\n        query = query.offset(offset)\n    if limit is not None:\n        query = query.limit(limit)\n\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/task_run_states/","title":"server.models.task_run_states","text":""},{"location":"api-ref/server/models/task_run_states/#prefect.server.models.task_run_states","title":"prefect.server.models.task_run_states","text":"

    Functions for interacting with task run state ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/task_run_states/#prefect.server.models.task_run_states.delete_task_run_state","title":"delete_task_run_state async","text":"

    Delete a task run state by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required task_run_state_id UUID

    a task run state id

    required

    Returns:

    Name Type Description bool bool

    whether or not the task run state was deleted

    Source code in prefect/server/models/task_run_states.py
    @inject_db\nasync def delete_task_run_state(\n    session: sa.orm.Session, task_run_state_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a task run state by id.\n\n    Args:\n        session: A database session\n        task_run_state_id: a task run state id\n\n    Returns:\n        bool: whether or not the task run state was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.TaskRunState).where(db.TaskRunState.id == task_run_state_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/task_run_states/#prefect.server.models.task_run_states.read_task_run_state","title":"read_task_run_state async","text":"

    Reads a task run state by id.

    Parameters:

    Name Type Description Default session Session

    A database session

    required task_run_state_id UUID

    a task run state id

    required

    Returns:

    Type Description

    db.TaskRunState: the task state

    Source code in prefect/server/models/task_run_states.py
    @inject_db\nasync def read_task_run_state(\n    session: sa.orm.Session, task_run_state_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads a task run state by id.\n\n    Args:\n        session: A database session\n        task_run_state_id: a task run state id\n\n    Returns:\n        db.TaskRunState: the task state\n    \"\"\"\n\n    return await session.get(db.TaskRunState, task_run_state_id)\n
    "},{"location":"api-ref/server/models/task_run_states/#prefect.server.models.task_run_states.read_task_run_states","title":"read_task_run_states async","text":"

    Reads task runs states for a task run.

    Parameters:

    Name Type Description Default session Session

    A database session

    required task_run_id UUID

    the task run id

    required

    Returns:

    Type Description

    List[db.TaskRunState]: the task run states

    Source code in prefect/server/models/task_run_states.py
    @inject_db\nasync def read_task_run_states(\n    session: sa.orm.Session, task_run_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Reads task runs states for a task run.\n\n    Args:\n        session: A database session\n        task_run_id: the task run id\n\n    Returns:\n        List[db.TaskRunState]: the task run states\n    \"\"\"\n\n    query = (\n        select(db.TaskRunState)\n        .filter_by(task_run_id=task_run_id)\n        .order_by(db.TaskRunState.timestamp)\n    )\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/task_runs/","title":"server.models.task_runs","text":""},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs","title":"prefect.server.models.task_runs","text":"

    Functions for interacting with task run ORM objects. Intended for internal use by the Prefect REST API.

    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.count_task_runs","title":"count_task_runs async","text":"

    Count task runs.

    Parameters:

    Name Type Description Default session Session

    a database session

    required flow_filter FlowFilter

    only count task runs whose flows match these filters

    None flow_run_filter FlowRunFilter

    only count task runs whose flow runs match these filters

    None task_run_filter TaskRunFilter

    only count task runs that match these filters

    None deployment_filter DeploymentFilter

    only count task runs whose deployments match these filters

    None Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def count_task_runs(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n) -> int:\n    \"\"\"\n    Count task runs.\n\n    Args:\n        session: a database session\n        flow_filter: only count task runs whose flows match these filters\n        flow_run_filter: only count task runs whose flow runs match these filters\n        task_run_filter: only count task runs that match these filters\n        deployment_filter: only count task runs whose deployments match these filters\n    Returns:\n        int: count of task runs\n    \"\"\"\n\n    query = select(sa.func.count(sa.text(\"*\"))).select_from(db.TaskRun)\n\n    query = await _apply_task_run_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        db=db,\n    )\n\n    result = await session.execute(query)\n    return result.scalar()\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.create_task_run","title":"create_task_run async","text":"

    Creates a new task run.

    If a task run with the same flow_run_id, task_key, and dynamic_key already exists, the existing task run will be returned. If the provided task run has a state attached, it will also be created.

    Parameters:

    Name Type Description Default session Session

    a database session

    required task_run TaskRun

    a task run model

    required

    Returns:

    Type Description

    db.TaskRun: the newly-created or existing task run

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def create_task_run(\n    session: sa.orm.Session,\n    task_run: schemas.core.TaskRun,\n    db: PrefectDBInterface,\n    orchestration_parameters: dict = None,\n):\n    \"\"\"\n    Creates a new task run.\n\n    If a task run with the same flow_run_id, task_key, and dynamic_key already exists,\n    the existing task run will be returned. If the provided task run has a state\n    attached, it will also be created.\n\n    Args:\n        session: a database session\n        task_run: a task run model\n\n    Returns:\n        db.TaskRun: the newly-created or existing task run\n    \"\"\"\n\n    now = pendulum.now(\"UTC\")\n\n    # if a dynamic key exists, we need to guard against conflicts\n    if task_run.flow_run_id:\n        insert_stmt = (\n            (await db.insert(db.TaskRun))\n            .values(\n                created=now,\n                **task_run.dict(\n                    shallow=True, exclude={\"state\", \"created\"}, exclude_unset=True\n                ),\n            )\n            .on_conflict_do_nothing(\n                index_elements=db.task_run_unique_upsert_columns,\n            )\n        )\n        await session.execute(insert_stmt)\n\n        query = (\n            sa.select(db.TaskRun)\n            .where(\n                sa.and_(\n                    db.TaskRun.flow_run_id == task_run.flow_run_id,\n                    db.TaskRun.task_key == task_run.task_key,\n                    db.TaskRun.dynamic_key == task_run.dynamic_key,\n                )\n            )\n            .limit(1)\n            .execution_options(populate_existing=True)\n        )\n        result = await session.execute(query)\n        model = result.scalar()\n    else:\n        # Upsert on (task_key, dynamic_key) application logic.\n        query = (\n            sa.select(db.TaskRun)\n            .where(\n                sa.and_(\n                    db.TaskRun.flow_run_id.is_(None),\n                    db.TaskRun.task_key == task_run.task_key,\n                    db.TaskRun.dynamic_key == task_run.dynamic_key,\n                )\n            )\n            .limit(1)\n            .execution_options(populate_existing=True)\n        )\n\n        result = await session.execute(query)\n        model = result.scalar()\n\n        if model is None:\n            model = db.TaskRun(\n                created=now,\n                **task_run.dict(\n                    shallow=True, exclude={\"state\", \"created\"}, exclude_unset=True\n                ),\n                state=None,\n            )\n            session.add(model)\n            await session.flush()\n\n    if model.created == now and task_run.state:\n        await models.task_runs.set_task_run_state(\n            session=session,\n            task_run_id=model.id,\n            state=task_run.state,\n            force=True,\n            orchestration_parameters=orchestration_parameters,\n        )\n    return model\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.delete_task_run","title":"delete_task_run async","text":"

    Delete a task run by id.

    Parameters:

    Name Type Description Default session Session

    a database session

    required task_run_id UUID

    the task run id to delete

    required

    Returns:

    Name Type Description bool bool

    whether or not the task run was deleted

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def delete_task_run(\n    session: sa.orm.Session, task_run_id: UUID, db: PrefectDBInterface\n) -> bool:\n    \"\"\"\n    Delete a task run by id.\n\n    Args:\n        session: a database session\n        task_run_id: the task run id to delete\n\n    Returns:\n        bool: whether or not the task run was deleted\n    \"\"\"\n\n    result = await session.execute(\n        delete(db.TaskRun).where(db.TaskRun.id == task_run_id)\n    )\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.read_task_run","title":"read_task_run async","text":"

    Read a task run by id.

    Parameters:

    Name Type Description Default session Session

    a database session

    required task_run_id UUID

    the task run id

    required

    Returns:

    Type Description

    db.TaskRun: the task run

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def read_task_run(\n    session: sa.orm.Session, task_run_id: UUID, db: PrefectDBInterface\n):\n    \"\"\"\n    Read a task run by id.\n\n    Args:\n        session: a database session\n        task_run_id: the task run id\n\n    Returns:\n        db.TaskRun: the task run\n    \"\"\"\n\n    model = await session.get(db.TaskRun, task_run_id)\n    return model\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.read_task_runs","title":"read_task_runs async","text":"

    Read task runs.

    Parameters:

    Name Type Description Default session Session

    a database session

    required flow_filter FlowFilter

    only select task runs whose flows match these filters

    None flow_run_filter FlowRunFilter

    only select task runs whose flow runs match these filters

    None task_run_filter TaskRunFilter

    only select task runs that match these filters

    None deployment_filter DeploymentFilter

    only select task runs whose deployments match these filters

    None offset int

    Query offset

    None limit int

    Query limit

    None sort TaskRunSort

    Query sort

    ID_DESC

    Returns:

    Type Description

    List[db.TaskRun]: the task runs

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def read_task_runs(\n    session: sa.orm.Session,\n    db: PrefectDBInterface,\n    flow_filter: schemas.filters.FlowFilter = None,\n    flow_run_filter: schemas.filters.FlowRunFilter = None,\n    task_run_filter: schemas.filters.TaskRunFilter = None,\n    deployment_filter: schemas.filters.DeploymentFilter = None,\n    offset: int = None,\n    limit: int = None,\n    sort: schemas.sorting.TaskRunSort = schemas.sorting.TaskRunSort.ID_DESC,\n):\n    \"\"\"\n    Read task runs.\n\n    Args:\n        session: a database session\n        flow_filter: only select task runs whose flows match these filters\n        flow_run_filter: only select task runs whose flow runs match these filters\n        task_run_filter: only select task runs that match these filters\n        deployment_filter: only select task runs whose deployments match these filters\n        offset: Query offset\n        limit: Query limit\n        sort: Query sort\n\n    Returns:\n        List[db.TaskRun]: the task runs\n    \"\"\"\n\n    query = select(db.TaskRun).order_by(sort.as_sql_sort(db))\n\n    query = await _apply_task_run_filters(\n        query,\n        flow_filter=flow_filter,\n        flow_run_filter=flow_run_filter,\n        task_run_filter=task_run_filter,\n        deployment_filter=deployment_filter,\n        db=db,\n    )\n\n    if offset is not None:\n        query = query.offset(offset)\n\n    if limit is not None:\n        query = query.limit(limit)\n\n    logger.debug(f\"In read_task_runs, query generated is:\\n{query}\")\n    result = await session.execute(query)\n    return result.scalars().unique().all()\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.set_task_run_state","title":"set_task_run_state async","text":"

    Creates a new orchestrated task run state.

    Setting a new state on a run is the one of the principal actions that is governed by Prefect's orchestration logic. Setting a new run state will not guarantee creation, but instead trigger orchestration rules to govern the proposed state input. If the state is considered valid, it will be written to the database. Otherwise, a it's possible a different state, or no state, will be created. A force flag is supplied to bypass a subset of orchestration logic.

    Parameters:

    Name Type Description Default session Session

    a database session

    required task_run_id UUID

    the task run id

    required state State

    a task run state model

    required force bool

    if False, orchestration rules will be applied that may alter or prevent the state transition. If True, orchestration rules are not applied.

    False

    Returns:

    Type Description OrchestrationResult

    OrchestrationResult object

    Source code in prefect/server/models/task_runs.py
    async def set_task_run_state(\n    session: sa.orm.Session,\n    task_run_id: UUID,\n    state: schemas.states.State,\n    force: bool = False,\n    task_policy: BaseOrchestrationPolicy = None,\n    orchestration_parameters: dict = None,\n) -> OrchestrationResult:\n    \"\"\"\n    Creates a new orchestrated task run state.\n\n    Setting a new state on a run is the one of the principal actions that is governed by\n    Prefect's orchestration logic. Setting a new run state will not guarantee creation,\n    but instead trigger orchestration rules to govern the proposed `state` input. If\n    the state is considered valid, it will be written to the database. Otherwise, a\n    it's possible a different state, or no state, will be created. A `force` flag is\n    supplied to bypass a subset of orchestration logic.\n\n    Args:\n        session: a database session\n        task_run_id: the task run id\n        state: a task run state model\n        force: if False, orchestration rules will be applied that may alter or prevent\n            the state transition. If True, orchestration rules are not applied.\n\n    Returns:\n        OrchestrationResult object\n    \"\"\"\n\n    # load the task run\n    run = await models.task_runs.read_task_run(session=session, task_run_id=task_run_id)\n\n    if not run:\n        raise ObjectNotFoundError(f\"Task run with id {task_run_id} not found\")\n\n    initial_state = run.state.as_state() if run.state else None\n    initial_state_type = initial_state.type if initial_state else None\n    proposed_state_type = state.type if state else None\n    intended_transition = (initial_state_type, proposed_state_type)\n\n    if force or task_policy is None:\n        task_policy = MinimalTaskPolicy\n\n    orchestration_rules = task_policy.compile_transition_rules(*intended_transition)\n    global_rules = GlobalTaskPolicy.compile_transition_rules(*intended_transition)\n\n    context = TaskOrchestrationContext(\n        session=session,\n        run=run,\n        initial_state=initial_state,\n        proposed_state=state,\n    )\n\n    if orchestration_parameters is not None:\n        context.parameters = orchestration_parameters\n\n    # apply orchestration rules and create the new task run state\n    async with contextlib.AsyncExitStack() as stack:\n        for rule in orchestration_rules:\n            context = await stack.enter_async_context(\n                rule(context, *intended_transition)\n            )\n\n        for rule in global_rules:\n            context = await stack.enter_async_context(\n                rule(context, *intended_transition)\n            )\n\n        await context.validate_proposed_state()\n\n    if context.orchestration_error is not None:\n        raise context.orchestration_error\n\n    result = OrchestrationResult(\n        state=context.validated_state,\n        status=context.response_status,\n        details=context.response_details,\n    )\n\n    return result\n
    "},{"location":"api-ref/server/models/task_runs/#prefect.server.models.task_runs.update_task_run","title":"update_task_run async","text":"

    Updates a task run.

    Parameters:

    Name Type Description Default session AsyncSession

    a database session

    required task_run_id UUID

    the task run id to update

    required task_run TaskRunUpdate

    a task run model

    required

    Returns:

    Name Type Description bool bool

    whether or not matching rows were found to update

    Source code in prefect/server/models/task_runs.py
    @inject_db\nasync def update_task_run(\n    session: AsyncSession,\n    task_run_id: UUID,\n    task_run: schemas.actions.TaskRunUpdate,\n    db: PrefectDBInterface,\n) -> bool:\n    \"\"\"\n    Updates a task run.\n\n    Args:\n        session: a database session\n        task_run_id: the task run id to update\n        task_run: a task run model\n\n    Returns:\n        bool: whether or not matching rows were found to update\n    \"\"\"\n    update_stmt = (\n        sa.update(db.TaskRun).where(db.TaskRun.id == task_run_id)\n        # exclude_unset=True allows us to only update values provided by\n        # the user, ignoring any defaults on the model\n        .values(**task_run.dict(shallow=True, exclude_unset=True))\n    )\n    result = await session.execute(update_stmt)\n    return result.rowcount > 0\n
    "},{"location":"api-ref/server/orchestration/core_policy/","title":"server.orchestration.core_policy","text":""},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy","title":"prefect.server.orchestration.core_policy","text":"

    Orchestration logic that fires on state transitions.

    CoreFlowPolicy and CoreTaskPolicy contain all default orchestration rules that Prefect enforces on a state transition.

    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CoreFlowPolicy","title":"CoreFlowPolicy","text":"

    Bases: BaseOrchestrationPolicy

    Orchestration rules that run against flow-run-state transitions in priority order.

    Source code in prefect/server/orchestration/core_policy.py
    class CoreFlowPolicy(BaseOrchestrationPolicy):\n    \"\"\"\n    Orchestration rules that run against flow-run-state transitions in priority order.\n    \"\"\"\n\n    def priority():\n        return [\n            HandleFlowTerminalStateTransitions,\n            EnforceCancellingToCancelledTransition,\n            BypassCancellingScheduledFlowRuns,\n            PreventPendingTransitions,\n            HandlePausingFlows,\n            HandleResumingPausedFlows,\n            CopyScheduledTime,\n            WaitForScheduledTime,\n            RetryFailedFlows,\n        ]\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CoreTaskPolicy","title":"CoreTaskPolicy","text":"

    Bases: BaseOrchestrationPolicy

    Orchestration rules that run against task-run-state transitions in priority order.

    Source code in prefect/server/orchestration/core_policy.py
    class CoreTaskPolicy(BaseOrchestrationPolicy):\n    \"\"\"\n    Orchestration rules that run against task-run-state transitions in priority order.\n    \"\"\"\n\n    def priority():\n        return [\n            CacheRetrieval,\n            HandleTaskTerminalStateTransitions,\n            PreventRunningTasksFromStoppedFlows,\n            SecureTaskConcurrencySlots,  # retrieve cached states even if slots are full\n            CopyScheduledTime,\n            WaitForScheduledTime,\n            RetryFailedTasks,\n            RenameReruns,\n            UpdateFlowRunTrackerOnTasks,\n            CacheInsertion,\n            ReleaseTaskConcurrencySlots,\n        ]\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.SecureTaskConcurrencySlots","title":"SecureTaskConcurrencySlots","text":"

    Bases: BaseOrchestrationRule

    Checks relevant concurrency slots are available before entering a Running state.

    This rule checks if concurrency limits have been set on the tags associated with a TaskRun. If so, a concurrency slot will be secured against each concurrency limit before being allowed to transition into a running state. If a concurrency limit has been reached, the client will be instructed to delay the transition for the duration specified by the \"PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS\" setting before trying again. If the concurrency limit set on a tag is 0, the transition will be aborted to prevent deadlocks.

    Source code in prefect/server/orchestration/core_policy.py
    class SecureTaskConcurrencySlots(BaseOrchestrationRule):\n    \"\"\"\n    Checks relevant concurrency slots are available before entering a Running state.\n\n    This rule checks if concurrency limits have been set on the tags associated with a\n    TaskRun. If so, a concurrency slot will be secured against each concurrency limit\n    before being allowed to transition into a running state. If a concurrency limit has\n    been reached, the client will be instructed to delay the transition for the duration\n    specified by the \"PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS\" setting\n    before trying again. If the concurrency limit set on a tag is 0, the transition will\n    be aborted to prevent deadlocks.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        self._applied_limits = []\n        filtered_limits = (\n            await concurrency_limits.filter_concurrency_limits_for_orchestration(\n                context.session, tags=context.run.tags\n            )\n        )\n        run_limits = {limit.tag: limit for limit in filtered_limits}\n        for tag, cl in run_limits.items():\n            limit = cl.concurrency_limit\n            if limit == 0:\n                # limits of 0 will deadlock, and the transition needs to abort\n                for stale_tag in self._applied_limits:\n                    stale_limit = run_limits.get(stale_tag, None)\n                    active_slots = set(stale_limit.active_slots)\n                    active_slots.discard(str(context.run.id))\n                    stale_limit.active_slots = list(active_slots)\n\n                await self.abort_transition(\n                    reason=(\n                        f'The concurrency limit on tag \"{tag}\" is 0 and will deadlock'\n                        \" if the task tries to run again.\"\n                    ),\n                )\n            elif len(cl.active_slots) >= limit:\n                # if the limit has already been reached, delay the transition\n                for stale_tag in self._applied_limits:\n                    stale_limit = run_limits.get(stale_tag, None)\n                    active_slots = set(stale_limit.active_slots)\n                    active_slots.discard(str(context.run.id))\n                    stale_limit.active_slots = list(active_slots)\n\n                await self.delay_transition(\n                    PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS.value(),\n                    f\"Concurrency limit for the {tag} tag has been reached\",\n                )\n            else:\n                # log the TaskRun ID to active_slots\n                self._applied_limits.append(tag)\n                active_slots = set(cl.active_slots)\n                active_slots.add(str(context.run.id))\n                cl.active_slots = list(active_slots)\n\n    async def cleanup(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        for tag in self._applied_limits:\n            cl = await concurrency_limits.read_concurrency_limit_by_tag(\n                context.session, tag\n            )\n            active_slots = set(cl.active_slots)\n            active_slots.discard(str(context.run.id))\n            cl.active_slots = list(active_slots)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.ReleaseTaskConcurrencySlots","title":"ReleaseTaskConcurrencySlots","text":"

    Bases: BaseUniversalTransform

    Releases any concurrency slots held by a run upon exiting a Running or Cancelling state.

    Source code in prefect/server/orchestration/core_policy.py
    class ReleaseTaskConcurrencySlots(BaseUniversalTransform):\n    \"\"\"\n    Releases any concurrency slots held by a run upon exiting a Running or\n    Cancelling state.\n    \"\"\"\n\n    async def after_transition(\n        self,\n        context: OrchestrationContext,\n    ):\n        if self.nullified_transition():\n            return\n\n        if context.validated_state and context.validated_state.type not in [\n            states.StateType.RUNNING,\n            states.StateType.CANCELLING,\n        ]:\n            filtered_limits = (\n                await concurrency_limits.filter_concurrency_limits_for_orchestration(\n                    context.session, tags=context.run.tags\n                )\n            )\n            run_limits = {limit.tag: limit for limit in filtered_limits}\n            for tag, cl in run_limits.items():\n                active_slots = set(cl.active_slots)\n                active_slots.discard(str(context.run.id))\n                cl.active_slots = list(active_slots)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CacheInsertion","title":"CacheInsertion","text":"

    Bases: BaseOrchestrationRule

    Caches completed states with cache keys after they are validated.

    Source code in prefect/server/orchestration/core_policy.py
    class CacheInsertion(BaseOrchestrationRule):\n    \"\"\"\n    Caches completed states with cache keys after they are validated.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.COMPLETED]\n\n    @inject_db\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n        db: PrefectDBInterface,\n    ) -> None:\n        if not validated_state or not context.session:\n            return\n\n        cache_key = validated_state.state_details.cache_key\n        if cache_key:\n            new_cache_item = db.TaskRunStateCache(\n                cache_key=cache_key,\n                cache_expiration=validated_state.state_details.cache_expiration,\n                task_run_state_id=validated_state.id,\n            )\n            context.session.add(new_cache_item)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CacheRetrieval","title":"CacheRetrieval","text":"

    Bases: BaseOrchestrationRule

    Rejects running states if a completed state has been cached.

    This rule rejects transitions into a running state with a cache key if the key has already been associated with a completed state in the cache table. The client will be instructed to transition into the cached completed state instead.

    Source code in prefect/server/orchestration/core_policy.py
    class CacheRetrieval(BaseOrchestrationRule):\n    \"\"\"\n    Rejects running states if a completed state has been cached.\n\n    This rule rejects transitions into a running state with a cache key if the key\n    has already been associated with a completed state in the cache table. The client\n    will be instructed to transition into the cached completed state instead.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    @inject_db\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n        db: PrefectDBInterface,\n    ) -> None:\n        cache_key = proposed_state.state_details.cache_key\n        if cache_key and not proposed_state.state_details.refresh_cache:\n            # Check for cached states matching the cache key\n            cached_state_id = (\n                select(db.TaskRunStateCache.task_run_state_id)\n                .where(\n                    sa.and_(\n                        db.TaskRunStateCache.cache_key == cache_key,\n                        sa.or_(\n                            db.TaskRunStateCache.cache_expiration.is_(None),\n                            db.TaskRunStateCache.cache_expiration > pendulum.now(\"utc\"),\n                        ),\n                    ),\n                )\n                .order_by(db.TaskRunStateCache.created.desc())\n                .limit(1)\n            ).scalar_subquery()\n            query = select(db.TaskRunState).where(db.TaskRunState.id == cached_state_id)\n            cached_state = (await context.session.execute(query)).scalar()\n            if cached_state:\n                new_state = cached_state.as_state().copy(reset_fields=True)\n                new_state.name = \"Cached\"\n                await self.reject_transition(\n                    state=new_state, reason=\"Retrieved state from cache\"\n                )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.RetryFailedFlows","title":"RetryFailedFlows","text":"

    Bases: BaseOrchestrationRule

    Rejects failed states and schedules a retry if the retry limit has not been reached.

    This rule rejects transitions into a failed state if retries has been set and the run count has not reached the specified limit. The client will be instructed to transition into a scheduled state to retry flow execution.

    Source code in prefect/server/orchestration/core_policy.py
    class RetryFailedFlows(BaseOrchestrationRule):\n    \"\"\"\n    Rejects failed states and schedules a retry if the retry limit has not been reached.\n\n    This rule rejects transitions into a failed state if `retries` has been\n    set and the run count has not reached the specified limit. The client will be\n    instructed to transition into a scheduled state to retry flow execution.\n    \"\"\"\n\n    FROM_STATES = [StateType.RUNNING]\n    TO_STATES = [StateType.FAILED]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: FlowOrchestrationContext,\n    ) -> None:\n        run_settings = context.run_settings\n        run_count = context.run.run_count\n\n        if run_settings.retries is None or run_count > run_settings.retries:\n            return  # Retry count exceeded, allow transition to failed\n\n        scheduled_start_time = pendulum.now(\"UTC\").add(\n            seconds=run_settings.retry_delay or 0\n        )\n\n        # support old-style flow run retries for older clients\n        # older flow retries require us to loop over failed tasks to update their state\n        # this is not required after API version 0.8.3\n        api_version = context.parameters.get(\"api-version\", None)\n        if api_version and api_version < Version(\"0.8.3\"):\n            failed_task_runs = await models.task_runs.read_task_runs(\n                context.session,\n                flow_run_filter=filters.FlowRunFilter(id={\"any_\": [context.run.id]}),\n                task_run_filter=filters.TaskRunFilter(\n                    state={\"type\": {\"any_\": [\"FAILED\"]}}\n                ),\n            )\n            for run in failed_task_runs:\n                await models.task_runs.set_task_run_state(\n                    context.session,\n                    run.id,\n                    state=states.AwaitingRetry(scheduled_time=scheduled_start_time),\n                    force=True,\n                )\n                # Reset the run count so that the task run retries still work correctly\n                run.run_count = 0\n\n        # Reset pause metadata on retry\n        # Pauses as a concept only exist after API version 0.8.4\n        api_version = context.parameters.get(\"api-version\", None)\n        if api_version is None or api_version >= Version(\"0.8.4\"):\n            updated_policy = context.run.empirical_policy.dict()\n            updated_policy[\"resuming\"] = False\n            updated_policy[\"pause_keys\"] = set()\n            context.run.empirical_policy = core.FlowRunPolicy(**updated_policy)\n\n        # Generate a new state for the flow\n        retry_state = states.AwaitingRetry(\n            scheduled_time=scheduled_start_time,\n            message=proposed_state.message,\n            data=proposed_state.data,\n        )\n        await self.reject_transition(state=retry_state, reason=\"Retrying\")\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.RetryFailedTasks","title":"RetryFailedTasks","text":"

    Bases: BaseOrchestrationRule

    Rejects failed states and schedules a retry if the retry limit has not been reached.

    This rule rejects transitions into a failed state if retries has been set, the run count has not reached the specified limit, and the client asserts it is a retriable task run. The client will be instructed to transition into a scheduled state to retry task execution.

    Source code in prefect/server/orchestration/core_policy.py
    class RetryFailedTasks(BaseOrchestrationRule):\n    \"\"\"\n    Rejects failed states and schedules a retry if the retry limit has not been reached.\n\n    This rule rejects transitions into a failed state if `retries` has been\n    set, the run count has not reached the specified limit, and the client\n    asserts it is a retriable task run. The client will be instructed to\n    transition into a scheduled state to retry task execution.\n    \"\"\"\n\n    FROM_STATES = [StateType.RUNNING]\n    TO_STATES = [StateType.FAILED]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        run_settings = context.run_settings\n        run_count = context.run.run_count\n        delay = run_settings.retry_delay\n\n        if isinstance(delay, list):\n            base_delay = delay[min(run_count - 1, len(delay) - 1)]\n        else:\n            base_delay = run_settings.retry_delay or 0\n\n        # guard against negative relative jitter inputs\n        if run_settings.retry_jitter_factor:\n            delay = clamped_poisson_interval(\n                base_delay, clamping_factor=run_settings.retry_jitter_factor\n            )\n        else:\n            delay = base_delay\n\n        # set by user to conditionally retry a task using @task(retry_condition_fn=...)\n        if getattr(proposed_state.state_details, \"retriable\", True) is False:\n            return\n\n        if run_settings.retries is not None and run_count <= run_settings.retries:\n            retry_state = states.AwaitingRetry(\n                scheduled_time=pendulum.now(\"UTC\").add(seconds=delay),\n                message=proposed_state.message,\n                data=proposed_state.data,\n            )\n            await self.reject_transition(state=retry_state, reason=\"Retrying\")\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.RenameReruns","title":"RenameReruns","text":"

    Bases: BaseOrchestrationRule

    Name the states if they have run more than once.

    In the special case where the initial state is an \"AwaitingRetry\" scheduled state, the proposed state will be renamed to \"Retrying\" instead.

    Source code in prefect/server/orchestration/core_policy.py
    class RenameReruns(BaseOrchestrationRule):\n    \"\"\"\n    Name the states if they have run more than once.\n\n    In the special case where the initial state is an \"AwaitingRetry\" scheduled state,\n    the proposed state will be renamed to \"Retrying\" instead.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        run_count = context.run.run_count\n        if run_count > 0:\n            if initial_state.name == \"AwaitingRetry\":\n                await self.rename_state(\"Retrying\")\n            else:\n                await self.rename_state(\"Rerunning\")\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.CopyScheduledTime","title":"CopyScheduledTime","text":"

    Bases: BaseOrchestrationRule

    Ensures scheduled time is copied from scheduled states to pending states.

    If a new scheduled time has been proposed on the pending state, the scheduled time on the scheduled state will be ignored.

    Source code in prefect/server/orchestration/core_policy.py
    class CopyScheduledTime(BaseOrchestrationRule):\n    \"\"\"\n    Ensures scheduled time is copied from scheduled states to pending states.\n\n    If a new scheduled time has been proposed on the pending state, the scheduled time\n    on the scheduled state will be ignored.\n    \"\"\"\n\n    FROM_STATES = [StateType.SCHEDULED]\n    TO_STATES = [StateType.PENDING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        if not proposed_state.state_details.scheduled_time:\n            proposed_state.state_details.scheduled_time = (\n                initial_state.state_details.scheduled_time\n            )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.WaitForScheduledTime","title":"WaitForScheduledTime","text":"

    Bases: BaseOrchestrationRule

    Prevents transitions to running states from happening to early.

    This rule enforces that all scheduled states will only start with the machine clock used by the Prefect REST API instance. This rule will identify transitions from scheduled states that are too early and nullify them. Instead, no state will be written to the database and the client will be sent an instruction to wait for delay_seconds before attempting the transition again.

    Source code in prefect/server/orchestration/core_policy.py
    class WaitForScheduledTime(BaseOrchestrationRule):\n    \"\"\"\n    Prevents transitions to running states from happening to early.\n\n    This rule enforces that all scheduled states will only start with the machine clock\n    used by the Prefect REST API instance. This rule will identify transitions from scheduled\n    states that are too early and nullify them. Instead, no state will be written to the\n    database and the client will be sent an instruction to wait for `delay_seconds`\n    before attempting the transition again.\n    \"\"\"\n\n    FROM_STATES = [StateType.SCHEDULED, StateType.PENDING]\n    TO_STATES = [StateType.RUNNING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        scheduled_time = initial_state.state_details.scheduled_time\n        if not scheduled_time:\n            return\n\n        # At this moment, we round delay to the nearest second as the API schema\n        # specifies an integer return value.\n        delay = scheduled_time - pendulum.now(\"UTC\")\n        delay_seconds = delay.in_seconds()\n        delay_seconds += round(delay.microseconds / 1e6)\n        if delay_seconds > 0:\n            await self.delay_transition(\n                delay_seconds, reason=\"Scheduled time is in the future\"\n            )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.HandlePausingFlows","title":"HandlePausingFlows","text":"

    Bases: BaseOrchestrationRule

    Governs runs attempting to enter a Paused/Suspended state

    Source code in prefect/server/orchestration/core_policy.py
    class HandlePausingFlows(BaseOrchestrationRule):\n    \"\"\"\n    Governs runs attempting to enter a Paused/Suspended state\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.PAUSED]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        verb = \"suspend\" if proposed_state.name == \"Suspended\" else \"pause\"\n\n        if initial_state is None:\n            await self.abort_transition(f\"Cannot {verb} flows with no state.\")\n            return\n\n        if not initial_state.is_running():\n            await self.reject_transition(\n                state=None,\n                reason=f\"Cannot {verb} flows that are not currently running.\",\n            )\n            return\n\n        self.key = proposed_state.state_details.pause_key\n        if self.key is None:\n            # if no pause key is provided, default to a UUID\n            self.key = str(uuid4())\n\n        if self.key in context.run.empirical_policy.pause_keys:\n            await self.reject_transition(\n                state=None, reason=f\"This {verb} has already fired.\"\n            )\n            return\n\n        if proposed_state.state_details.pause_reschedule:\n            if context.run.parent_task_run_id:\n                await self.abort_transition(\n                    reason=f\"Cannot {verb} subflows.\",\n                )\n                return\n\n            if context.run.deployment_id is None:\n                await self.abort_transition(\n                    reason=f\"Cannot {verb} flows without a deployment.\",\n                )\n                return\n\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        updated_policy = context.run.empirical_policy.dict()\n        updated_policy[\"pause_keys\"].add(self.key)\n        context.run.empirical_policy = core.FlowRunPolicy(**updated_policy)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.HandleResumingPausedFlows","title":"HandleResumingPausedFlows","text":"

    Bases: BaseOrchestrationRule

    Governs runs attempting to leave a Paused state

    Source code in prefect/server/orchestration/core_policy.py
    class HandleResumingPausedFlows(BaseOrchestrationRule):\n    \"\"\"\n    Governs runs attempting to leave a Paused state\n    \"\"\"\n\n    FROM_STATES = [StateType.PAUSED]\n    TO_STATES = ALL_ORCHESTRATION_STATES\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        if not (\n            proposed_state.is_running()\n            or proposed_state.is_scheduled()\n            or proposed_state.is_final()\n        ):\n            await self.reject_transition(\n                state=None,\n                reason=(\n                    f\"This run cannot transition to the {proposed_state.type} state\"\n                    f\" from the {initial_state.type} state.\"\n                ),\n            )\n            return\n\n        verb = \"suspend\" if proposed_state.name == \"Suspended\" else \"pause\"\n\n        if initial_state.state_details.pause_reschedule:\n            if not context.run.deployment_id:\n                await self.reject_transition(\n                    state=None,\n                    reason=(\n                        f\"Cannot reschedule a {proposed_state.name.lower()} flow run\"\n                        \" without a deployment.\"\n                    ),\n                )\n                return\n        pause_timeout = initial_state.state_details.pause_timeout\n        if pause_timeout and pause_timeout < pendulum.now(\"UTC\"):\n            pause_timeout_failure = states.Failed(\n                message=(\n                    f\"The flow was {proposed_state.name.lower()} and never resumed.\"\n                ),\n            )\n            await self.reject_transition(\n                state=pause_timeout_failure,\n                reason=f\"The flow run {verb} has timed out and can no longer resume.\",\n            )\n            return\n\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        updated_policy = context.run.empirical_policy.dict()\n        updated_policy[\"resuming\"] = True\n        context.run.empirical_policy = core.FlowRunPolicy(**updated_policy)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.UpdateFlowRunTrackerOnTasks","title":"UpdateFlowRunTrackerOnTasks","text":"

    Bases: BaseOrchestrationRule

    Tracks the flow run attempt a task run state is associated with.

    Source code in prefect/server/orchestration/core_policy.py
    class UpdateFlowRunTrackerOnTasks(BaseOrchestrationRule):\n    \"\"\"\n    Tracks the flow run attempt a task run state is associated with.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        if context.run.flow_run_id is not None:\n            self.flow_run = await context.flow_run()\n            if self.flow_run:\n                context.run.flow_run_run_count = self.flow_run.run_count\n            else:\n                raise ObjectNotFoundError(\n                    (\n                        \"Unable to read flow run associated with task run:\"\n                        f\" {context.run.id}, this flow run might have been deleted\"\n                    ),\n                )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.HandleTaskTerminalStateTransitions","title":"HandleTaskTerminalStateTransitions","text":"

    Bases: BaseOrchestrationRule

    We do not allow tasks to leave terminal states if: - The task is completed and has a persisted result - The task is going to CANCELLING / PAUSED / CRASHED

    We reset the run count when a task leaves a terminal state for a non-terminal state which resets task run retries; this is particularly relevant for flow run retries.

    Source code in prefect/server/orchestration/core_policy.py
    class HandleTaskTerminalStateTransitions(BaseOrchestrationRule):\n    \"\"\"\n    We do not allow tasks to leave terminal states if:\n    - The task is completed and has a persisted result\n    - The task is going to CANCELLING / PAUSED / CRASHED\n\n    We reset the run count when a task leaves a terminal state for a non-terminal state\n    which resets task run retries; this is particularly relevant for flow run retries.\n    \"\"\"\n\n    FROM_STATES = TERMINAL_STATES\n    TO_STATES = ALL_ORCHESTRATION_STATES\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        self.original_run_count = context.run.run_count\n\n        # Do not allow runs to be marked as crashed, paused, or cancelling if already terminal\n        if proposed_state.type in {\n            StateType.CANCELLING,\n            StateType.PAUSED,\n            StateType.CRASHED,\n        }:\n            await self.abort_transition(f\"Run is already {initial_state.type.value}.\")\n            return\n\n        # Only allow departure from a happily completed state if the result is not persisted\n        if (\n            initial_state.is_completed()\n            and initial_state.data\n            and initial_state.data.get(\"type\") != \"unpersisted\"\n        ):\n            await self.reject_transition(None, \"This run is already completed.\")\n            return\n\n        if not proposed_state.is_final():\n            # Reset run count to reset retries\n            context.run.run_count = 0\n\n        # Change the name of the state to retrying if its a flow run retry\n        if proposed_state.is_running() and context.run.flow_run_id is not None:\n            self.flow_run = await context.flow_run()\n            flow_retrying = context.run.flow_run_run_count < self.flow_run.run_count\n            if flow_retrying:\n                await self.rename_state(\"Retrying\")\n\n    async def cleanup(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ):\n        # reset run count\n        context.run.run_count = self.original_run_count\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.HandleFlowTerminalStateTransitions","title":"HandleFlowTerminalStateTransitions","text":"

    Bases: BaseOrchestrationRule

    We do not allow flows to leave terminal states if: - The flow is completed and has a persisted result - The flow is going to CANCELLING / PAUSED / CRASHED - The flow is going to scheduled and has no deployment

    We reset the pause metadata when a flow leaves a terminal state for a non-terminal state. This resets pause behavior during manual flow run retries.

    Source code in prefect/server/orchestration/core_policy.py
    class HandleFlowTerminalStateTransitions(BaseOrchestrationRule):\n    \"\"\"\n    We do not allow flows to leave terminal states if:\n    - The flow is completed and has a persisted result\n    - The flow is going to CANCELLING / PAUSED / CRASHED\n    - The flow is going to scheduled and has no deployment\n\n    We reset the pause metadata when a flow leaves a terminal state for a non-terminal\n    state. This resets pause behavior during manual flow run retries.\n    \"\"\"\n\n    FROM_STATES = TERMINAL_STATES\n    TO_STATES = ALL_ORCHESTRATION_STATES\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: FlowOrchestrationContext,\n    ) -> None:\n        self.original_flow_policy = context.run.empirical_policy.dict()\n\n        # Do not allow runs to be marked as crashed, paused, or cancelling if already terminal\n        if proposed_state.type in {\n            StateType.CANCELLING,\n            StateType.PAUSED,\n            StateType.CRASHED,\n        }:\n            await self.abort_transition(\n                f\"Run is already in terminal state {initial_state.type.value}.\"\n            )\n            return\n\n        # Only allow departure from a happily completed state if the result is not\n        # persisted and the a rerun is being proposed\n        if (\n            initial_state.is_completed()\n            and not proposed_state.is_final()\n            and initial_state.data\n            and initial_state.data.get(\"type\") != \"unpersisted\"\n        ):\n            await self.reject_transition(None, \"Run is already COMPLETED.\")\n            return\n\n        # Do not allows runs to be rescheduled without a deployment\n        if proposed_state.is_scheduled() and not context.run.deployment_id:\n            await self.abort_transition(\n                \"Cannot reschedule a run without an associated deployment.\"\n            )\n            return\n\n        if not proposed_state.is_final():\n            # Reset pause metadata when leaving a terminal state\n            api_version = context.parameters.get(\"api-version\", None)\n            if api_version is None or api_version >= Version(\"0.8.4\"):\n                updated_policy = context.run.empirical_policy.dict()\n                updated_policy[\"resuming\"] = False\n                updated_policy[\"pause_keys\"] = set()\n                context.run.empirical_policy = core.FlowRunPolicy(**updated_policy)\n\n    async def cleanup(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ):\n        context.run.empirical_policy = core.FlowRunPolicy(**self.original_flow_policy)\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.PreventPendingTransitions","title":"PreventPendingTransitions","text":"

    Bases: BaseOrchestrationRule

    Prevents transitions to PENDING.

    This rule is only used for flow runs.

    This is intended to prevent race conditions during duplicate submissions of runs. Before a run is submitted to its execution environment, it should be placed in a PENDING state. If two workers attempt to submit the same run, one of them should encounter a PENDING -> PENDING transition and abort orchestration of the run.

    Similarly, if the execution environment starts quickly the run may be in a RUNNING state when the second worker attempts the PENDING transition. We deny these state changes as well to prevent duplicate submission. If a run has transitioned to a RUNNING state a worker should not attempt to submit it again unless it has moved into a terminal state.

    CANCELLING and CANCELLED runs should not be allowed to transition to PENDING. For re-runs of deployed runs, they should transition to SCHEDULED first. For re-runs of ad-hoc runs, they should transition directly to RUNNING.

    Source code in prefect/server/orchestration/core_policy.py
    class PreventPendingTransitions(BaseOrchestrationRule):\n    \"\"\"\n    Prevents transitions to PENDING.\n\n    This rule is only used for flow runs.\n\n    This is intended to prevent race conditions during duplicate submissions of runs.\n    Before a run is submitted to its execution environment, it should be placed in a\n    PENDING state. If two workers attempt to submit the same run, one of them should\n    encounter a PENDING -> PENDING transition and abort orchestration of the run.\n\n    Similarly, if the execution environment starts quickly the run may be in a RUNNING\n    state when the second worker attempts the PENDING transition. We deny these state\n    changes as well to prevent duplicate submission. If a run has transitioned to a\n    RUNNING state a worker should not attempt to submit it again unless it has moved\n    into a terminal state.\n\n    CANCELLING and CANCELLED runs should not be allowed to transition to PENDING.\n    For re-runs of deployed runs, they should transition to SCHEDULED first.\n    For re-runs of ad-hoc runs, they should transition directly to RUNNING.\n    \"\"\"\n\n    FROM_STATES = [\n        StateType.PENDING,\n        StateType.CANCELLING,\n        StateType.RUNNING,\n        StateType.CANCELLED,\n    ]\n    TO_STATES = [StateType.PENDING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        await self.abort_transition(\n            reason=(\n                f\"This run is in a {initial_state.type.name} state and cannot\"\n                \" transition to a PENDING state.\"\n            )\n        )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.PreventRunningTasksFromStoppedFlows","title":"PreventRunningTasksFromStoppedFlows","text":"

    Bases: BaseOrchestrationRule

    Prevents running tasks from stopped flows.

    A running state implies execution, but also the converse. This rule ensures that a flow's tasks cannot be run unless the flow is also running.

    Source code in prefect/server/orchestration/core_policy.py
    class PreventRunningTasksFromStoppedFlows(BaseOrchestrationRule):\n    \"\"\"\n    Prevents running tasks from stopped flows.\n\n    A running state implies execution, but also the converse. This rule ensures that a\n    flow's tasks cannot be run unless the flow is also running.\n    \"\"\"\n\n    FROM_STATES = ALL_ORCHESTRATION_STATES\n    TO_STATES = [StateType.RUNNING]\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        flow_run = await context.flow_run()\n        if flow_run is not None:\n            if flow_run.state is None:\n                await self.abort_transition(\n                    reason=\"The enclosing flow must be running to begin task execution.\"\n                )\n            elif flow_run.state.type == StateType.PAUSED:\n                # Use the flow run's Paused state details to preserve data like\n                # timeouts.\n                paused_state = states.Paused(\n                    name=\"NotReady\",\n                    pause_expiration_time=flow_run.state.state_details.pause_timeout,\n                    reschedule=flow_run.state.state_details.pause_reschedule,\n                )\n                await self.reject_transition(\n                    state=paused_state,\n                    reason=(\n                        \"The flow is paused, new tasks can execute after resuming flow\"\n                        f\" run: {flow_run.id}.\"\n                    ),\n                )\n            elif not flow_run.state.type == StateType.RUNNING:\n                # task runners should abort task run execution\n                await self.abort_transition(\n                    reason=(\n                        \"The enclosing flow must be running to begin task execution.\"\n                    ),\n                )\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.EnforceCancellingToCancelledTransition","title":"EnforceCancellingToCancelledTransition","text":"

    Bases: BaseOrchestrationRule

    Rejects transitions from Cancelling to any terminal state except for Cancelled.

    Source code in prefect/server/orchestration/core_policy.py
    class EnforceCancellingToCancelledTransition(BaseOrchestrationRule):\n    \"\"\"\n    Rejects transitions from Cancelling to any terminal state except for Cancelled.\n    \"\"\"\n\n    FROM_STATES = {StateType.CANCELLED, StateType.CANCELLING}\n    TO_STATES = ALL_ORCHESTRATION_STATES - {StateType.CANCELLED}\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: TaskOrchestrationContext,\n    ) -> None:\n        await self.reject_transition(\n            state=None,\n            reason=(\n                \"Cannot transition flows that are cancelling to a state other \"\n                \"than Cancelled.\"\n            ),\n        )\n        return\n
    "},{"location":"api-ref/server/orchestration/core_policy/#prefect.server.orchestration.core_policy.BypassCancellingScheduledFlowRuns","title":"BypassCancellingScheduledFlowRuns","text":"

    Bases: BaseOrchestrationRule

    Rejects transitions from Scheduled to Cancelling, and instead sets the state to Cancelled, if the flow run has no associated infrastructure process ID.

    The Cancelling state is used to clean up infrastructure. If there is not infrastructure to clean up, we can transition directly to Cancelled. Runs that are AwaitingRetry are a Scheduled state that may have associated infrastructure.

    Source code in prefect/server/orchestration/core_policy.py
    class BypassCancellingScheduledFlowRuns(BaseOrchestrationRule):\n    \"\"\"Rejects transitions from Scheduled to Cancelling, and instead sets the state to Cancelled,\n    if the flow run has no associated infrastructure process ID.\n\n    The `Cancelling` state is used to clean up infrastructure. If there is not infrastructure\n    to clean up, we can transition directly to `Cancelled`. Runs that are `AwaitingRetry` are\n    a `Scheduled` state that may have associated infrastructure.\n    \"\"\"\n\n    FROM_STATES = {StateType.SCHEDULED}\n    TO_STATES = {StateType.CANCELLING}\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: FlowOrchestrationContext,\n    ) -> None:\n        if not context.run.infrastructure_pid:\n            await self.reject_transition(\n                state=states.Cancelled(),\n                reason=\"Scheduled flow run has no infrastructure to terminate.\",\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/","title":"server.orchestration.global_policy","text":""},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy","title":"prefect.server.orchestration.global_policy","text":"

    Bookkeeping logic that fires on every state transition.

    For clarity, GlobalFlowpolicy and GlobalTaskPolicy contain all transition logic implemented using BaseUniversalTransform. None of these operations modify state, and regardless of what orchestration Prefect REST API might enforce on a transition, the global policies contain Prefect's necessary bookkeeping. Because these transforms record information about the validated state committed to the state database, they should be the most deeply nested contexts in orchestration loop.

    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.GlobalFlowPolicy","title":"GlobalFlowPolicy","text":"

    Bases: BaseOrchestrationPolicy

    Global transforms that run against flow-run-state transitions in priority order.

    These transforms are intended to run immediately before and after a state transition is validated.

    Source code in prefect/server/orchestration/global_policy.py
    class GlobalFlowPolicy(BaseOrchestrationPolicy):\n    \"\"\"\n    Global transforms that run against flow-run-state transitions in priority order.\n\n    These transforms are intended to run immediately before and after a state transition\n    is validated.\n    \"\"\"\n\n    def priority():\n        return COMMON_GLOBAL_TRANSFORMS() + [\n            UpdateSubflowParentTask,\n            UpdateSubflowStateDetails,\n            IncrementFlowRunCount,\n            RemoveResumingIndicator,\n        ]\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.GlobalTaskPolicy","title":"GlobalTaskPolicy","text":"

    Bases: BaseOrchestrationPolicy

    Global transforms that run against task-run-state transitions in priority order.

    These transforms are intended to run immediately before and after a state transition is validated.

    Source code in prefect/server/orchestration/global_policy.py
    class GlobalTaskPolicy(BaseOrchestrationPolicy):\n    \"\"\"\n    Global transforms that run against task-run-state transitions in priority order.\n\n    These transforms are intended to run immediately before and after a state transition\n    is validated.\n    \"\"\"\n\n    def priority():\n        return COMMON_GLOBAL_TRANSFORMS() + [\n            IncrementTaskRunCount,\n        ]\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetRunStateType","title":"SetRunStateType","text":"

    Bases: BaseUniversalTransform

    Updates the state type of a run on a state transition.

    Source code in prefect/server/orchestration/global_policy.py
    class SetRunStateType(BaseUniversalTransform):\n    \"\"\"\n    Updates the state type of a run on a state transition.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # record the new state's type\n        context.run.state_type = context.proposed_state.type\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetRunStateName","title":"SetRunStateName","text":"

    Bases: BaseUniversalTransform

    Updates the state name of a run on a state transition.

    Source code in prefect/server/orchestration/global_policy.py
    class SetRunStateName(BaseUniversalTransform):\n    \"\"\"\n    Updates the state name of a run on a state transition.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # record the new state's name\n        context.run.state_name = context.proposed_state.name\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetStartTime","title":"SetStartTime","text":"

    Bases: BaseUniversalTransform

    Records the time a run enters a running state for the first time.

    Source code in prefect/server/orchestration/global_policy.py
    class SetStartTime(BaseUniversalTransform):\n    \"\"\"\n    Records the time a run enters a running state for the first time.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if entering a running state and no start time is set...\n        if context.proposed_state.is_running() and context.run.start_time is None:\n            # set the start time\n            context.run.start_time = context.proposed_state.timestamp\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetRunStateTimestamp","title":"SetRunStateTimestamp","text":"

    Bases: BaseUniversalTransform

    Records the time a run changes states.

    Source code in prefect/server/orchestration/global_policy.py
    class SetRunStateTimestamp(BaseUniversalTransform):\n    \"\"\"\n    Records the time a run changes states.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # record the new state's timestamp\n        context.run.state_timestamp = context.proposed_state.timestamp\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetEndTime","title":"SetEndTime","text":"

    Bases: BaseUniversalTransform

    Records the time a run enters a terminal state.

    With normal client usage, a run will not transition out of a terminal state. However, it's possible to force these transitions manually via the API. While leaving a terminal state, the end time will be unset.

    Source code in prefect/server/orchestration/global_policy.py
    class SetEndTime(BaseUniversalTransform):\n    \"\"\"\n    Records the time a run enters a terminal state.\n\n    With normal client usage, a run will not transition out of a terminal state.\n    However, it's possible to force these transitions manually via the API. While\n    leaving a terminal state, the end time will be unset.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if exiting a final state for a non-final state...\n        if (\n            context.initial_state\n            and context.initial_state.is_final()\n            and not context.proposed_state.is_final()\n        ):\n            # clear the end time\n            context.run.end_time = None\n\n        # if entering a final state...\n        if context.proposed_state.is_final():\n            # if the run has a start time and no end time, give it one\n            if context.run.start_time and not context.run.end_time:\n                context.run.end_time = context.proposed_state.timestamp\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.IncrementRunTime","title":"IncrementRunTime","text":"

    Bases: BaseUniversalTransform

    Records the amount of time a run spends in the running state.

    Source code in prefect/server/orchestration/global_policy.py
    class IncrementRunTime(BaseUniversalTransform):\n    \"\"\"\n    Records the amount of time a run spends in the running state.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if exiting a running state...\n        if context.initial_state and context.initial_state.is_running():\n            # increment the run time by the time spent in the previous state\n            context.run.total_run_time += (\n                context.proposed_state.timestamp - context.initial_state.timestamp\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.IncrementFlowRunCount","title":"IncrementFlowRunCount","text":"

    Bases: BaseUniversalTransform

    Records the number of times a run enters a running state. For use with retries.

    Source code in prefect/server/orchestration/global_policy.py
    class IncrementFlowRunCount(BaseUniversalTransform):\n    \"\"\"\n    Records the number of times a run enters a running state. For use with retries.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if entering a running state...\n        if context.proposed_state.is_running():\n            # do not increment the run count if resuming a paused flow\n            api_version = context.parameters.get(\"api-version\", None)\n            if api_version is None or api_version >= Version(\"0.8.4\"):\n                if context.run.empirical_policy.resuming:\n                    return\n\n            # increment the run count\n            context.run.run_count += 1\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.RemoveResumingIndicator","title":"RemoveResumingIndicator","text":"

    Bases: BaseUniversalTransform

    Removes the indicator on a flow run that marks it as resuming.

    Source code in prefect/server/orchestration/global_policy.py
    class RemoveResumingIndicator(BaseUniversalTransform):\n    \"\"\"\n    Removes the indicator on a flow run that marks it as resuming.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        proposed_state = context.proposed_state\n\n        api_version = context.parameters.get(\"api-version\", None)\n        if api_version is None or api_version >= Version(\"0.8.4\"):\n            if proposed_state.is_running() or proposed_state.is_final():\n                if context.run.empirical_policy.resuming:\n                    updated_policy = context.run.empirical_policy.dict()\n                    updated_policy[\"resuming\"] = False\n                    context.run.empirical_policy = FlowRunPolicy(**updated_policy)\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.IncrementTaskRunCount","title":"IncrementTaskRunCount","text":"

    Bases: BaseUniversalTransform

    Records the number of times a run enters a running state. For use with retries.

    Source code in prefect/server/orchestration/global_policy.py
    class IncrementTaskRunCount(BaseUniversalTransform):\n    \"\"\"\n    Records the number of times a run enters a running state. For use with retries.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # if entering a running state...\n        if context.proposed_state.is_running():\n            # increment the run count\n            context.run.run_count += 1\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetExpectedStartTime","title":"SetExpectedStartTime","text":"

    Bases: BaseUniversalTransform

    Estimates the time a state is expected to start running if not set.

    For scheduled states, this estimate is simply the scheduled time. For other states, this is set to the time the proposed state was created by Prefect.

    Source code in prefect/server/orchestration/global_policy.py
    class SetExpectedStartTime(BaseUniversalTransform):\n    \"\"\"\n    Estimates the time a state is expected to start running if not set.\n\n    For scheduled states, this estimate is simply the scheduled time. For other states,\n    this is set to the time the proposed state was created by Prefect.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # set expected start time if this is the first state\n        if not context.run.expected_start_time:\n            if context.proposed_state.is_scheduled():\n                context.run.expected_start_time = (\n                    context.proposed_state.state_details.scheduled_time\n                )\n            else:\n                context.run.expected_start_time = context.proposed_state.timestamp\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.SetNextScheduledStartTime","title":"SetNextScheduledStartTime","text":"

    Bases: BaseUniversalTransform

    Records the scheduled time on a run.

    When a run enters a scheduled state, run.next_scheduled_start_time is set to the state's scheduled time. When leaving a scheduled state, run.next_scheduled_start_time is unset.

    Source code in prefect/server/orchestration/global_policy.py
    class SetNextScheduledStartTime(BaseUniversalTransform):\n    \"\"\"\n    Records the scheduled time on a run.\n\n    When a run enters a scheduled state, `run.next_scheduled_start_time` is set to\n    the state's scheduled time. When leaving a scheduled state,\n    `run.next_scheduled_start_time` is unset.\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # remove the next scheduled start time if exiting a scheduled state\n        if context.initial_state and context.initial_state.is_scheduled():\n            context.run.next_scheduled_start_time = None\n\n        # set next scheduled start time if entering a scheduled state\n        if context.proposed_state.is_scheduled():\n            context.run.next_scheduled_start_time = (\n                context.proposed_state.state_details.scheduled_time\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.UpdateSubflowParentTask","title":"UpdateSubflowParentTask","text":"

    Bases: BaseUniversalTransform

    Whenever a subflow changes state, it must update its parent task run's state.

    Source code in prefect/server/orchestration/global_policy.py
    class UpdateSubflowParentTask(BaseUniversalTransform):\n    \"\"\"\n    Whenever a subflow changes state, it must update its parent task run's state.\n    \"\"\"\n\n    async def after_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # only applies to flow runs with a parent task run id\n        if context.run.parent_task_run_id is not None:\n            # avoid mutation of the flow run state\n            subflow_parent_task_state = context.validated_state.copy(\n                reset_fields=True,\n                include={\n                    \"type\",\n                    \"timestamp\",\n                    \"name\",\n                    \"message\",\n                    \"state_details\",\n                    \"data\",\n                },\n            )\n\n            # set the task's \"child flow run id\" to be the subflow run id\n            subflow_parent_task_state.state_details.child_flow_run_id = context.run.id\n\n            await models.task_runs.set_task_run_state(\n                session=context.session,\n                task_run_id=context.run.parent_task_run_id,\n                state=subflow_parent_task_state,\n                force=True,\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.UpdateSubflowStateDetails","title":"UpdateSubflowStateDetails","text":"

    Bases: BaseUniversalTransform

    Update a child subflow state's references to a corresponding tracking task run id in the parent flow run

    Source code in prefect/server/orchestration/global_policy.py
    class UpdateSubflowStateDetails(BaseUniversalTransform):\n    \"\"\"\n    Update a child subflow state's references to a corresponding tracking task run id\n    in the parent flow run\n    \"\"\"\n\n    async def before_transition(self, context: OrchestrationContext) -> None:\n        if self.nullified_transition():\n            return\n\n        # only applies to flow runs with a parent task run id\n        if context.run.parent_task_run_id is not None:\n            context.proposed_state.state_details.task_run_id = (\n                context.run.parent_task_run_id\n            )\n
    "},{"location":"api-ref/server/orchestration/global_policy/#prefect.server.orchestration.global_policy.UpdateStateDetails","title":"UpdateStateDetails","text":"

    Bases: BaseUniversalTransform

    Update a state's references to a corresponding flow- or task- run.

    Source code in prefect/server/orchestration/global_policy.py
    class UpdateStateDetails(BaseUniversalTransform):\n    \"\"\"\n    Update a state's references to a corresponding flow- or task- run.\n    \"\"\"\n\n    async def before_transition(\n        self,\n        context: OrchestrationContext,\n    ) -> None:\n        if self.nullified_transition():\n            return\n\n        if isinstance(context, FlowOrchestrationContext):\n            flow_run = await context.flow_run()\n            context.proposed_state.state_details.flow_run_id = flow_run.id\n\n        elif isinstance(context, TaskOrchestrationContext):\n            task_run = await context.task_run()\n            context.proposed_state.state_details.flow_run_id = task_run.flow_run_id\n            context.proposed_state.state_details.task_run_id = task_run.id\n
    "},{"location":"api-ref/server/orchestration/policies/","title":"server.orchestration.policies","text":""},{"location":"api-ref/server/orchestration/policies/#prefect.server.orchestration.policies","title":"prefect.server.orchestration.policies","text":"

    Policies are collections of orchestration rules and transforms.

    Prefect implements (most) orchestration with logic that governs a Prefect flow or task changing state. Policies organize of orchestration logic both to provide an ordering mechanism as well as provide observability into the orchestration process.

    While Prefect's orchestration rules can gracefully run independently of one another, ordering can still have an impact on the observed behavior of the system. For example, it makes no sense to secure a concurrency slot for a run if a cached state exists. Furthermore, policies, provide a mechanism to configure and observe exactly what logic will fire against a transition.

    "},{"location":"api-ref/server/orchestration/policies/#prefect.server.orchestration.policies.BaseOrchestrationPolicy","title":"BaseOrchestrationPolicy","text":"

    Bases: ABC

    An abstract base class used to organize orchestration rules in priority order.

    Different collections of orchestration rules might be used to govern various kinds of transitions. For example, flow-run states and task-run states might require different orchestration logic.

    Source code in prefect/server/orchestration/policies.py
    class BaseOrchestrationPolicy(ABC):\n    \"\"\"\n    An abstract base class used to organize orchestration rules in priority order.\n\n    Different collections of orchestration rules might be used to govern various kinds\n    of transitions. For example, flow-run states and task-run states might require\n    different orchestration logic.\n    \"\"\"\n\n    @staticmethod\n    @abstractmethod\n    def priority():\n        \"\"\"\n        A list of orchestration rules in priority order.\n        \"\"\"\n\n        return []\n\n    @classmethod\n    def compile_transition_rules(cls, from_state=None, to_state=None):\n        \"\"\"\n        Returns rules in policy that are valid for the specified state transition.\n        \"\"\"\n\n        transition_rules = []\n        for rule in cls.priority():\n            if from_state in rule.FROM_STATES and to_state in rule.TO_STATES:\n                transition_rules.append(rule)\n        return transition_rules\n
    "},{"location":"api-ref/server/orchestration/policies/#prefect.server.orchestration.policies.BaseOrchestrationPolicy.priority","title":"priority abstractmethod staticmethod","text":"

    A list of orchestration rules in priority order.

    Source code in prefect/server/orchestration/policies.py
    @staticmethod\n@abstractmethod\ndef priority():\n    \"\"\"\n    A list of orchestration rules in priority order.\n    \"\"\"\n\n    return []\n
    "},{"location":"api-ref/server/orchestration/policies/#prefect.server.orchestration.policies.BaseOrchestrationPolicy.compile_transition_rules","title":"compile_transition_rules classmethod","text":"

    Returns rules in policy that are valid for the specified state transition.

    Source code in prefect/server/orchestration/policies.py
    @classmethod\ndef compile_transition_rules(cls, from_state=None, to_state=None):\n    \"\"\"\n    Returns rules in policy that are valid for the specified state transition.\n    \"\"\"\n\n    transition_rules = []\n    for rule in cls.priority():\n        if from_state in rule.FROM_STATES and to_state in rule.TO_STATES:\n            transition_rules.append(rule)\n    return transition_rules\n
    "},{"location":"api-ref/server/orchestration/rules/","title":"server.orchestration.rules","text":""},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules","title":"prefect.server.orchestration.rules","text":"

    Prefect's flow and task-run orchestration machinery.

    This module contains all the core concepts necessary to implement Prefect's state orchestration engine. These states correspond to intuitive descriptions of all the points that a Prefect flow or task can observe executing user code and intervene, if necessary. A detailed description of states can be found in our concept documentation.

    Prefect's orchestration engine operates under the assumption that no governed user code will execute without first requesting Prefect REST API validate a change in state and record metadata about the run. With all attempts to run user code being checked against a Prefect instance, the Prefect REST API database becomes the unambiguous source of truth for managing the execution of complex interacting workflows. Orchestration rules can be implemented as discrete units of logic that operate against each state transition and can be fully observable, extensible, and customizable -- all without needing to store or parse a single line of user code.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext","title":"OrchestrationContext","text":"

    Bases: PrefectBaseModel

    A container for a state transition, governed by orchestration rules.

    Note

    An OrchestrationContext should not be instantiated directly, instead use the flow- or task- specific subclasses, FlowOrchestrationContext and TaskOrchestrationContext.

    When a flow- or task- run attempts to change state, Prefect REST API has an opportunity to decide whether this transition can proceed. All the relevant information associated with the state transition is stored in an OrchestrationContext, which is subsequently governed by nested orchestration rules implemented using the BaseOrchestrationRule ABC.

    OrchestrationContext introduces the concept of a state being None in the context of an intended state transition. An initial state can be None if a run is is attempting to set a state for the first time. The proposed state might be None if a rule governing the transition determines that no state change should occur at all and nothing is written to the database.

    Attributes:

    Name Type Description session Optional[Union[Session, AsyncSession]]

    a SQLAlchemy database session

    initial_state Optional[State]

    the initial state of a run

    proposed_state Optional[State]

    the proposed state a run is transitioning into

    validated_state Optional[State]

    a proposed state that has committed to the database

    rule_signature List[str]

    a record of rules that have fired on entry into a managed context, currently only used for debugging purposes

    finalization_signature List[str]

    a record of rules that have fired on exit from a managed context, currently only used for debugging purposes

    response_status SetStateStatus

    a SetStateStatus object used to build the API response

    response_details StateResponseDetails

    a StateResponseDetails object use to build the API response

    Parameters:

    Name Type Description Default session

    a SQLAlchemy database session

    required initial_state

    the initial state of a run

    required proposed_state

    the proposed state a run is transitioning into

    required Source code in prefect/server/orchestration/rules.py
    class OrchestrationContext(PrefectBaseModel):\n    \"\"\"\n    A container for a state transition, governed by orchestration rules.\n\n    Note:\n        An `OrchestrationContext` should not be instantiated directly, instead\n        use the flow- or task- specific subclasses, `FlowOrchestrationContext` and\n        `TaskOrchestrationContext`.\n\n    When a flow- or task- run attempts to change state, Prefect REST API has an opportunity\n    to decide whether this transition can proceed. All the relevant information\n    associated with the state transition is stored in an `OrchestrationContext`,\n    which is subsequently governed by nested orchestration rules implemented using\n    the `BaseOrchestrationRule` ABC.\n\n    `OrchestrationContext` introduces the concept of a state being `None` in the\n    context of an intended state transition. An initial state can be `None` if a run\n    is is attempting to set a state for the first time. The proposed state might be\n    `None` if a rule governing the transition determines that no state change\n    should occur at all and nothing is written to the database.\n\n    Attributes:\n        session: a SQLAlchemy database session\n        initial_state: the initial state of a run\n        proposed_state: the proposed state a run is transitioning into\n        validated_state: a proposed state that has committed to the database\n        rule_signature: a record of rules that have fired on entry into a\n            managed context, currently only used for debugging purposes\n        finalization_signature: a record of rules that have fired on exit from a\n            managed context, currently only used for debugging purposes\n        response_status: a SetStateStatus object used to build the API response\n        response_details:a StateResponseDetails object use to build the API response\n\n    Args:\n        session: a SQLAlchemy database session\n        initial_state: the initial state of a run\n        proposed_state: the proposed state a run is transitioning into\n    \"\"\"\n\n    class Config:\n        arbitrary_types_allowed = True\n\n    session: Optional[Union[sa.orm.Session, AsyncSession]] = ...\n    initial_state: Optional[states.State] = ...\n    proposed_state: Optional[states.State] = ...\n    validated_state: Optional[states.State]\n    rule_signature: List[str] = Field(default_factory=list)\n    finalization_signature: List[str] = Field(default_factory=list)\n    response_status: SetStateStatus = Field(default=SetStateStatus.ACCEPT)\n    response_details: StateResponseDetails = Field(default_factory=StateAcceptDetails)\n    orchestration_error: Optional[Exception] = Field(default=None)\n    parameters: Dict[Any, Any] = Field(default_factory=dict)\n\n    @property\n    def initial_state_type(self) -> Optional[states.StateType]:\n        \"\"\"The state type of `self.initial_state` if it exists.\"\"\"\n\n        return self.initial_state.type if self.initial_state else None\n\n    @property\n    def proposed_state_type(self) -> Optional[states.StateType]:\n        \"\"\"The state type of `self.proposed_state` if it exists.\"\"\"\n\n        return self.proposed_state.type if self.proposed_state else None\n\n    @property\n    def validated_state_type(self) -> Optional[states.StateType]:\n        \"\"\"The state type of `self.validated_state` if it exists.\"\"\"\n        return self.validated_state.type if self.validated_state else None\n\n    def safe_copy(self):\n        \"\"\"\n        Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n        Orchestration rules govern state transitions using information stored in\n        an `OrchestrationContext`. However, mutating objects stored on the context\n        directly can have unintended side-effects. To guard against this,\n        `self.safe_copy` can be used to pass information to orchestration rules\n        without risking mutation.\n\n        Returns:\n            A mutation-safe copy of the `OrchestrationContext`\n        \"\"\"\n\n        safe_copy = self.copy()\n\n        safe_copy.initial_state = (\n            self.initial_state.copy() if self.initial_state else None\n        )\n        safe_copy.proposed_state = (\n            self.proposed_state.copy() if self.proposed_state else None\n        )\n        safe_copy.validated_state = (\n            self.validated_state.copy() if self.validated_state else None\n        )\n        safe_copy.parameters = self.parameters.copy()\n        return safe_copy\n\n    def entry_context(self):\n        \"\"\"\n        A convenience method that generates input parameters for orchestration rules.\n\n        An `OrchestrationContext` defines a state transition that is managed by\n        orchestration rules which can fire hooks before a transition has been committed\n        to the database. These hooks have a consistent interface which can be generated\n        with this method.\n        \"\"\"\n\n        safe_context = self.safe_copy()\n        return safe_context.initial_state, safe_context.proposed_state, safe_context\n\n    def exit_context(self):\n        \"\"\"\n        A convenience method that generates input parameters for orchestration rules.\n\n        An `OrchestrationContext` defines a state transition that is managed by\n        orchestration rules which can fire hooks after a transition has been committed\n        to the database. These hooks have a consistent interface which can be generated\n        with this method.\n        \"\"\"\n\n        safe_context = self.safe_copy()\n        return safe_context.initial_state, safe_context.validated_state, safe_context\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.initial_state_type","title":"initial_state_type: Optional[states.StateType] property","text":"

    The state type of self.initial_state if it exists.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.proposed_state_type","title":"proposed_state_type: Optional[states.StateType] property","text":"

    The state type of self.proposed_state if it exists.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.validated_state_type","title":"validated_state_type: Optional[states.StateType] property","text":"

    The state type of self.validated_state if it exists.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.safe_copy","title":"safe_copy","text":"

    Creates a mostly-mutation-safe copy for use in orchestration rules.

    Orchestration rules govern state transitions using information stored in an OrchestrationContext. However, mutating objects stored on the context directly can have unintended side-effects. To guard against this, self.safe_copy can be used to pass information to orchestration rules without risking mutation.

    Returns:

    Type Description

    A mutation-safe copy of the OrchestrationContext

    Source code in prefect/server/orchestration/rules.py
    def safe_copy(self):\n    \"\"\"\n    Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n    Orchestration rules govern state transitions using information stored in\n    an `OrchestrationContext`. However, mutating objects stored on the context\n    directly can have unintended side-effects. To guard against this,\n    `self.safe_copy` can be used to pass information to orchestration rules\n    without risking mutation.\n\n    Returns:\n        A mutation-safe copy of the `OrchestrationContext`\n    \"\"\"\n\n    safe_copy = self.copy()\n\n    safe_copy.initial_state = (\n        self.initial_state.copy() if self.initial_state else None\n    )\n    safe_copy.proposed_state = (\n        self.proposed_state.copy() if self.proposed_state else None\n    )\n    safe_copy.validated_state = (\n        self.validated_state.copy() if self.validated_state else None\n    )\n    safe_copy.parameters = self.parameters.copy()\n    return safe_copy\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.entry_context","title":"entry_context","text":"

    A convenience method that generates input parameters for orchestration rules.

    An OrchestrationContext defines a state transition that is managed by orchestration rules which can fire hooks before a transition has been committed to the database. These hooks have a consistent interface which can be generated with this method.

    Source code in prefect/server/orchestration/rules.py
    def entry_context(self):\n    \"\"\"\n    A convenience method that generates input parameters for orchestration rules.\n\n    An `OrchestrationContext` defines a state transition that is managed by\n    orchestration rules which can fire hooks before a transition has been committed\n    to the database. These hooks have a consistent interface which can be generated\n    with this method.\n    \"\"\"\n\n    safe_context = self.safe_copy()\n    return safe_context.initial_state, safe_context.proposed_state, safe_context\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.OrchestrationContext.exit_context","title":"exit_context","text":"

    A convenience method that generates input parameters for orchestration rules.

    An OrchestrationContext defines a state transition that is managed by orchestration rules which can fire hooks after a transition has been committed to the database. These hooks have a consistent interface which can be generated with this method.

    Source code in prefect/server/orchestration/rules.py
    def exit_context(self):\n    \"\"\"\n    A convenience method that generates input parameters for orchestration rules.\n\n    An `OrchestrationContext` defines a state transition that is managed by\n    orchestration rules which can fire hooks after a transition has been committed\n    to the database. These hooks have a consistent interface which can be generated\n    with this method.\n    \"\"\"\n\n    safe_context = self.safe_copy()\n    return safe_context.initial_state, safe_context.validated_state, safe_context\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.FlowOrchestrationContext","title":"FlowOrchestrationContext","text":"

    Bases: OrchestrationContext

    A container for a flow run state transition, governed by orchestration rules.

    When a flow- run attempts to change state, Prefect REST API has an opportunity to decide whether this transition can proceed. All the relevant information associated with the state transition is stored in an OrchestrationContext, which is subsequently governed by nested orchestration rules implemented using the BaseOrchestrationRule ABC.

    FlowOrchestrationContext introduces the concept of a state being None in the context of an intended state transition. An initial state can be None if a run is is attempting to set a state for the first time. The proposed state might be None if a rule governing the transition determines that no state change should occur at all and nothing is written to the database.

    Attributes:

    Name Type Description session

    a SQLAlchemy database session

    run Any

    the flow run attempting to change state

    initial_state Any

    the initial state of the run

    proposed_state Any

    the proposed state the run is transitioning into

    validated_state Any

    a proposed state that has committed to the database

    rule_signature Any

    a record of rules that have fired on entry into a managed context, currently only used for debugging purposes

    finalization_signature Any

    a record of rules that have fired on exit from a managed context, currently only used for debugging purposes

    response_status Any

    a SetStateStatus object used to build the API response

    response_details Any

    a StateResponseDetails object use to build the API response

    Parameters:

    Name Type Description Default session

    a SQLAlchemy database session

    required run

    the flow run attempting to change state

    required initial_state

    the initial state of a run

    required proposed_state

    the proposed state a run is transitioning into

    required Source code in prefect/server/orchestration/rules.py
    class FlowOrchestrationContext(OrchestrationContext):\n    \"\"\"\n    A container for a flow run state transition, governed by orchestration rules.\n\n    When a flow- run attempts to change state, Prefect REST API has an opportunity\n    to decide whether this transition can proceed. All the relevant information\n    associated with the state transition is stored in an `OrchestrationContext`,\n    which is subsequently governed by nested orchestration rules implemented using\n    the `BaseOrchestrationRule` ABC.\n\n    `FlowOrchestrationContext` introduces the concept of a state being `None` in the\n    context of an intended state transition. An initial state can be `None` if a run\n    is is attempting to set a state for the first time. The proposed state might be\n    `None` if a rule governing the transition determines that no state change\n    should occur at all and nothing is written to the database.\n\n    Attributes:\n        session: a SQLAlchemy database session\n        run: the flow run attempting to change state\n        initial_state: the initial state of the run\n        proposed_state: the proposed state the run is transitioning into\n        validated_state: a proposed state that has committed to the database\n        rule_signature: a record of rules that have fired on entry into a\n            managed context, currently only used for debugging purposes\n        finalization_signature: a record of rules that have fired on exit from a\n            managed context, currently only used for debugging purposes\n        response_status: a SetStateStatus object used to build the API response\n        response_details:a StateResponseDetails object use to build the API response\n\n    Args:\n        session: a SQLAlchemy database session\n        run: the flow run attempting to change state\n        initial_state: the initial state of a run\n        proposed_state: the proposed state a run is transitioning into\n    \"\"\"\n\n    # run: db.FlowRun = ...\n    run: Any = ...\n\n    @inject_db\n    async def validate_proposed_state(\n        self,\n        db: PrefectDBInterface,\n    ):\n        \"\"\"\n        Validates a proposed state by committing it to the database.\n\n        After the `FlowOrchestrationContext` is governed by orchestration rules, the\n        proposed state can be validated: the proposed state is added to the current\n        SQLAlchemy session and is flushed. `self.validated_state` set to the flushed\n        state. The state on the run is set to the validated state as well.\n\n        If the proposed state is `None` when this method is called, no state will be\n        written and `self.validated_state` will be set to the run's current state.\n\n        Returns:\n            None\n        \"\"\"\n        # (circular import)\n        from prefect.server.api.server import is_client_retryable_exception\n\n        try:\n            await self._validate_proposed_state()\n            return\n        except Exception as exc:\n            logger.exception(\"Encountered error during state validation\")\n            self.proposed_state = None\n\n            if is_client_retryable_exception(exc):\n                # Do not capture retryable database exceptions, this exception will be\n                # raised as a 503 in the API layer\n                raise\n\n            reason = f\"Error validating state: {exc!r}\"\n            self.response_status = SetStateStatus.ABORT\n            self.response_details = StateAbortDetails(reason=reason)\n\n    @inject_db\n    async def _validate_proposed_state(\n        self,\n        db: PrefectDBInterface,\n    ):\n        if self.proposed_state is None:\n            validated_orm_state = self.run.state\n            # We cannot access `self.run.state.data` directly for unknown reasons\n            state_data = (\n                (\n                    await artifacts.read_artifact(\n                        self.session, self.run.state.result_artifact_id\n                    )\n                ).data\n                if self.run.state.result_artifact_id\n                else None\n            )\n        else:\n            state_payload = self.proposed_state.dict(shallow=True)\n            state_data = state_payload.pop(\"data\", None)\n\n            if state_data is not None:\n                state_result_artifact = core.Artifact.from_result(state_data)\n                state_result_artifact.flow_run_id = self.run.id\n                await artifacts.create_artifact(self.session, state_result_artifact)\n                state_payload[\"result_artifact_id\"] = state_result_artifact.id\n\n            validated_orm_state = db.FlowRunState(\n                flow_run_id=self.run.id,\n                **state_payload,\n            )\n\n        self.session.add(validated_orm_state)\n        self.run.set_state(validated_orm_state)\n\n        await self.session.flush()\n        if validated_orm_state:\n            self.validated_state = states.State.from_orm_without_result(\n                validated_orm_state, with_data=state_data\n            )\n        else:\n            self.validated_state = None\n\n    def safe_copy(self):\n        \"\"\"\n        Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n        Orchestration rules govern state transitions using information stored in\n        an `OrchestrationContext`. However, mutating objects stored on the context\n        directly can have unintended side-effects. To guard against this,\n        `self.safe_copy` can be used to pass information to orchestration rules\n        without risking mutation.\n\n        Note:\n            `self.run` is an ORM model, and even when copied is unsafe to mutate\n\n        Returns:\n            A mutation-safe copy of `FlowOrchestrationContext`\n        \"\"\"\n\n        return super().safe_copy()\n\n    @property\n    def run_settings(self) -> Dict:\n        \"\"\"Run-level settings used to orchestrate the state transition.\"\"\"\n\n        return self.run.empirical_policy\n\n    async def task_run(self):\n        return None\n\n    async def flow_run(self):\n        return self.run\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.FlowOrchestrationContext.run_settings","title":"run_settings: Dict property","text":"

    Run-level settings used to orchestrate the state transition.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.FlowOrchestrationContext.validate_proposed_state","title":"validate_proposed_state async","text":"

    Validates a proposed state by committing it to the database.

    After the FlowOrchestrationContext is governed by orchestration rules, the proposed state can be validated: the proposed state is added to the current SQLAlchemy session and is flushed. self.validated_state set to the flushed state. The state on the run is set to the validated state as well.

    If the proposed state is None when this method is called, no state will be written and self.validated_state will be set to the run's current state.

    Returns:

    Type Description

    None

    Source code in prefect/server/orchestration/rules.py
    @inject_db\nasync def validate_proposed_state(\n    self,\n    db: PrefectDBInterface,\n):\n    \"\"\"\n    Validates a proposed state by committing it to the database.\n\n    After the `FlowOrchestrationContext` is governed by orchestration rules, the\n    proposed state can be validated: the proposed state is added to the current\n    SQLAlchemy session and is flushed. `self.validated_state` set to the flushed\n    state. The state on the run is set to the validated state as well.\n\n    If the proposed state is `None` when this method is called, no state will be\n    written and `self.validated_state` will be set to the run's current state.\n\n    Returns:\n        None\n    \"\"\"\n    # (circular import)\n    from prefect.server.api.server import is_client_retryable_exception\n\n    try:\n        await self._validate_proposed_state()\n        return\n    except Exception as exc:\n        logger.exception(\"Encountered error during state validation\")\n        self.proposed_state = None\n\n        if is_client_retryable_exception(exc):\n            # Do not capture retryable database exceptions, this exception will be\n            # raised as a 503 in the API layer\n            raise\n\n        reason = f\"Error validating state: {exc!r}\"\n        self.response_status = SetStateStatus.ABORT\n        self.response_details = StateAbortDetails(reason=reason)\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.FlowOrchestrationContext.safe_copy","title":"safe_copy","text":"

    Creates a mostly-mutation-safe copy for use in orchestration rules.

    Orchestration rules govern state transitions using information stored in an OrchestrationContext. However, mutating objects stored on the context directly can have unintended side-effects. To guard against this, self.safe_copy can be used to pass information to orchestration rules without risking mutation.

    Note

    self.run is an ORM model, and even when copied is unsafe to mutate

    Returns:

    Type Description

    A mutation-safe copy of FlowOrchestrationContext

    Source code in prefect/server/orchestration/rules.py
    def safe_copy(self):\n    \"\"\"\n    Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n    Orchestration rules govern state transitions using information stored in\n    an `OrchestrationContext`. However, mutating objects stored on the context\n    directly can have unintended side-effects. To guard against this,\n    `self.safe_copy` can be used to pass information to orchestration rules\n    without risking mutation.\n\n    Note:\n        `self.run` is an ORM model, and even when copied is unsafe to mutate\n\n    Returns:\n        A mutation-safe copy of `FlowOrchestrationContext`\n    \"\"\"\n\n    return super().safe_copy()\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.TaskOrchestrationContext","title":"TaskOrchestrationContext","text":"

    Bases: OrchestrationContext

    A container for a task run state transition, governed by orchestration rules.

    When a task- run attempts to change state, Prefect REST API has an opportunity to decide whether this transition can proceed. All the relevant information associated with the state transition is stored in an OrchestrationContext, which is subsequently governed by nested orchestration rules implemented using the BaseOrchestrationRule ABC.

    TaskOrchestrationContext introduces the concept of a state being None in the context of an intended state transition. An initial state can be None if a run is is attempting to set a state for the first time. The proposed state might be None if a rule governing the transition determines that no state change should occur at all and nothing is written to the database.

    Attributes:

    Name Type Description session

    a SQLAlchemy database session

    run Any

    the task run attempting to change state

    initial_state Any

    the initial state of the run

    proposed_state Any

    the proposed state the run is transitioning into

    validated_state Any

    a proposed state that has committed to the database

    rule_signature Any

    a record of rules that have fired on entry into a managed context, currently only used for debugging purposes

    finalization_signature Any

    a record of rules that have fired on exit from a managed context, currently only used for debugging purposes

    response_status Any

    a SetStateStatus object used to build the API response

    response_details Any

    a StateResponseDetails object use to build the API response

    Parameters:

    Name Type Description Default session

    a SQLAlchemy database session

    required run

    the task run attempting to change state

    required initial_state

    the initial state of a run

    required proposed_state

    the proposed state a run is transitioning into

    required Source code in prefect/server/orchestration/rules.py
    class TaskOrchestrationContext(OrchestrationContext):\n    \"\"\"\n    A container for a task run state transition, governed by orchestration rules.\n\n    When a task- run attempts to change state, Prefect REST API has an opportunity\n    to decide whether this transition can proceed. All the relevant information\n    associated with the state transition is stored in an `OrchestrationContext`,\n    which is subsequently governed by nested orchestration rules implemented using\n    the `BaseOrchestrationRule` ABC.\n\n    `TaskOrchestrationContext` introduces the concept of a state being `None` in the\n    context of an intended state transition. An initial state can be `None` if a run\n    is is attempting to set a state for the first time. The proposed state might be\n    `None` if a rule governing the transition determines that no state change\n    should occur at all and nothing is written to the database.\n\n    Attributes:\n        session: a SQLAlchemy database session\n        run: the task run attempting to change state\n        initial_state: the initial state of the run\n        proposed_state: the proposed state the run is transitioning into\n        validated_state: a proposed state that has committed to the database\n        rule_signature: a record of rules that have fired on entry into a\n            managed context, currently only used for debugging purposes\n        finalization_signature: a record of rules that have fired on exit from a\n            managed context, currently only used for debugging purposes\n        response_status: a SetStateStatus object used to build the API response\n        response_details:a StateResponseDetails object use to build the API response\n\n    Args:\n        session: a SQLAlchemy database session\n        run: the task run attempting to change state\n        initial_state: the initial state of a run\n        proposed_state: the proposed state a run is transitioning into\n    \"\"\"\n\n    # run: db.TaskRun = ...\n    run: Any = ...\n\n    @inject_db\n    async def validate_proposed_state(\n        self,\n        db: PrefectDBInterface,\n    ):\n        \"\"\"\n        Validates a proposed state by committing it to the database.\n\n        After the `TaskOrchestrationContext` is governed by orchestration rules, the\n        proposed state can be validated: the proposed state is added to the current\n        SQLAlchemy session and is flushed. `self.validated_state` set to the flushed\n        state. The state on the run is set to the validated state as well.\n\n        If the proposed state is `None` when this method is called, no state will be\n        written and `self.validated_state` will be set to the run's current state.\n\n        Returns:\n            None\n        \"\"\"\n        # (circular import)\n        from prefect.server.api.server import is_client_retryable_exception\n\n        try:\n            await self._validate_proposed_state()\n            return\n        except Exception as exc:\n            logger.exception(\"Encountered error during state validation\")\n            self.proposed_state = None\n\n            if is_client_retryable_exception(exc):\n                # Do not capture retryable database exceptions, this exception will be\n                # raised as a 503 in the API layer\n                raise\n\n            reason = f\"Error validating state: {exc!r}\"\n            self.response_status = SetStateStatus.ABORT\n            self.response_details = StateAbortDetails(reason=reason)\n\n    @inject_db\n    async def _validate_proposed_state(\n        self,\n        db: PrefectDBInterface,\n    ):\n        if self.proposed_state is None:\n            validated_orm_state = self.run.state\n            # We cannot access `self.run.state.data` directly for unknown reasons\n            state_data = (\n                (\n                    await artifacts.read_artifact(\n                        self.session, self.run.state.result_artifact_id\n                    )\n                ).data\n                if self.run.state.result_artifact_id\n                else None\n            )\n        else:\n            state_payload = self.proposed_state.dict(shallow=True)\n            state_data = state_payload.pop(\"data\", None)\n\n            if state_data is not None:\n                state_result_artifact = core.Artifact.from_result(state_data)\n                state_result_artifact.task_run_id = self.run.id\n\n                if self.run.flow_run_id is not None:\n                    flow_run = await self.flow_run()\n                    state_result_artifact.flow_run_id = flow_run.id\n\n                await artifacts.create_artifact(self.session, state_result_artifact)\n                state_payload[\"result_artifact_id\"] = state_result_artifact.id\n\n            validated_orm_state = db.TaskRunState(\n                task_run_id=self.run.id,\n                **state_payload,\n            )\n\n        self.session.add(validated_orm_state)\n        self.run.set_state(validated_orm_state)\n\n        await self.session.flush()\n        if validated_orm_state:\n            self.validated_state = states.State.from_orm_without_result(\n                validated_orm_state, with_data=state_data\n            )\n        else:\n            self.validated_state = None\n\n    def safe_copy(self):\n        \"\"\"\n        Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n        Orchestration rules govern state transitions using information stored in\n        an `OrchestrationContext`. However, mutating objects stored on the context\n        directly can have unintended side-effects. To guard against this,\n        `self.safe_copy` can be used to pass information to orchestration rules\n        without risking mutation.\n\n        Note:\n            `self.run` is an ORM model, and even when copied is unsafe to mutate\n\n        Returns:\n            A mutation-safe copy of `TaskOrchestrationContext`\n        \"\"\"\n\n        return super().safe_copy()\n\n    @property\n    def run_settings(self) -> Dict:\n        \"\"\"Run-level settings used to orchestrate the state transition.\"\"\"\n\n        return self.run.empirical_policy\n\n    async def task_run(self):\n        return self.run\n\n    async def flow_run(self):\n        return await flow_runs.read_flow_run(\n            session=self.session,\n            flow_run_id=self.run.flow_run_id,\n        )\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.TaskOrchestrationContext.run_settings","title":"run_settings: Dict property","text":"

    Run-level settings used to orchestrate the state transition.

    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.TaskOrchestrationContext.validate_proposed_state","title":"validate_proposed_state async","text":"

    Validates a proposed state by committing it to the database.

    After the TaskOrchestrationContext is governed by orchestration rules, the proposed state can be validated: the proposed state is added to the current SQLAlchemy session and is flushed. self.validated_state set to the flushed state. The state on the run is set to the validated state as well.

    If the proposed state is None when this method is called, no state will be written and self.validated_state will be set to the run's current state.

    Returns:

    Type Description

    None

    Source code in prefect/server/orchestration/rules.py
    @inject_db\nasync def validate_proposed_state(\n    self,\n    db: PrefectDBInterface,\n):\n    \"\"\"\n    Validates a proposed state by committing it to the database.\n\n    After the `TaskOrchestrationContext` is governed by orchestration rules, the\n    proposed state can be validated: the proposed state is added to the current\n    SQLAlchemy session and is flushed. `self.validated_state` set to the flushed\n    state. The state on the run is set to the validated state as well.\n\n    If the proposed state is `None` when this method is called, no state will be\n    written and `self.validated_state` will be set to the run's current state.\n\n    Returns:\n        None\n    \"\"\"\n    # (circular import)\n    from prefect.server.api.server import is_client_retryable_exception\n\n    try:\n        await self._validate_proposed_state()\n        return\n    except Exception as exc:\n        logger.exception(\"Encountered error during state validation\")\n        self.proposed_state = None\n\n        if is_client_retryable_exception(exc):\n            # Do not capture retryable database exceptions, this exception will be\n            # raised as a 503 in the API layer\n            raise\n\n        reason = f\"Error validating state: {exc!r}\"\n        self.response_status = SetStateStatus.ABORT\n        self.response_details = StateAbortDetails(reason=reason)\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.TaskOrchestrationContext.safe_copy","title":"safe_copy","text":"

    Creates a mostly-mutation-safe copy for use in orchestration rules.

    Orchestration rules govern state transitions using information stored in an OrchestrationContext. However, mutating objects stored on the context directly can have unintended side-effects. To guard against this, self.safe_copy can be used to pass information to orchestration rules without risking mutation.

    Note

    self.run is an ORM model, and even when copied is unsafe to mutate

    Returns:

    Type Description

    A mutation-safe copy of TaskOrchestrationContext

    Source code in prefect/server/orchestration/rules.py
    def safe_copy(self):\n    \"\"\"\n    Creates a mostly-mutation-safe copy for use in orchestration rules.\n\n    Orchestration rules govern state transitions using information stored in\n    an `OrchestrationContext`. However, mutating objects stored on the context\n    directly can have unintended side-effects. To guard against this,\n    `self.safe_copy` can be used to pass information to orchestration rules\n    without risking mutation.\n\n    Note:\n        `self.run` is an ORM model, and even when copied is unsafe to mutate\n\n    Returns:\n        A mutation-safe copy of `TaskOrchestrationContext`\n    \"\"\"\n\n    return super().safe_copy()\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule","title":"BaseOrchestrationRule","text":"

    Bases: AbstractAsyncContextManager

    An abstract base class used to implement a discrete piece of orchestration logic.

    An OrchestrationRule is a stateful context manager that directly governs a state transition. Complex orchestration is achieved by nesting multiple rules. Each rule runs against an OrchestrationContext that contains the transition details; this context is then passed to subsequent rules. The context can be modified by hooks that fire before and after a new state is validated and committed to the database. These hooks will fire as long as the state transition is considered \"valid\" and govern a transition by either modifying the proposed state before it is validated or by producing a side-effect.

    A state transition occurs whenever a flow- or task- run changes state, prompting Prefect REST API to decide whether or not this transition can proceed. The current state of the run is referred to as the \"initial state\", and the state a run is attempting to transition into is the \"proposed state\". Together, the initial state transitioning into the proposed state is the intended transition that is governed by these orchestration rules. After using rules to enter a runtime context, the OrchestrationContext will contain a proposed state that has been governed by each rule, and at that point can validate the proposed state and commit it to the database. The validated state will be set on the context as context.validated_state, and rules will call the self.after_transition hook upon exiting the managed context.

    Examples:

    Create a rule:\n\n>>> class BasicRule(BaseOrchestrationRule):\n>>>     # allowed initial state types\n>>>     FROM_STATES = [StateType.RUNNING]\n>>>     # allowed proposed state types\n>>>     TO_STATES = [StateType.COMPLETED, StateType.FAILED]\n>>>\n>>>     async def before_transition(initial_state, proposed_state, ctx):\n>>>         # side effects and proposed state mutation can happen here\n>>>         ...\n>>>\n>>>     async def after_transition(initial_state, validated_state, ctx):\n>>>         # operations on states that have been validated can happen here\n>>>         ...\n>>>\n>>>     async def cleanup(intitial_state, validated_state, ctx):\n>>>         # reverts side effects generated by `before_transition` if necessary\n>>>         ...\n\nUse a rule:\n\n>>> intended_transition = (StateType.RUNNING, StateType.COMPLETED)\n>>> async with BasicRule(context, *intended_transition):\n>>>     # context.proposed_state has been governed by BasicRule\n>>>     ...\n\nUse multiple rules:\n\n>>> rules = [BasicRule, BasicRule]\n>>> intended_transition = (StateType.RUNNING, StateType.COMPLETED)\n>>> async with contextlib.AsyncExitStack() as stack:\n>>>     for rule in rules:\n>>>         stack.enter_async_context(rule(context, *intended_transition))\n>>>\n>>>     # context.proposed_state has been governed by all rules\n>>>     ...\n

    Attributes:

    Name Type Description FROM_STATES Iterable

    list of valid initial state types this rule governs

    TO_STATES Iterable

    list of valid proposed state types this rule governs

    context

    the orchestration context

    from_state_type

    the state type a run is currently in

    to_state_type

    the intended proposed state type prior to any orchestration

    Parameters:

    Name Type Description Default context OrchestrationContext

    A FlowOrchestrationContext or TaskOrchestrationContext that is passed between rules

    required from_state_type Optional[StateType]

    The state type of the initial state of a run, if this state type is not contained in FROM_STATES, no hooks will fire

    required to_state_type Optional[StateType]

    The state type of the proposed state before orchestration, if this state type is not contained in TO_STATES, no hooks will fire

    required Source code in prefect/server/orchestration/rules.py
    class BaseOrchestrationRule(contextlib.AbstractAsyncContextManager):\n    \"\"\"\n    An abstract base class used to implement a discrete piece of orchestration logic.\n\n    An `OrchestrationRule` is a stateful context manager that directly governs a state\n    transition. Complex orchestration is achieved by nesting multiple rules.\n    Each rule runs against an `OrchestrationContext` that contains the transition\n    details; this context is then passed to subsequent rules. The context can be\n    modified by hooks that fire before and after a new state is validated and committed\n    to the database. These hooks will fire as long as the state transition is\n    considered \"valid\" and govern a transition by either modifying the proposed state\n    before it is validated or by producing a side-effect.\n\n    A state transition occurs whenever a flow- or task- run changes state, prompting\n    Prefect REST API to decide whether or not this transition can proceed. The current state of\n    the run is referred to as the \"initial state\", and the state a run is\n    attempting to transition into is the \"proposed state\". Together, the initial state\n    transitioning into the proposed state is the intended transition that is governed\n    by these orchestration rules. After using rules to enter a runtime context, the\n    `OrchestrationContext` will contain a proposed state that has been governed by\n    each rule, and at that point can validate the proposed state and commit it to\n    the database. The validated state will be set on the context as\n    `context.validated_state`, and rules will call the `self.after_transition` hook\n    upon exiting the managed context.\n\n    Examples:\n\n        Create a rule:\n\n        >>> class BasicRule(BaseOrchestrationRule):\n        >>>     # allowed initial state types\n        >>>     FROM_STATES = [StateType.RUNNING]\n        >>>     # allowed proposed state types\n        >>>     TO_STATES = [StateType.COMPLETED, StateType.FAILED]\n        >>>\n        >>>     async def before_transition(initial_state, proposed_state, ctx):\n        >>>         # side effects and proposed state mutation can happen here\n        >>>         ...\n        >>>\n        >>>     async def after_transition(initial_state, validated_state, ctx):\n        >>>         # operations on states that have been validated can happen here\n        >>>         ...\n        >>>\n        >>>     async def cleanup(intitial_state, validated_state, ctx):\n        >>>         # reverts side effects generated by `before_transition` if necessary\n        >>>         ...\n\n        Use a rule:\n\n        >>> intended_transition = (StateType.RUNNING, StateType.COMPLETED)\n        >>> async with BasicRule(context, *intended_transition):\n        >>>     # context.proposed_state has been governed by BasicRule\n        >>>     ...\n\n        Use multiple rules:\n\n        >>> rules = [BasicRule, BasicRule]\n        >>> intended_transition = (StateType.RUNNING, StateType.COMPLETED)\n        >>> async with contextlib.AsyncExitStack() as stack:\n        >>>     for rule in rules:\n        >>>         stack.enter_async_context(rule(context, *intended_transition))\n        >>>\n        >>>     # context.proposed_state has been governed by all rules\n        >>>     ...\n\n    Attributes:\n        FROM_STATES: list of valid initial state types this rule governs\n        TO_STATES: list of valid proposed state types this rule governs\n        context: the orchestration context\n        from_state_type: the state type a run is currently in\n        to_state_type: the intended proposed state type prior to any orchestration\n\n    Args:\n        context: A `FlowOrchestrationContext` or `TaskOrchestrationContext` that is\n            passed between rules\n        from_state_type: The state type of the initial state of a run, if this\n            state type is not contained in `FROM_STATES`, no hooks will fire\n        to_state_type: The state type of the proposed state before orchestration, if\n            this state type is not contained in `TO_STATES`, no hooks will fire\n    \"\"\"\n\n    FROM_STATES: Iterable = []\n    TO_STATES: Iterable = []\n\n    def __init__(\n        self,\n        context: OrchestrationContext,\n        from_state_type: Optional[states.StateType],\n        to_state_type: Optional[states.StateType],\n    ):\n        self.context = context\n        self.from_state_type = from_state_type\n        self.to_state_type = to_state_type\n        self._invalid_on_entry = None\n\n    async def __aenter__(self) -> OrchestrationContext:\n        \"\"\"\n        Enter an async runtime context governed by this rule.\n\n        The `with` statement will bind a governed `OrchestrationContext` to the target\n        specified by the `as` clause. If the transition proposed by the\n        `OrchestrationContext` is considered invalid on entry, entering this context\n        will do nothing. Otherwise, `self.before_transition` will fire.\n        \"\"\"\n\n        if await self.invalid():\n            pass\n        else:\n            try:\n                entry_context = self.context.entry_context()\n                await self.before_transition(*entry_context)\n                self.context.rule_signature.append(str(self.__class__))\n            except Exception as before_transition_error:\n                reason = (\n                    f\"Aborting orchestration due to error in {self.__class__!r}:\"\n                    f\" !{before_transition_error!r}\"\n                )\n                logger.exception(\n                    f\"Error running before-transition hook in rule {self.__class__!r}:\"\n                    f\" !{before_transition_error!r}\"\n                )\n\n                self.context.proposed_state = None\n                self.context.response_status = SetStateStatus.ABORT\n                self.context.response_details = StateAbortDetails(reason=reason)\n                self.context.orchestration_error = before_transition_error\n\n        return self.context\n\n    async def __aexit__(\n        self,\n        exc_type: Optional[Type[BaseException]],\n        exc_val: Optional[BaseException],\n        exc_tb: Optional[TracebackType],\n    ) -> None:\n        \"\"\"\n        Exit the async runtime context governed by this rule.\n\n        One of three outcomes can happen upon exiting this rule's context depending on\n        the state of the rule. If the rule was found to be invalid on entry, nothing\n        happens. If the rule was valid on entry and continues to be valid on exit,\n        `self.after_transition` will fire. If the rule was valid on entry but invalid\n        on exit, the rule will \"fizzle\" and `self.cleanup` will fire in order to revert\n        any side-effects produced by `self.before_transition`.\n        \"\"\"\n\n        exit_context = self.context.exit_context()\n        if await self.invalid():\n            pass\n        elif await self.fizzled():\n            await self.cleanup(*exit_context)\n        else:\n            await self.after_transition(*exit_context)\n            self.context.finalization_signature.append(str(self.__class__))\n\n    async def before_transition(\n        self,\n        initial_state: Optional[states.State],\n        proposed_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        \"\"\"\n        Implements a hook that can fire before a state is committed to the database.\n\n        This hook may produce side-effects or mutate the proposed state of a\n        transition using one of four methods: `self.reject_transition`,\n        `self.delay_transition`, `self.abort_transition`, and `self.rename_state`.\n\n        Note:\n            As currently implemented, the `before_transition` hook is not\n            perfectly isolated from mutating the transition. It is a standard instance\n            method that has access to `self`, and therefore `self.context`. This should\n            never be modified directly. Furthermore, `context.run` is an ORM model, and\n            mutating the run can also cause unintended writes to the database.\n\n        Args:\n            initial_state: The initial state of a transition\n            proposed_state: The proposed state of a transition\n            context: A safe copy of the `OrchestrationContext`, with the exception of\n                `context.run`, mutating this context will have no effect on the broader\n                orchestration environment.\n\n        Returns:\n            None\n        \"\"\"\n\n    async def after_transition(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        \"\"\"\n        Implements a hook that can fire after a state is committed to the database.\n\n        Args:\n            initial_state: The initial state of a transition\n            validated_state: The governed state that has been committed to the database\n            context: A safe copy of the `OrchestrationContext`, with the exception of\n                `context.run`, mutating this context will have no effect on the broader\n                orchestration environment.\n\n        Returns:\n            None\n        \"\"\"\n\n    async def cleanup(\n        self,\n        initial_state: Optional[states.State],\n        validated_state: Optional[states.State],\n        context: OrchestrationContext,\n    ) -> None:\n        \"\"\"\n        Implements a hook that can fire after a state is committed to the database.\n\n        The intended use of this method is to revert side-effects produced by\n        `self.before_transition` when the transition is found to be invalid on exit.\n        This allows multiple rules to be gracefully run in sequence, without logic that\n        keeps track of all other rules that might govern a transition.\n\n        Args:\n            initial_state: The initial state of a transition\n            validated_state: The governed state that has been committed to the database\n            context: A safe copy of the `OrchestrationContext`, with the exception of\n                `context.run`, mutating this context will have no effect on the broader\n                orchestration environment.\n\n        Returns:\n            None\n        \"\"\"\n\n    async def invalid(self) -> bool:\n        \"\"\"\n        Determines if a rule is invalid.\n\n        Invalid rules do nothing and no hooks fire upon entering or exiting a governed\n        context. Rules are invalid if the transition states types are not contained in\n        `self.FROM_STATES` and `self.TO_STATES`, or if the context is proposing\n        a transition that differs from the transition the rule was instantiated with.\n\n        Returns:\n            True if the rules in invalid, False otherwise.\n        \"\"\"\n        # invalid and fizzled states are mutually exclusive,\n        # `_invalid_on_entry` holds this statefulness\n        if self.from_state_type not in self.FROM_STATES:\n            self._invalid_on_entry = True\n        if self.to_state_type not in self.TO_STATES:\n            self._invalid_on_entry = True\n\n        if self._invalid_on_entry is None:\n            self._invalid_on_entry = await self.invalid_transition()\n        return self._invalid_on_entry\n\n    async def fizzled(self) -> bool:\n        \"\"\"\n        Determines if a rule is fizzled and side-effects need to be reverted.\n\n        Rules are fizzled if the transitions were valid on entry (thus firing\n        `self.before_transition`) but are invalid upon exiting the governed context,\n        most likely caused by another rule mutating the transition.\n\n        Returns:\n            True if the rule is fizzled, False otherwise.\n        \"\"\"\n\n        if self._invalid_on_entry:\n            return False\n        return await self.invalid_transition()\n\n    async def invalid_transition(self) -> bool:\n        \"\"\"\n        Determines if the transition proposed by the `OrchestrationContext` is invalid.\n\n        If the `OrchestrationContext` is attempting to manage a transition with this\n        rule that differs from the transition the rule was instantiated with, the\n        transition is considered to be invalid. Depending on the context, a rule with an\n        invalid transition is either \"invalid\" or \"fizzled\".\n\n        Returns:\n            True if the transition is invalid, False otherwise.\n        \"\"\"\n\n        initial_state_type = self.context.initial_state_type\n        proposed_state_type = self.context.proposed_state_type\n        return (self.from_state_type != initial_state_type) or (\n            self.to_state_type != proposed_state_type\n        )\n\n    async def reject_transition(self, state: Optional[states.State], reason: str):\n        \"\"\"\n        Rejects a proposed transition before the transition is validated.\n\n        This method will reject a proposed transition, mutating the proposed state to\n        the provided `state`. A reason for rejecting the transition is also passed on\n        to the `OrchestrationContext`. Rules that reject the transition will not fizzle,\n        despite the proposed state type changing.\n\n        Args:\n            state: The new proposed state. If `None`, the current run state will be\n                returned in the result instead.\n            reason: The reason for rejecting the transition\n        \"\"\"\n\n        # don't run if the transition is already validated\n        if self.context.validated_state:\n            raise RuntimeError(\"The transition is already validated\")\n\n        # the current state will be used if a new one is not provided\n        if state is None:\n            if self.from_state_type is None:\n                raise OrchestrationError(\n                    \"The current run has no state; this transition cannot be \"\n                    \"rejected without providing a new state.\"\n                )\n            self.to_state_type = None\n            self.context.proposed_state = None\n        else:\n            # a rule that mutates state should not fizzle itself\n            self.to_state_type = state.type\n            self.context.proposed_state = state\n\n        self.context.response_status = SetStateStatus.REJECT\n        self.context.response_details = StateRejectDetails(reason=reason)\n\n    async def delay_transition(\n        self,\n        delay_seconds: int,\n        reason: str,\n    ):\n        \"\"\"\n        Delays a proposed transition before the transition is validated.\n\n        This method will delay a proposed transition, setting the proposed state to\n        `None`, signaling to the `OrchestrationContext` that no state should be\n        written to the database. The number of seconds a transition should be delayed is\n        passed to the `OrchestrationContext`. A reason for delaying the transition is\n        also provided. Rules that delay the transition will not fizzle, despite the\n        proposed state type changing.\n\n        Args:\n            delay_seconds: The number of seconds the transition should be delayed\n            reason: The reason for delaying the transition\n        \"\"\"\n\n        # don't run if the transition is already validated\n        if self.context.validated_state:\n            raise RuntimeError(\"The transition is already validated\")\n\n        # a rule that mutates state should not fizzle itself\n        self.to_state_type = None\n        self.context.proposed_state = None\n        self.context.response_status = SetStateStatus.WAIT\n        self.context.response_details = StateWaitDetails(\n            delay_seconds=delay_seconds, reason=reason\n        )\n\n    async def abort_transition(self, reason: str):\n        \"\"\"\n        Aborts a proposed transition before the transition is validated.\n\n        This method will abort a proposed transition, expecting no further action to\n        occur for this run. The proposed state is set to `None`, signaling to the\n        `OrchestrationContext` that no state should be written to the database. A\n        reason for aborting the transition is also provided. Rules that abort the\n        transition will not fizzle, despite the proposed state type changing.\n\n        Args:\n            reason: The reason for aborting the transition\n        \"\"\"\n\n        # don't run if the transition is already validated\n        if self.context.validated_state:\n            raise RuntimeError(\"The transition is already validated\")\n\n        # a rule that mutates state should not fizzle itself\n        self.to_state_type = None\n        self.context.proposed_state = None\n        self.context.response_status = SetStateStatus.ABORT\n        self.context.response_details = StateAbortDetails(reason=reason)\n\n    async def rename_state(self, state_name):\n        \"\"\"\n        Sets the \"name\" attribute on a proposed state.\n\n        The name of a state is an annotation intended to provide rich, human-readable\n        context for how a run is progressing. This method only updates the name and not\n        the canonical state TYPE, and will not fizzle or invalidate any other rules\n        that might govern this state transition.\n        \"\"\"\n        if self.context.proposed_state is not None:\n            self.context.proposed_state.name = state_name\n\n    async def update_context_parameters(self, key, value):\n        \"\"\"\n        Updates the \"parameters\" dictionary attribute with the specified key-value pair.\n\n        This mechanism streamlines the process of passing messages and information\n        between orchestration rules if necessary and is simpler and more ephemeral than\n        message-passing via the database or some other side-effect. This mechanism can\n        be used to break up large rules for ease of testing or comprehension, but note\n        that any rules coupled this way (or any other way) are no longer independent and\n        the order in which they appear in the orchestration policy priority will matter.\n        \"\"\"\n\n        self.context.parameters.update({key: value})\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.before_transition","title":"before_transition async","text":"

    Implements a hook that can fire before a state is committed to the database.

    This hook may produce side-effects or mutate the proposed state of a transition using one of four methods: self.reject_transition, self.delay_transition, self.abort_transition, and self.rename_state.

    Note

    As currently implemented, the before_transition hook is not perfectly isolated from mutating the transition. It is a standard instance method that has access to self, and therefore self.context. This should never be modified directly. Furthermore, context.run is an ORM model, and mutating the run can also cause unintended writes to the database.

    Parameters:

    Name Type Description Default initial_state Optional[State]

    The initial state of a transition

    required proposed_state Optional[State]

    The proposed state of a transition

    required context OrchestrationContext

    A safe copy of the OrchestrationContext, with the exception of context.run, mutating this context will have no effect on the broader orchestration environment.

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def before_transition(\n    self,\n    initial_state: Optional[states.State],\n    proposed_state: Optional[states.State],\n    context: OrchestrationContext,\n) -> None:\n    \"\"\"\n    Implements a hook that can fire before a state is committed to the database.\n\n    This hook may produce side-effects or mutate the proposed state of a\n    transition using one of four methods: `self.reject_transition`,\n    `self.delay_transition`, `self.abort_transition`, and `self.rename_state`.\n\n    Note:\n        As currently implemented, the `before_transition` hook is not\n        perfectly isolated from mutating the transition. It is a standard instance\n        method that has access to `self`, and therefore `self.context`. This should\n        never be modified directly. Furthermore, `context.run` is an ORM model, and\n        mutating the run can also cause unintended writes to the database.\n\n    Args:\n        initial_state: The initial state of a transition\n        proposed_state: The proposed state of a transition\n        context: A safe copy of the `OrchestrationContext`, with the exception of\n            `context.run`, mutating this context will have no effect on the broader\n            orchestration environment.\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.after_transition","title":"after_transition async","text":"

    Implements a hook that can fire after a state is committed to the database.

    Parameters:

    Name Type Description Default initial_state Optional[State]

    The initial state of a transition

    required validated_state Optional[State]

    The governed state that has been committed to the database

    required context OrchestrationContext

    A safe copy of the OrchestrationContext, with the exception of context.run, mutating this context will have no effect on the broader orchestration environment.

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def after_transition(\n    self,\n    initial_state: Optional[states.State],\n    validated_state: Optional[states.State],\n    context: OrchestrationContext,\n) -> None:\n    \"\"\"\n    Implements a hook that can fire after a state is committed to the database.\n\n    Args:\n        initial_state: The initial state of a transition\n        validated_state: The governed state that has been committed to the database\n        context: A safe copy of the `OrchestrationContext`, with the exception of\n            `context.run`, mutating this context will have no effect on the broader\n            orchestration environment.\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.cleanup","title":"cleanup async","text":"

    Implements a hook that can fire after a state is committed to the database.

    The intended use of this method is to revert side-effects produced by self.before_transition when the transition is found to be invalid on exit. This allows multiple rules to be gracefully run in sequence, without logic that keeps track of all other rules that might govern a transition.

    Parameters:

    Name Type Description Default initial_state Optional[State]

    The initial state of a transition

    required validated_state Optional[State]

    The governed state that has been committed to the database

    required context OrchestrationContext

    A safe copy of the OrchestrationContext, with the exception of context.run, mutating this context will have no effect on the broader orchestration environment.

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def cleanup(\n    self,\n    initial_state: Optional[states.State],\n    validated_state: Optional[states.State],\n    context: OrchestrationContext,\n) -> None:\n    \"\"\"\n    Implements a hook that can fire after a state is committed to the database.\n\n    The intended use of this method is to revert side-effects produced by\n    `self.before_transition` when the transition is found to be invalid on exit.\n    This allows multiple rules to be gracefully run in sequence, without logic that\n    keeps track of all other rules that might govern a transition.\n\n    Args:\n        initial_state: The initial state of a transition\n        validated_state: The governed state that has been committed to the database\n        context: A safe copy of the `OrchestrationContext`, with the exception of\n            `context.run`, mutating this context will have no effect on the broader\n            orchestration environment.\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.invalid","title":"invalid async","text":"

    Determines if a rule is invalid.

    Invalid rules do nothing and no hooks fire upon entering or exiting a governed context. Rules are invalid if the transition states types are not contained in self.FROM_STATES and self.TO_STATES, or if the context is proposing a transition that differs from the transition the rule was instantiated with.

    Returns:

    Type Description bool

    True if the rules in invalid, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    async def invalid(self) -> bool:\n    \"\"\"\n    Determines if a rule is invalid.\n\n    Invalid rules do nothing and no hooks fire upon entering or exiting a governed\n    context. Rules are invalid if the transition states types are not contained in\n    `self.FROM_STATES` and `self.TO_STATES`, or if the context is proposing\n    a transition that differs from the transition the rule was instantiated with.\n\n    Returns:\n        True if the rules in invalid, False otherwise.\n    \"\"\"\n    # invalid and fizzled states are mutually exclusive,\n    # `_invalid_on_entry` holds this statefulness\n    if self.from_state_type not in self.FROM_STATES:\n        self._invalid_on_entry = True\n    if self.to_state_type not in self.TO_STATES:\n        self._invalid_on_entry = True\n\n    if self._invalid_on_entry is None:\n        self._invalid_on_entry = await self.invalid_transition()\n    return self._invalid_on_entry\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.fizzled","title":"fizzled async","text":"

    Determines if a rule is fizzled and side-effects need to be reverted.

    Rules are fizzled if the transitions were valid on entry (thus firing self.before_transition) but are invalid upon exiting the governed context, most likely caused by another rule mutating the transition.

    Returns:

    Type Description bool

    True if the rule is fizzled, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    async def fizzled(self) -> bool:\n    \"\"\"\n    Determines if a rule is fizzled and side-effects need to be reverted.\n\n    Rules are fizzled if the transitions were valid on entry (thus firing\n    `self.before_transition`) but are invalid upon exiting the governed context,\n    most likely caused by another rule mutating the transition.\n\n    Returns:\n        True if the rule is fizzled, False otherwise.\n    \"\"\"\n\n    if self._invalid_on_entry:\n        return False\n    return await self.invalid_transition()\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.invalid_transition","title":"invalid_transition async","text":"

    Determines if the transition proposed by the OrchestrationContext is invalid.

    If the OrchestrationContext is attempting to manage a transition with this rule that differs from the transition the rule was instantiated with, the transition is considered to be invalid. Depending on the context, a rule with an invalid transition is either \"invalid\" or \"fizzled\".

    Returns:

    Type Description bool

    True if the transition is invalid, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    async def invalid_transition(self) -> bool:\n    \"\"\"\n    Determines if the transition proposed by the `OrchestrationContext` is invalid.\n\n    If the `OrchestrationContext` is attempting to manage a transition with this\n    rule that differs from the transition the rule was instantiated with, the\n    transition is considered to be invalid. Depending on the context, a rule with an\n    invalid transition is either \"invalid\" or \"fizzled\".\n\n    Returns:\n        True if the transition is invalid, False otherwise.\n    \"\"\"\n\n    initial_state_type = self.context.initial_state_type\n    proposed_state_type = self.context.proposed_state_type\n    return (self.from_state_type != initial_state_type) or (\n        self.to_state_type != proposed_state_type\n    )\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.reject_transition","title":"reject_transition async","text":"

    Rejects a proposed transition before the transition is validated.

    This method will reject a proposed transition, mutating the proposed state to the provided state. A reason for rejecting the transition is also passed on to the OrchestrationContext. Rules that reject the transition will not fizzle, despite the proposed state type changing.

    Parameters:

    Name Type Description Default state Optional[State]

    The new proposed state. If None, the current run state will be returned in the result instead.

    required reason str

    The reason for rejecting the transition

    required Source code in prefect/server/orchestration/rules.py
    async def reject_transition(self, state: Optional[states.State], reason: str):\n    \"\"\"\n    Rejects a proposed transition before the transition is validated.\n\n    This method will reject a proposed transition, mutating the proposed state to\n    the provided `state`. A reason for rejecting the transition is also passed on\n    to the `OrchestrationContext`. Rules that reject the transition will not fizzle,\n    despite the proposed state type changing.\n\n    Args:\n        state: The new proposed state. If `None`, the current run state will be\n            returned in the result instead.\n        reason: The reason for rejecting the transition\n    \"\"\"\n\n    # don't run if the transition is already validated\n    if self.context.validated_state:\n        raise RuntimeError(\"The transition is already validated\")\n\n    # the current state will be used if a new one is not provided\n    if state is None:\n        if self.from_state_type is None:\n            raise OrchestrationError(\n                \"The current run has no state; this transition cannot be \"\n                \"rejected without providing a new state.\"\n            )\n        self.to_state_type = None\n        self.context.proposed_state = None\n    else:\n        # a rule that mutates state should not fizzle itself\n        self.to_state_type = state.type\n        self.context.proposed_state = state\n\n    self.context.response_status = SetStateStatus.REJECT\n    self.context.response_details = StateRejectDetails(reason=reason)\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.delay_transition","title":"delay_transition async","text":"

    Delays a proposed transition before the transition is validated.

    This method will delay a proposed transition, setting the proposed state to None, signaling to the OrchestrationContext that no state should be written to the database. The number of seconds a transition should be delayed is passed to the OrchestrationContext. A reason for delaying the transition is also provided. Rules that delay the transition will not fizzle, despite the proposed state type changing.

    Parameters:

    Name Type Description Default delay_seconds int

    The number of seconds the transition should be delayed

    required reason str

    The reason for delaying the transition

    required Source code in prefect/server/orchestration/rules.py
    async def delay_transition(\n    self,\n    delay_seconds: int,\n    reason: str,\n):\n    \"\"\"\n    Delays a proposed transition before the transition is validated.\n\n    This method will delay a proposed transition, setting the proposed state to\n    `None`, signaling to the `OrchestrationContext` that no state should be\n    written to the database. The number of seconds a transition should be delayed is\n    passed to the `OrchestrationContext`. A reason for delaying the transition is\n    also provided. Rules that delay the transition will not fizzle, despite the\n    proposed state type changing.\n\n    Args:\n        delay_seconds: The number of seconds the transition should be delayed\n        reason: The reason for delaying the transition\n    \"\"\"\n\n    # don't run if the transition is already validated\n    if self.context.validated_state:\n        raise RuntimeError(\"The transition is already validated\")\n\n    # a rule that mutates state should not fizzle itself\n    self.to_state_type = None\n    self.context.proposed_state = None\n    self.context.response_status = SetStateStatus.WAIT\n    self.context.response_details = StateWaitDetails(\n        delay_seconds=delay_seconds, reason=reason\n    )\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.abort_transition","title":"abort_transition async","text":"

    Aborts a proposed transition before the transition is validated.

    This method will abort a proposed transition, expecting no further action to occur for this run. The proposed state is set to None, signaling to the OrchestrationContext that no state should be written to the database. A reason for aborting the transition is also provided. Rules that abort the transition will not fizzle, despite the proposed state type changing.

    Parameters:

    Name Type Description Default reason str

    The reason for aborting the transition

    required Source code in prefect/server/orchestration/rules.py
    async def abort_transition(self, reason: str):\n    \"\"\"\n    Aborts a proposed transition before the transition is validated.\n\n    This method will abort a proposed transition, expecting no further action to\n    occur for this run. The proposed state is set to `None`, signaling to the\n    `OrchestrationContext` that no state should be written to the database. A\n    reason for aborting the transition is also provided. Rules that abort the\n    transition will not fizzle, despite the proposed state type changing.\n\n    Args:\n        reason: The reason for aborting the transition\n    \"\"\"\n\n    # don't run if the transition is already validated\n    if self.context.validated_state:\n        raise RuntimeError(\"The transition is already validated\")\n\n    # a rule that mutates state should not fizzle itself\n    self.to_state_type = None\n    self.context.proposed_state = None\n    self.context.response_status = SetStateStatus.ABORT\n    self.context.response_details = StateAbortDetails(reason=reason)\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.rename_state","title":"rename_state async","text":"

    Sets the \"name\" attribute on a proposed state.

    The name of a state is an annotation intended to provide rich, human-readable context for how a run is progressing. This method only updates the name and not the canonical state TYPE, and will not fizzle or invalidate any other rules that might govern this state transition.

    Source code in prefect/server/orchestration/rules.py
    async def rename_state(self, state_name):\n    \"\"\"\n    Sets the \"name\" attribute on a proposed state.\n\n    The name of a state is an annotation intended to provide rich, human-readable\n    context for how a run is progressing. This method only updates the name and not\n    the canonical state TYPE, and will not fizzle or invalidate any other rules\n    that might govern this state transition.\n    \"\"\"\n    if self.context.proposed_state is not None:\n        self.context.proposed_state.name = state_name\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseOrchestrationRule.update_context_parameters","title":"update_context_parameters async","text":"

    Updates the \"parameters\" dictionary attribute with the specified key-value pair.

    This mechanism streamlines the process of passing messages and information between orchestration rules if necessary and is simpler and more ephemeral than message-passing via the database or some other side-effect. This mechanism can be used to break up large rules for ease of testing or comprehension, but note that any rules coupled this way (or any other way) are no longer independent and the order in which they appear in the orchestration policy priority will matter.

    Source code in prefect/server/orchestration/rules.py
    async def update_context_parameters(self, key, value):\n    \"\"\"\n    Updates the \"parameters\" dictionary attribute with the specified key-value pair.\n\n    This mechanism streamlines the process of passing messages and information\n    between orchestration rules if necessary and is simpler and more ephemeral than\n    message-passing via the database or some other side-effect. This mechanism can\n    be used to break up large rules for ease of testing or comprehension, but note\n    that any rules coupled this way (or any other way) are no longer independent and\n    the order in which they appear in the orchestration policy priority will matter.\n    \"\"\"\n\n    self.context.parameters.update({key: value})\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform","title":"BaseUniversalTransform","text":"

    Bases: AbstractAsyncContextManager

    An abstract base class used to implement privileged bookkeeping logic.

    Warning

    In almost all cases, use the BaseOrchestrationRule base class instead.

    Beyond the orchestration rules implemented with the BaseOrchestrationRule ABC, Universal transforms are not stateful, and fire their before- and after- transition hooks on every state transition unless the proposed state is None, indicating that no state should be written to the database. Because there are no guardrails in place to prevent directly mutating state or other parts of the orchestration context, universal transforms should only be used with care.

    Attributes:

    Name Type Description FROM_STATES Iterable

    for compatibility with BaseOrchestrationPolicy

    TO_STATES Iterable

    for compatibility with BaseOrchestrationPolicy

    context

    the orchestration context

    from_state_type

    the state type a run is currently in

    to_state_type

    the intended proposed state type prior to any orchestration

    Parameters:

    Name Type Description Default context OrchestrationContext

    A FlowOrchestrationContext or TaskOrchestrationContext that is passed between transforms

    required Source code in prefect/server/orchestration/rules.py
    class BaseUniversalTransform(contextlib.AbstractAsyncContextManager):\n    \"\"\"\n    An abstract base class used to implement privileged bookkeeping logic.\n\n    Warning:\n        In almost all cases, use the `BaseOrchestrationRule` base class instead.\n\n    Beyond the orchestration rules implemented with the `BaseOrchestrationRule` ABC,\n    Universal transforms are not stateful, and fire their before- and after- transition\n    hooks on every state transition unless the proposed state is `None`, indicating that\n    no state should be written to the database. Because there are no guardrails in place\n    to prevent directly mutating state or other parts of the orchestration context,\n    universal transforms should only be used with care.\n\n    Attributes:\n        FROM_STATES: for compatibility with `BaseOrchestrationPolicy`\n        TO_STATES: for compatibility with `BaseOrchestrationPolicy`\n        context: the orchestration context\n        from_state_type: the state type a run is currently in\n        to_state_type: the intended proposed state type prior to any orchestration\n\n    Args:\n        context: A `FlowOrchestrationContext` or `TaskOrchestrationContext` that is\n            passed between transforms\n    \"\"\"\n\n    # `BaseUniversalTransform` will always fire on non-null transitions\n    FROM_STATES: Iterable = ALL_ORCHESTRATION_STATES\n    TO_STATES: Iterable = ALL_ORCHESTRATION_STATES\n\n    def __init__(\n        self,\n        context: OrchestrationContext,\n        from_state_type: Optional[states.StateType],\n        to_state_type: Optional[states.StateType],\n    ):\n        self.context = context\n        self.from_state_type = from_state_type\n        self.to_state_type = to_state_type\n\n    async def __aenter__(self):\n        \"\"\"\n        Enter an async runtime context governed by this transform.\n\n        The `with` statement will bind a governed `OrchestrationContext` to the target\n        specified by the `as` clause. If the transition proposed by the\n        `OrchestrationContext` has been nullified on entry and `context.proposed_state`\n        is `None`, entering this context will do nothing. Otherwise\n        `self.before_transition` will fire.\n        \"\"\"\n\n        await self.before_transition(self.context)\n        self.context.rule_signature.append(str(self.__class__))\n        return self.context\n\n    async def __aexit__(\n        self,\n        exc_type: Optional[Type[BaseException]],\n        exc_val: Optional[BaseException],\n        exc_tb: Optional[TracebackType],\n    ) -> None:\n        \"\"\"\n        Exit the async runtime context governed by this transform.\n\n        If the transition has been nullified or errorred upon exiting this transforms's context,\n        nothing happens. Otherwise, `self.after_transition` will fire on every non-null\n        proposed state.\n        \"\"\"\n\n        if not self.exception_in_transition():\n            await self.after_transition(self.context)\n            self.context.finalization_signature.append(str(self.__class__))\n\n    async def before_transition(self, context) -> None:\n        \"\"\"\n        Implements a hook that fires before a state is committed to the database.\n\n        Args:\n            context: the `OrchestrationContext` that contains transition details\n\n        Returns:\n            None\n        \"\"\"\n\n    async def after_transition(self, context) -> None:\n        \"\"\"\n        Implements a hook that can fire after a state is committed to the database.\n\n        Args:\n            context: the `OrchestrationContext` that contains transition details\n\n        Returns:\n            None\n        \"\"\"\n\n    def nullified_transition(self) -> bool:\n        \"\"\"\n        Determines if the transition has been nullified.\n\n        Transitions are nullified if the proposed state is `None`, indicating that\n        nothing should be written to the database.\n\n        Returns:\n            True if the transition is nullified, False otherwise.\n        \"\"\"\n\n        return self.context.proposed_state is None\n\n    def exception_in_transition(self) -> bool:\n        \"\"\"\n        Determines if the transition has encountered an exception.\n\n        Returns:\n            True if the transition is encountered an exception, False otherwise.\n        \"\"\"\n\n        return self.context.orchestration_error is not None\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform.before_transition","title":"before_transition async","text":"

    Implements a hook that fires before a state is committed to the database.

    Parameters:

    Name Type Description Default context

    the OrchestrationContext that contains transition details

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def before_transition(self, context) -> None:\n    \"\"\"\n    Implements a hook that fires before a state is committed to the database.\n\n    Args:\n        context: the `OrchestrationContext` that contains transition details\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform.after_transition","title":"after_transition async","text":"

    Implements a hook that can fire after a state is committed to the database.

    Parameters:

    Name Type Description Default context

    the OrchestrationContext that contains transition details

    required

    Returns:

    Type Description None

    None

    Source code in prefect/server/orchestration/rules.py
    async def after_transition(self, context) -> None:\n    \"\"\"\n    Implements a hook that can fire after a state is committed to the database.\n\n    Args:\n        context: the `OrchestrationContext` that contains transition details\n\n    Returns:\n        None\n    \"\"\"\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform.nullified_transition","title":"nullified_transition","text":"

    Determines if the transition has been nullified.

    Transitions are nullified if the proposed state is None, indicating that nothing should be written to the database.

    Returns:

    Type Description bool

    True if the transition is nullified, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    def nullified_transition(self) -> bool:\n    \"\"\"\n    Determines if the transition has been nullified.\n\n    Transitions are nullified if the proposed state is `None`, indicating that\n    nothing should be written to the database.\n\n    Returns:\n        True if the transition is nullified, False otherwise.\n    \"\"\"\n\n    return self.context.proposed_state is None\n
    "},{"location":"api-ref/server/orchestration/rules/#prefect.server.orchestration.rules.BaseUniversalTransform.exception_in_transition","title":"exception_in_transition","text":"

    Determines if the transition has encountered an exception.

    Returns:

    Type Description bool

    True if the transition is encountered an exception, False otherwise.

    Source code in prefect/server/orchestration/rules.py
    def exception_in_transition(self) -> bool:\n    \"\"\"\n    Determines if the transition has encountered an exception.\n\n    Returns:\n        True if the transition is encountered an exception, False otherwise.\n    \"\"\"\n\n    return self.context.orchestration_error is not None\n
    "},{"location":"api-ref/server/schemas/actions/","title":"server.schemas.actions","text":""},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions","title":"prefect.server.schemas.actions","text":"

    Reduced schemas for accepting API actions.

    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ArtifactCreate","title":"ArtifactCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create an artifact.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ArtifactCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create an artifact.\"\"\"\n\n    key: Optional[str] = FieldFrom(schemas.core.Artifact)\n    type: Optional[str] = FieldFrom(schemas.core.Artifact)\n    description: Optional[str] = FieldFrom(schemas.core.Artifact)\n    data: Optional[Union[Dict[str, Any], Any]] = FieldFrom(schemas.core.Artifact)\n    metadata_: Optional[Dict[str, str]] = FieldFrom(schemas.core.Artifact)\n    flow_run_id: Optional[UUID] = FieldFrom(schemas.core.Artifact)\n    task_run_id: Optional[UUID] = FieldFrom(schemas.core.Artifact)\n\n    _validate_artifact_format = validator(\"key\", allow_reuse=True)(\n        validate_artifact_key\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ArtifactCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ArtifactUpdate","title":"ArtifactUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update an artifact.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ArtifactUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update an artifact.\"\"\"\n\n    data: Optional[Union[Dict[str, Any], Any]] = FieldFrom(schemas.core.Artifact)\n    description: Optional[str] = FieldFrom(schemas.core.Artifact)\n    metadata_: Optional[Dict[str, str]] = FieldFrom(schemas.core.Artifact)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ArtifactUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentCreate","title":"BlockDocumentCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block document.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block document.\"\"\"\n\n    name: Optional[str] = FieldFrom(schemas.core.BlockDocument)\n    data: dict = FieldFrom(schemas.core.BlockDocument)\n    block_schema_id: UUID = FieldFrom(schemas.core.BlockDocument)\n    block_type_id: UUID = FieldFrom(schemas.core.BlockDocument)\n    is_anonymous: bool = FieldFrom(schemas.core.BlockDocument)\n\n    _validate_name_format = validator(\"name\", allow_reuse=True)(\n        validate_block_document_name\n    )\n\n    @root_validator\n    def validate_name_is_present_if_not_anonymous(cls, values):\n        # TODO: We should find an elegant way to reuse this logic from the origin model\n        if not values.get(\"is_anonymous\") and not values.get(\"name\"):\n            raise ValueError(\"Names must be provided for block documents.\")\n        return values\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentReferenceCreate","title":"BlockDocumentReferenceCreate","text":"

    Bases: ActionBaseModel

    Data used to create block document reference.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentReferenceCreate(ActionBaseModel):\n    \"\"\"Data used to create block document reference.\"\"\"\n\n    id: UUID = FieldFrom(schemas.core.BlockDocumentReference)\n    parent_block_document_id: UUID = FieldFrom(schemas.core.BlockDocumentReference)\n    reference_block_document_id: UUID = FieldFrom(schemas.core.BlockDocumentReference)\n    name: str = FieldFrom(schemas.core.BlockDocumentReference)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentReferenceCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentUpdate","title":"BlockDocumentUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a block document.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockDocumentUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a block document.\"\"\"\n\n    block_schema_id: Optional[UUID] = Field(\n        default=None, description=\"A block schema ID\"\n    )\n    data: dict = FieldFrom(schemas.core.BlockDocument)\n    merge_existing_data: bool = True\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockDocumentUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockSchemaCreate","title":"BlockSchemaCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block schema.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockSchemaCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block schema.\"\"\"\n\n    fields: dict = FieldFrom(schemas.core.BlockSchema)\n    block_type_id: Optional[UUID] = FieldFrom(schemas.core.BlockSchema)\n    capabilities: List[str] = FieldFrom(schemas.core.BlockSchema)\n    version: str = FieldFrom(schemas.core.BlockSchema)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockSchemaCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockTypeCreate","title":"BlockTypeCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a block type.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockTypeCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a block type.\"\"\"\n\n    name: str = FieldFrom(schemas.core.BlockType)\n    slug: str = FieldFrom(schemas.core.BlockType)\n    logo_url: Optional[schemas.core.HttpUrl] = FieldFrom(schemas.core.BlockType)\n    documentation_url: Optional[schemas.core.HttpUrl] = FieldFrom(\n        schemas.core.BlockType\n    )\n    description: Optional[str] = FieldFrom(schemas.core.BlockType)\n    code_example: Optional[str] = FieldFrom(schemas.core.BlockType)\n\n    # validators\n    _validate_slug_format = validator(\"slug\", allow_reuse=True)(\n        validate_block_type_slug\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockTypeCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockTypeUpdate","title":"BlockTypeUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a block type.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass BlockTypeUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a block type.\"\"\"\n\n    logo_url: Optional[schemas.core.HttpUrl] = FieldFrom(schemas.core.BlockType)\n    documentation_url: Optional[schemas.core.HttpUrl] = FieldFrom(\n        schemas.core.BlockType\n    )\n    description: Optional[str] = FieldFrom(schemas.core.BlockType)\n    code_example: Optional[str] = FieldFrom(schemas.core.BlockType)\n\n    @classmethod\n    def updatable_fields(cls) -> set:\n        return get_class_fields_only(cls)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.BlockTypeUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitCreate","title":"ConcurrencyLimitCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a concurrency limit.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ConcurrencyLimitCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a concurrency limit.\"\"\"\n\n    tag: str = FieldFrom(schemas.core.ConcurrencyLimit)\n    concurrency_limit: int = FieldFrom(schemas.core.ConcurrencyLimit)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitV2Create","title":"ConcurrencyLimitV2Create","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a v2 concurrency limit.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ConcurrencyLimitV2Create(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a v2 concurrency limit.\"\"\"\n\n    active: bool = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    name: str = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    limit: int = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    active_slots: int = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    denied_slots: int = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    slot_decay_per_second: float = FieldFrom(schemas.core.ConcurrencyLimitV2)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitV2Create.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitV2Update","title":"ConcurrencyLimitV2Update","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a v2 concurrency limit.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass ConcurrencyLimitV2Update(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a v2 concurrency limit.\"\"\"\n\n    active: Optional[bool] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    name: Optional[str] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    limit: Optional[int] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    active_slots: Optional[int] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    denied_slots: Optional[int] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n    slot_decay_per_second: Optional[float] = FieldFrom(schemas.core.ConcurrencyLimitV2)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.ConcurrencyLimitV2Update.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentCreate","title":"DeploymentCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a deployment.

    Source code in prefect/server/schemas/actions.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None,\n)\n@copy_model_fields\nclass DeploymentCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a deployment.\"\"\"\n\n    @root_validator(pre=True)\n    def remove_old_fields(cls, values):\n        # 2.7.7 removed worker_pool_queue_id in lieu of worker_pool_name and\n        # worker_pool_queue_name. Those fields were later renamed to work_pool_name\n        # and work_queue_name. This validator removes old fields provided\n        # by older clients to avoid 422 errors.\n        values_copy = copy(values)\n        worker_pool_queue_id = values_copy.pop(\"worker_pool_queue_id\", None)\n        worker_pool_name = values_copy.pop(\"worker_pool_name\", None)\n        worker_pool_queue_name = values_copy.pop(\"worker_pool_queue_name\", None)\n        work_pool_queue_name = values_copy.pop(\"work_pool_queue_name\", None)\n        if worker_pool_queue_id:\n            warnings.warn(\n                (\n                    \"`worker_pool_queue_id` is no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        if worker_pool_name or worker_pool_queue_name or work_pool_queue_name:\n            warnings.warn(\n                (\n                    \"`worker_pool_name`, `worker_pool_queue_name`, and \"\n                    \"`work_pool_name` are\"\n                    \"no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        return values_copy\n\n    name: str = FieldFrom(schemas.core.Deployment)\n    flow_id: UUID = FieldFrom(schemas.core.Deployment)\n    is_schedule_active: Optional[bool] = FieldFrom(schemas.core.Deployment)\n    enforce_parameter_schema: bool = FieldFrom(schemas.core.Deployment)\n    parameter_openapi_schema: Optional[Dict[str, Any]] = FieldFrom(\n        schemas.core.Deployment\n    )\n    parameters: Dict[str, Any] = FieldFrom(schemas.core.Deployment)\n    tags: List[str] = FieldFrom(schemas.core.Deployment)\n    pull_steps: Optional[List[dict]] = FieldFrom(schemas.core.Deployment)\n\n    manifest_path: Optional[str] = FieldFrom(schemas.core.Deployment)\n    work_queue_name: Optional[str] = FieldFrom(schemas.core.Deployment)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the deployment's work pool.\",\n        example=\"my-work-pool\",\n    )\n    storage_document_id: Optional[UUID] = FieldFrom(schemas.core.Deployment)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.Deployment)\n    schedule: Optional[schemas.schedules.SCHEDULE_TYPES] = FieldFrom(\n        schemas.core.Deployment\n    )\n    description: Optional[str] = FieldFrom(schemas.core.Deployment)\n    path: Optional[str] = FieldFrom(schemas.core.Deployment)\n    version: Optional[str] = FieldFrom(schemas.core.Deployment)\n    entrypoint: Optional[str] = FieldFrom(schemas.core.Deployment)\n    infra_overrides: Optional[Dict[str, Any]] = FieldFrom(schemas.core.Deployment)\n\n    def check_valid_configuration(self, base_job_template: dict):\n        \"\"\"Check that the combination of base_job_template defaults\n        and infra_overrides conforms to the specified schema.\n        \"\"\"\n        variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n        if variables_schema is not None:\n            # jsonschema considers required fields, even if that field has a default,\n            # to still be required. To get around this we remove the fields from\n            # required if there is a default present.\n            required = variables_schema.get(\"required\")\n            properties = variables_schema.get(\"properties\")\n            if required is not None and properties is not None:\n                for k, v in properties.items():\n                    if \"default\" in v and k in required:\n                        required.remove(k)\n\n            jsonschema.validate(self.infra_overrides, variables_schema)\n\n    @validator(\"parameters\")\n    def _validate_parameters_conform_to_schema(cls, value, values):\n        \"\"\"Validate that the parameters conform to the parameter schema.\"\"\"\n        if values.get(\"enforce_parameter_schema\"):\n            validate_values_conform_to_schema(\n                value, values.get(\"parameter_openapi_schema\"), ignore_required=True\n            )\n        return value\n\n    @validator(\"parameter_openapi_schema\")\n    def _validate_parameter_openapi_schema(cls, value, values):\n        \"\"\"Validate that the parameter_openapi_schema is a valid json schema.\"\"\"\n        if values.get(\"enforce_parameter_schema\"):\n            validate_schema(value)\n        return value\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentCreate.check_valid_configuration","title":"check_valid_configuration","text":"

    Check that the combination of base_job_template defaults and infra_overrides conforms to the specified schema.

    Source code in prefect/server/schemas/actions.py
    def check_valid_configuration(self, base_job_template: dict):\n    \"\"\"Check that the combination of base_job_template defaults\n    and infra_overrides conforms to the specified schema.\n    \"\"\"\n    variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n    if variables_schema is not None:\n        # jsonschema considers required fields, even if that field has a default,\n        # to still be required. To get around this we remove the fields from\n        # required if there is a default present.\n        required = variables_schema.get(\"required\")\n        properties = variables_schema.get(\"properties\")\n        if required is not None and properties is not None:\n            for k, v in properties.items():\n                if \"default\" in v and k in required:\n                    required.remove(k)\n\n        jsonschema.validate(self.infra_overrides, variables_schema)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentFlowRunCreate","title":"DeploymentFlowRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run from a deployment.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass DeploymentFlowRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run from a deployment.\"\"\"\n\n    # FlowRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the flow run to create\"\n    )\n\n    name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parameters: dict = FieldFrom(schemas.core.FlowRun)\n    context: dict = FieldFrom(schemas.core.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    empirical_policy: schemas.core.FlowRunPolicy = FieldFrom(schemas.core.FlowRun)\n    tags: List[str] = FieldFrom(schemas.core.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    work_queue_name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentFlowRunCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentUpdate","title":"DeploymentUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a deployment.

    Source code in prefect/server/schemas/actions.py
    @experimental_field(\n    \"work_pool_name\",\n    group=\"work_pools\",\n    when=lambda x: x is not None,\n)\n@copy_model_fields\nclass DeploymentUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a deployment.\"\"\"\n\n    @root_validator(pre=True)\n    def remove_old_fields(cls, values):\n        # 2.7.7 removed worker_pool_queue_id in lieu of worker_pool_name and\n        # worker_pool_queue_name. Those fields were later renamed to work_pool_name\n        # and work_queue_name. This validator removes old fields provided\n        # by older clients to avoid 422 errors.\n        values_copy = copy(values)\n        worker_pool_queue_id = values_copy.pop(\"worker_pool_queue_id\", None)\n        worker_pool_name = values_copy.pop(\"worker_pool_name\", None)\n        worker_pool_queue_name = values_copy.pop(\"worker_pool_queue_name\", None)\n        work_pool_queue_name = values_copy.pop(\"work_pool_queue_name\", None)\n        if worker_pool_queue_id:\n            warnings.warn(\n                (\n                    \"`worker_pool_queue_id` is no longer supported for updating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        if worker_pool_name or worker_pool_queue_name or work_pool_queue_name:\n            warnings.warn(\n                (\n                    \"`worker_pool_name`, `worker_pool_queue_name`, and \"\n                    \"`work_pool_name` are\"\n                    \"no longer supported for creating \"\n                    \"deployments. Please use `work_pool_name` and \"\n                    \"`work_queue_name` instead.\"\n                ),\n                UserWarning,\n            )\n        return values_copy\n\n    version: Optional[str] = FieldFrom(schemas.core.Deployment)\n    schedule: Optional[schemas.schedules.SCHEDULE_TYPES] = FieldFrom(\n        schemas.core.Deployment\n    )\n    description: Optional[str] = FieldFrom(schemas.core.Deployment)\n    is_schedule_active: bool = FieldFrom(schemas.core.Deployment)\n    parameters: Optional[Dict[str, Any]] = Field(\n        default=None,\n        description=\"Parameters for flow runs scheduled by the deployment.\",\n    )\n    tags: List[str] = FieldFrom(schemas.core.Deployment)\n    work_queue_name: Optional[str] = FieldFrom(schemas.core.Deployment)\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the deployment's work pool.\",\n        example=\"my-work-pool\",\n    )\n    path: Optional[str] = FieldFrom(schemas.core.Deployment)\n    infra_overrides: Optional[Dict[str, Any]] = FieldFrom(schemas.core.Deployment)\n    entrypoint: Optional[str] = FieldFrom(schemas.core.Deployment)\n    manifest_path: Optional[str] = FieldFrom(schemas.core.Deployment)\n    storage_document_id: Optional[UUID] = FieldFrom(schemas.core.Deployment)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.Deployment)\n    enforce_parameter_schema: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n\n    def check_valid_configuration(self, base_job_template: dict):\n        \"\"\"Check that the combination of base_job_template defaults\n        and infra_overrides conforms to the specified schema.\n        \"\"\"\n        variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n        if variables_schema is not None:\n            # jsonschema considers required fields, even if that field has a default,\n            # to still be required. To get around this we remove the fields from\n            # required if there is a default present.\n            required = variables_schema.get(\"required\")\n            properties = variables_schema.get(\"properties\")\n            if required is not None and properties is not None:\n                for k, v in properties.items():\n                    if \"default\" in v and k in required:\n                        required.remove(k)\n\n        if variables_schema is not None:\n            jsonschema.validate(self.infra_overrides, variables_schema)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentUpdate.check_valid_configuration","title":"check_valid_configuration","text":"

    Check that the combination of base_job_template defaults and infra_overrides conforms to the specified schema.

    Source code in prefect/server/schemas/actions.py
    def check_valid_configuration(self, base_job_template: dict):\n    \"\"\"Check that the combination of base_job_template defaults\n    and infra_overrides conforms to the specified schema.\n    \"\"\"\n    variables_schema = deepcopy(base_job_template.get(\"variables\"))\n\n    if variables_schema is not None:\n        # jsonschema considers required fields, even if that field has a default,\n        # to still be required. To get around this we remove the fields from\n        # required if there is a default present.\n        required = variables_schema.get(\"required\")\n        properties = variables_schema.get(\"properties\")\n        if required is not None and properties is not None:\n            for k, v in properties.items():\n                if \"default\" in v and k in required:\n                    required.remove(k)\n\n    if variables_schema is not None:\n        jsonschema.validate(self.infra_overrides, variables_schema)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.DeploymentUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowCreate","title":"FlowCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow.\"\"\"\n\n    name: str = FieldFrom(schemas.core.Flow)\n    tags: List[str] = FieldFrom(schemas.core.Flow)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunCreate","title":"FlowRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run.\"\"\"\n\n    # FlowRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the flow run to create\"\n    )\n\n    name: str = FieldFrom(schemas.core.FlowRun)\n    flow_id: UUID = FieldFrom(schemas.core.FlowRun)\n    flow_version: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parameters: dict = FieldFrom(schemas.core.FlowRun)\n    context: dict = FieldFrom(schemas.core.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    empirical_policy: schemas.core.FlowRunPolicy = FieldFrom(schemas.core.FlowRun)\n    tags: List[str] = FieldFrom(schemas.core.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(schemas.core.FlowRun)\n\n    # DEPRECATED\n\n    deployment_id: Optional[UUID] = Field(\n        None,\n        description=(\n            \"DEPRECATED: The id of the deployment associated with this flow run, if\"\n            \" available.\"\n        ),\n        deprecated=True,\n    )\n\n    class Config(ActionBaseModel.Config):\n        json_dumps = orjson_dumps_extra_compatible\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunNotificationPolicyCreate","title":"FlowRunNotificationPolicyCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a flow run notification policy.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowRunNotificationPolicyCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a flow run notification policy.\"\"\"\n\n    is_active: bool = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    state_names: List[str] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    tags: List[str] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    block_document_id: UUID = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    message_template: Optional[str] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunNotificationPolicyCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunNotificationPolicyUpdate","title":"FlowRunNotificationPolicyUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow run notification policy.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowRunNotificationPolicyUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow run notification policy.\"\"\"\n\n    is_active: Optional[bool] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    state_names: Optional[List[str]] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    tags: Optional[List[str]] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n    block_document_id: Optional[UUID] = FieldFrom(\n        schemas.core.FlowRunNotificationPolicy\n    )\n    message_template: Optional[str] = FieldFrom(schemas.core.FlowRunNotificationPolicy)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunNotificationPolicyUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunUpdate","title":"FlowRunUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow run.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowRunUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow run.\"\"\"\n\n    name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    flow_version: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parameters: dict = FieldFrom(schemas.core.FlowRun)\n    empirical_policy: schemas.core.FlowRunPolicy = FieldFrom(schemas.core.FlowRun)\n    tags: List[str] = FieldFrom(schemas.core.FlowRun)\n    infrastructure_pid: Optional[str] = FieldFrom(schemas.core.FlowRun)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowRunUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowUpdate","title":"FlowUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a flow.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass FlowUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a flow.\"\"\"\n\n    tags: List[str] = FieldFrom(schemas.core.Flow)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.FlowUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.LogCreate","title":"LogCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a log.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass LogCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a log.\"\"\"\n\n    name: str = FieldFrom(schemas.core.Log)\n    level: int = FieldFrom(schemas.core.Log)\n    message: str = FieldFrom(schemas.core.Log)\n    timestamp: DateTimeTZ = FieldFrom(schemas.core.Log)\n    flow_run_id: Optional[UUID] = FieldFrom(schemas.core.Log)\n    task_run_id: Optional[UUID] = FieldFrom(schemas.core.Log)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.LogCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.SavedSearchCreate","title":"SavedSearchCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a saved search.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass SavedSearchCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a saved search.\"\"\"\n\n    name: str = FieldFrom(schemas.core.SavedSearch)\n    filters: List[schemas.core.SavedSearchFilter] = FieldFrom(schemas.core.SavedSearch)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.SavedSearchCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.StateCreate","title":"StateCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a new state.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass StateCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a new state.\"\"\"\n\n    type: schemas.states.StateType = FieldFrom(schemas.states.State)\n    name: Optional[str] = FieldFrom(schemas.states.State)\n    message: Optional[str] = FieldFrom(schemas.states.State)\n    data: Optional[Any] = FieldFrom(schemas.states.State)\n    state_details: schemas.states.StateDetails = FieldFrom(schemas.states.State)\n\n    # DEPRECATED\n\n    timestamp: Optional[DateTimeTZ] = Field(\n        default=None,\n        repr=False,\n        ignored=True,\n    )\n    id: Optional[UUID] = Field(default=None, repr=False, ignored=True)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.StateCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.TaskRunCreate","title":"TaskRunCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a task run

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass TaskRunCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a task run\"\"\"\n\n    # TaskRunCreate states must be provided as StateCreate objects\n    state: Optional[StateCreate] = Field(\n        default=None, description=\"The state of the task run to create\"\n    )\n\n    name: str = FieldFrom(schemas.core.TaskRun)\n    flow_run_id: Optional[UUID] = FieldFrom(schemas.core.TaskRun)\n    task_key: str = FieldFrom(schemas.core.TaskRun)\n    dynamic_key: str = FieldFrom(schemas.core.TaskRun)\n    cache_key: Optional[str] = FieldFrom(schemas.core.TaskRun)\n    cache_expiration: Optional[DateTimeTZ] = FieldFrom(schemas.core.TaskRun)\n    task_version: Optional[str] = FieldFrom(schemas.core.TaskRun)\n    empirical_policy: schemas.core.TaskRunPolicy = FieldFrom(schemas.core.TaskRun)\n    tags: List[str] = FieldFrom(schemas.core.TaskRun)\n    task_inputs: Dict[\n        str,\n        List[\n            Union[\n                schemas.core.TaskRunResult,\n                schemas.core.Parameter,\n                schemas.core.Constant,\n            ]\n        ],\n    ] = FieldFrom(schemas.core.TaskRun)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.TaskRunCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.TaskRunUpdate","title":"TaskRunUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a task run

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass TaskRunUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a task run\"\"\"\n\n    name: str = FieldFrom(schemas.core.TaskRun)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.TaskRunUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.VariableCreate","title":"VariableCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a Variable.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass VariableCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a Variable.\"\"\"\n\n    name: str = FieldFrom(schemas.core.Variable)\n    value: str = FieldFrom(schemas.core.Variable)\n    tags: Optional[List[str]] = FieldFrom(schemas.core.Variable)\n\n    # validators\n    _validate_name_format = validator(\"name\", allow_reuse=True)(validate_variable_name)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.VariableCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.VariableUpdate","title":"VariableUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a Variable.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass VariableUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a Variable.\"\"\"\n\n    name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the variable\",\n        example=\"my_variable\",\n        max_length=schemas.core.MAX_VARIABLE_NAME_LENGTH,\n    )\n    value: Optional[str] = Field(\n        default=None,\n        description=\"The value of the variable\",\n        example=\"my-value\",\n        max_length=schemas.core.MAX_VARIABLE_VALUE_LENGTH,\n    )\n    tags: Optional[List[str]] = FieldFrom(schemas.core.Variable)\n\n    # validators\n    _validate_name_format = validator(\"name\", allow_reuse=True)(validate_variable_name)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.VariableUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkPoolCreate","title":"WorkPoolCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a work pool.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass WorkPoolCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a work pool.\"\"\"\n\n    name: str = FieldFrom(schemas.core.WorkPool)\n    description: Optional[str] = FieldFrom(schemas.core.WorkPool)\n    type: str = Field(description=\"The work pool type.\", default=\"prefect-agent\")\n    base_job_template: Dict[str, Any] = FieldFrom(schemas.core.WorkPool)\n    is_paused: bool = FieldFrom(schemas.core.WorkPool)\n    concurrency_limit: Optional[int] = FieldFrom(schemas.core.WorkPool)\n\n    _validate_base_job_template = validator(\"base_job_template\", allow_reuse=True)(\n        validate_base_job_template\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkPoolCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkPoolUpdate","title":"WorkPoolUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a work pool.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass WorkPoolUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a work pool.\"\"\"\n\n    description: Optional[str] = FieldFrom(schemas.core.WorkPool)\n    is_paused: Optional[bool] = FieldFrom(schemas.core.WorkPool)\n    base_job_template: Optional[Dict[str, Any]] = FieldFrom(schemas.core.WorkPool)\n    concurrency_limit: Optional[int] = FieldFrom(schemas.core.WorkPool)\n\n    _validate_base_job_template = validator(\"base_job_template\", allow_reuse=True)(\n        validate_base_job_template\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkPoolUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkQueueCreate","title":"WorkQueueCreate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to create a work queue.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass WorkQueueCreate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to create a work queue.\"\"\"\n\n    name: str = FieldFrom(schemas.core.WorkQueue)\n    description: Optional[str] = FieldFrom(schemas.core.WorkQueue)\n    is_paused: bool = FieldFrom(schemas.core.WorkQueue)\n    concurrency_limit: Optional[int] = FieldFrom(schemas.core.WorkQueue)\n    priority: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The queue's priority. Lower values are higher priority (1 is the highest).\"\n        ),\n    )\n\n    # DEPRECATED\n\n    filter: Optional[schemas.core.QueueFilter] = Field(\n        None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkQueueCreate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkQueueUpdate","title":"WorkQueueUpdate","text":"

    Bases: ActionBaseModel

    Data used by the Prefect REST API to update a work queue.

    Source code in prefect/server/schemas/actions.py
    @copy_model_fields\nclass WorkQueueUpdate(ActionBaseModel):\n    \"\"\"Data used by the Prefect REST API to update a work queue.\"\"\"\n\n    name: str = FieldFrom(schemas.core.WorkQueue)\n    description: Optional[str] = FieldFrom(schemas.core.WorkQueue)\n    is_paused: bool = FieldFrom(schemas.core.WorkQueue)\n    concurrency_limit: Optional[int] = FieldFrom(schemas.core.WorkQueue)\n    priority: Optional[int] = FieldFrom(schemas.core.WorkQueue)\n    last_polled: Optional[DateTimeTZ] = FieldFrom(schemas.core.WorkQueue)\n\n    # DEPRECATED\n\n    filter: Optional[schemas.core.QueueFilter] = Field(\n        None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n
    "},{"location":"api-ref/server/schemas/actions/#prefect.server.schemas.actions.WorkQueueUpdate.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/core/","title":"server.schemas.core","text":""},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core","title":"prefect.server.schemas.core","text":"

    Full schemas of Prefect REST API objects.

    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Agent","title":"Agent","text":"

    Bases: ORMBaseModel

    An ORM representation of an agent

    Source code in prefect/server/schemas/core.py
    class Agent(ORMBaseModel):\n    \"\"\"An ORM representation of an agent\"\"\"\n\n    name: str = Field(\n        default_factory=lambda: generate_slug(2),\n        description=(\n            \"The name of the agent. If a name is not provided, it will be\"\n            \" auto-generated.\"\n        ),\n    )\n    work_queue_id: UUID = Field(\n        default=..., description=\"The work queue with which the agent is associated.\"\n    )\n    last_activity_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The last time this agent polled for work.\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockDocument","title":"BlockDocument","text":"

    Bases: ORMBaseModel

    An ORM representation of a block document.

    Source code in prefect/server/schemas/core.py
    class BlockDocument(ORMBaseModel):\n    \"\"\"An ORM representation of a block document.\"\"\"\n\n    name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The block document's name. Not required for anonymous block documents.\"\n        ),\n    )\n    data: dict = Field(default_factory=dict, description=\"The block document's data\")\n    block_schema_id: UUID = Field(default=..., description=\"A block schema ID\")\n    block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The associated block schema\"\n    )\n    block_type_id: UUID = Field(default=..., description=\"A block type ID\")\n    block_type_name: Optional[str] = Field(\n        default=None, description=\"The associated block type's name\"\n    )\n    block_type: Optional[BlockType] = Field(\n        default=None, description=\"The associated block type\"\n    )\n    block_document_references: Dict[str, Dict[str, Any]] = Field(\n        default_factory=dict, description=\"Record of the block document's references\"\n    )\n    is_anonymous: bool = Field(\n        default=False,\n        description=(\n            \"Whether the block is anonymous (anonymous blocks are usually created by\"\n            \" Prefect automatically)\"\n        ),\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        # the BlockDocumentCreate subclass allows name=None\n        # and will inherit this validator\n        if v is not None:\n            raise_on_name_with_banned_characters(v)\n        return v\n\n    @root_validator\n    def validate_name_is_present_if_not_anonymous(cls, values):\n        # anonymous blocks may have no name prior to actually being\n        # stored in the database\n        if not values.get(\"is_anonymous\") and not values.get(\"name\"):\n            raise ValueError(\"Names must be provided for block documents.\")\n        return values\n\n    @classmethod\n    async def from_orm_model(\n        cls,\n        session,\n        orm_block_document: \"prefect.server.database.orm_models.ORMBlockDocument\",\n        include_secrets: bool = False,\n    ):\n        data = await orm_block_document.decrypt_data(session=session)\n        # if secrets are not included, obfuscate them based on the schema's\n        # `secret_fields`. Note this walks any nested blocks as well. If the\n        # nested blocks were recovered from named blocks, they will already\n        # be obfuscated, but if nested fields were hardcoded into the parent\n        # blocks data, this is the only opportunity to obfuscate them.\n        if not include_secrets:\n            flat_data = dict_to_flatdict(data)\n            # iterate over the (possibly nested) secret fields\n            # and obfuscate their data\n            for secret_field in orm_block_document.block_schema.fields.get(\n                \"secret_fields\", []\n            ):\n                secret_key = tuple(secret_field.split(\".\"))\n                if flat_data.get(secret_key) is not None:\n                    flat_data[secret_key] = obfuscate_string(flat_data[secret_key])\n                # If a wildcard (*) is in the current secret key path, we take the portion\n                # of the path before the wildcard and compare it to the same level of each\n                # key. A match means that the field is nested under the secret key and should\n                # be obfuscated.\n                elif \"*\" in secret_key:\n                    wildcard_index = secret_key.index(\"*\")\n                    for data_key in flat_data.keys():\n                        if secret_key[0:wildcard_index] == data_key[0:wildcard_index]:\n                            flat_data[data_key] = obfuscate(flat_data[data_key])\n            data = flatdict_to_dict(flat_data)\n\n        return cls(\n            id=orm_block_document.id,\n            created=orm_block_document.created,\n            updated=orm_block_document.updated,\n            name=orm_block_document.name,\n            data=data,\n            block_schema_id=orm_block_document.block_schema_id,\n            block_schema=orm_block_document.block_schema,\n            block_type_id=orm_block_document.block_type_id,\n            block_type_name=orm_block_document.block_type_name,\n            block_type=orm_block_document.block_type,\n            is_anonymous=orm_block_document.is_anonymous,\n        )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockDocumentReference","title":"BlockDocumentReference","text":"

    Bases: ORMBaseModel

    An ORM representation of a block document reference.

    Source code in prefect/server/schemas/core.py
    class BlockDocumentReference(ORMBaseModel):\n    \"\"\"An ORM representation of a block document reference.\"\"\"\n\n    parent_block_document_id: UUID = Field(\n        default=..., description=\"ID of block document the reference is nested within\"\n    )\n    parent_block_document: Optional[BlockDocument] = Field(\n        default=None, description=\"The block document the reference is nested within\"\n    )\n    reference_block_document_id: UUID = Field(\n        default=..., description=\"ID of the nested block document\"\n    )\n    reference_block_document: Optional[BlockDocument] = Field(\n        default=None, description=\"The nested block document\"\n    )\n    name: str = Field(\n        default=..., description=\"The name that the reference is nested under\"\n    )\n\n    @root_validator\n    def validate_parent_and_ref_are_different(cls, values):\n        parent_id = values.get(\"parent_block_document_id\")\n        ref_id = values.get(\"reference_block_document_id\")\n        if parent_id and ref_id and parent_id == ref_id:\n            raise ValueError(\n                \"`parent_block_document_id` and `reference_block_document_id` cannot be\"\n                \" the same\"\n            )\n        return values\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockSchema","title":"BlockSchema","text":"

    Bases: ORMBaseModel

    An ORM representation of a block schema.

    Source code in prefect/server/schemas/core.py
    class BlockSchema(ORMBaseModel):\n    \"\"\"An ORM representation of a block schema.\"\"\"\n\n    checksum: str = Field(default=..., description=\"The block schema's unique checksum\")\n    fields: dict = Field(\n        default_factory=dict, description=\"The block schema's field schema\"\n    )\n    block_type_id: Optional[UUID] = Field(default=..., description=\"A block type ID\")\n    block_type: Optional[BlockType] = Field(\n        default=None, description=\"The associated block type\"\n    )\n    capabilities: List[str] = Field(\n        default_factory=list,\n        description=\"A list of Block capabilities\",\n    )\n    version: str = Field(\n        default=DEFAULT_BLOCK_SCHEMA_VERSION,\n        description=\"Human readable identifier for the block schema\",\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockSchemaReference","title":"BlockSchemaReference","text":"

    Bases: ORMBaseModel

    An ORM representation of a block schema reference.

    Source code in prefect/server/schemas/core.py
    class BlockSchemaReference(ORMBaseModel):\n    \"\"\"An ORM representation of a block schema reference.\"\"\"\n\n    parent_block_schema_id: UUID = Field(\n        default=..., description=\"ID of block schema the reference is nested within\"\n    )\n    parent_block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The block schema the reference is nested within\"\n    )\n    reference_block_schema_id: UUID = Field(\n        default=..., description=\"ID of the nested block schema\"\n    )\n    reference_block_schema: Optional[BlockSchema] = Field(\n        default=None, description=\"The nested block schema\"\n    )\n    name: str = Field(\n        default=..., description=\"The name that the reference is nested under\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.BlockType","title":"BlockType","text":"

    Bases: ORMBaseModel

    An ORM representation of a block type

    Source code in prefect/server/schemas/core.py
    class BlockType(ORMBaseModel):\n    \"\"\"An ORM representation of a block type\"\"\"\n\n    name: str = Field(default=..., description=\"A block type's name\")\n    slug: str = Field(default=..., description=\"A block type's slug\")\n    logo_url: Optional[HttpUrl] = Field(\n        default=None, description=\"Web URL for the block type's logo\"\n    )\n    documentation_url: Optional[HttpUrl] = Field(\n        default=None, description=\"Web URL for the block type's documentation\"\n    )\n    description: Optional[str] = Field(\n        default=None,\n        description=\"A short blurb about the corresponding block's intended use\",\n    )\n    code_example: Optional[str] = Field(\n        default=None,\n        description=\"A code snippet demonstrating use of the corresponding block\",\n    )\n    is_protected: bool = Field(\n        default=False, description=\"Protected block types cannot be modified via API.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.ConcurrencyLimit","title":"ConcurrencyLimit","text":"

    Bases: ORMBaseModel

    An ORM representation of a concurrency limit.

    Source code in prefect/server/schemas/core.py
    class ConcurrencyLimit(ORMBaseModel):\n    \"\"\"An ORM representation of a concurrency limit.\"\"\"\n\n    tag: str = Field(\n        default=..., description=\"A tag the concurrency limit is applied to.\"\n    )\n    concurrency_limit: int = Field(default=..., description=\"The concurrency limit.\")\n    active_slots: List[UUID] = Field(\n        default_factory=list,\n        description=\"A list of active run ids using a concurrency slot\",\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.ConcurrencyLimitV2","title":"ConcurrencyLimitV2","text":"

    Bases: ORMBaseModel

    An ORM representation of a v2 concurrency limit.

    Source code in prefect/server/schemas/core.py
    class ConcurrencyLimitV2(ORMBaseModel):\n    \"\"\"An ORM representation of a v2 concurrency limit.\"\"\"\n\n    active: bool = Field(\n        default=True, description=\"Whether the concurrency limit is active.\"\n    )\n    name: str = Field(default=..., description=\"The name of the concurrency limit.\")\n    limit: int = Field(default=..., description=\"The concurrency limit.\")\n    active_slots: int = Field(default=0, description=\"The number of active slots.\")\n    denied_slots: int = Field(default=0, description=\"The number of denied slots.\")\n    slot_decay_per_second: float = Field(\n        default=0,\n        description=\"The decay rate for active slots when used as a rate limit.\",\n    )\n    avg_slot_occupancy_seconds: float = Field(\n        default=2.0, description=\"The average amount of time a slot is occupied.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Configuration","title":"Configuration","text":"

    Bases: ORMBaseModel

    An ORM representation of account info.

    Source code in prefect/server/schemas/core.py
    class Configuration(ORMBaseModel):\n    \"\"\"An ORM representation of account info.\"\"\"\n\n    key: str = Field(default=..., description=\"Account info key\")\n    value: dict = Field(default=..., description=\"Account info\")\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Deployment","title":"Deployment","text":"

    Bases: ORMBaseModel

    An ORM representation of deployment data.

    Source code in prefect/server/schemas/core.py
    class Deployment(ORMBaseModel):\n    \"\"\"An ORM representation of deployment data.\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the deployment.\")\n    version: Optional[str] = Field(\n        default=None, description=\"An optional version for the deployment.\"\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"A description for the deployment.\"\n    )\n    flow_id: UUID = Field(\n        default=..., description=\"The flow id associated with the deployment.\"\n    )\n    schedule: Optional[schedules.SCHEDULE_TYPES] = Field(\n        default=None, description=\"A schedule for the deployment.\"\n    )\n    is_schedule_active: bool = Field(\n        default=True, description=\"Whether or not the deployment schedule is active.\"\n    )\n    infra_overrides: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Overrides to apply to the base infrastructure block at runtime.\",\n    )\n    parameters: Dict[str, Any] = Field(\n        default_factory=dict,\n        description=\"Parameters for flow runs scheduled by the deployment.\",\n    )\n    pull_steps: Optional[List[dict]] = Field(\n        default=None,\n        description=\"Pull steps for cloning and running this deployment.\",\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags for the deployment\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The work queue for the deployment. If no work queue is set, work will not\"\n            \" be scheduled.\"\n        ),\n    )\n    last_polled: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The last time the deployment was polled for status updates.\",\n    )\n    parameter_openapi_schema: Optional[Dict[str, Any]] = Field(\n        default=None,\n        description=\"The parameter schema of the flow, including defaults.\",\n    )\n    path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the working directory for the workflow, relative to remote\"\n            \" storage or an absolute path.\"\n        ),\n    )\n    entrypoint: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the entrypoint for the workflow, relative to the `path`.\"\n        ),\n    )\n    manifest_path: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The path to the flow's manifest file, relative to the chosen storage.\"\n        ),\n    )\n    storage_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining storage used for this flow.\",\n    )\n    infrastructure_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining infrastructure to use for flow runs.\",\n    )\n    created_by: Optional[CreatedBy] = Field(\n        default=None,\n        description=\"Optional information about the creator of this deployment.\",\n    )\n    updated_by: Optional[UpdatedBy] = Field(\n        default=None,\n        description=\"Optional information about the updater of this deployment.\",\n    )\n    work_queue_id: UUID = Field(\n        default=None,\n        description=(\n            \"The id of the work pool queue to which this deployment is assigned.\"\n        ),\n    )\n    enforce_parameter_schema: bool = Field(\n        default=False,\n        description=(\n            \"Whether or not the deployment should enforce the parameter schema.\"\n        ),\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Flow","title":"Flow","text":"

    Bases: ORMBaseModel

    An ORM representation of flow data.

    Source code in prefect/server/schemas/core.py
    class Flow(ORMBaseModel):\n    \"\"\"An ORM representation of flow data.\"\"\"\n\n    name: str = Field(\n        default=..., description=\"The name of the flow\", example=\"my-flow\"\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of flow tags\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRun","title":"FlowRun","text":"

    Bases: ORMBaseModel

    An ORM representation of flow run data.

    Source code in prefect/server/schemas/core.py
    class FlowRun(ORMBaseModel):\n    \"\"\"An ORM representation of flow run data.\"\"\"\n\n    name: str = Field(\n        default_factory=lambda: generate_slug(2),\n        description=(\n            \"The name of the flow run. Defaults to a random slug if not specified.\"\n        ),\n        example=\"my-flow-run\",\n    )\n    flow_id: UUID = Field(default=..., description=\"The id of the flow being run.\")\n    state_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the flow run's current state.\"\n    )\n    deployment_id: Optional[UUID] = Field(\n        default=None,\n        description=(\n            \"The id of the deployment associated with this flow run, if available.\"\n        ),\n    )\n    work_queue_name: Optional[str] = Field(\n        default=None, description=\"The work queue that handled this flow run.\"\n    )\n    flow_version: Optional[str] = Field(\n        default=None,\n        description=\"The version of the flow executed in this flow run.\",\n        example=\"1.0\",\n    )\n    parameters: dict = Field(\n        default_factory=dict, description=\"Parameters for the flow run.\"\n    )\n    idempotency_key: Optional[str] = Field(\n        default=None,\n        description=(\n            \"An optional idempotency key for the flow run. Used to ensure the same flow\"\n            \" run is not created multiple times.\"\n        ),\n    )\n    context: dict = Field(\n        default_factory=dict,\n        description=\"Additional context for the flow run.\",\n        example={\"my_var\": \"my_val\"},\n    )\n    empirical_policy: FlowRunPolicy = Field(\n        default_factory=FlowRunPolicy,\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags on the flow run\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    parent_task_run_id: Optional[UUID] = Field(\n        default=None,\n        description=(\n            \"If the flow run is a subflow, the id of the 'dummy' task in the parent\"\n            \" flow used to track subflow state.\"\n        ),\n    )\n\n    state_type: Optional[states.StateType] = Field(\n        default=None, description=\"The type of the current flow run state.\"\n    )\n    state_name: Optional[str] = Field(\n        default=None, description=\"The name of the current flow run state.\"\n    )\n    run_count: int = Field(\n        default=0, description=\"The number of times the flow run was executed.\"\n    )\n    expected_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The flow run's expected start time.\",\n    )\n    next_scheduled_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The next time the flow run is scheduled to start.\",\n    )\n    start_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual start time.\"\n    )\n    end_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual end time.\"\n    )\n    total_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=(\n            \"Total run time. If the flow run was executed multiple times, the time of\"\n            \" each run will be summed.\"\n        ),\n    )\n    estimated_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"A real-time estimate of the total run time.\",\n    )\n    estimated_start_time_delta: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"The difference between actual and expected start time.\",\n    )\n    auto_scheduled: bool = Field(\n        default=False,\n        description=\"Whether or not the flow run was automatically scheduled.\",\n    )\n    infrastructure_document_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The block document defining infrastructure to use this flow run.\",\n    )\n    infrastructure_pid: Optional[str] = Field(\n        default=None,\n        description=\"The id of the flow run as returned by an infrastructure block.\",\n    )\n    created_by: Optional[CreatedBy] = Field(\n        default=None,\n        description=\"Optional information about the creator of this flow run.\",\n    )\n    work_queue_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the run's work pool queue.\"\n    )\n\n    # relationships\n    # flow: Flow = None\n    # task_runs: List[\"TaskRun\"] = Field(default_factory=list)\n    state: Optional[states.State] = Field(\n        default=None, description=\"The current state of the flow run.\"\n    )\n    # parent_task_run: \"TaskRun\" = None\n\n    @validator(\"name\", pre=True)\n    def set_name(cls, name):\n        return name or generate_slug(2)\n\n    def __eq__(self, other: Any) -> bool:\n        \"\"\"\n        Check for \"equality\" to another flow run schema\n\n        Estimates times are rolling and will always change with repeated queries for\n        a flow run so we ignore them during equality checks.\n        \"\"\"\n        if isinstance(other, FlowRun):\n            exclude_fields = {\"estimated_run_time\", \"estimated_start_time_delta\"}\n            return self.dict(exclude=exclude_fields) == other.dict(\n                exclude=exclude_fields\n            )\n        return super().__eq__(other)\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRunNotificationPolicy","title":"FlowRunNotificationPolicy","text":"

    Bases: ORMBaseModel

    An ORM representation of a flow run notification.

    Source code in prefect/server/schemas/core.py
    class FlowRunNotificationPolicy(ORMBaseModel):\n    \"\"\"An ORM representation of a flow run notification.\"\"\"\n\n    is_active: bool = Field(\n        default=True, description=\"Whether the policy is currently active\"\n    )\n    state_names: List[str] = Field(\n        default=..., description=\"The flow run states that trigger notifications\"\n    )\n    tags: List[str] = Field(\n        default=...,\n        description=\"The flow run tags that trigger notifications (set [] to disable)\",\n    )\n    block_document_id: UUID = Field(\n        default=..., description=\"The block document ID used for sending notifications\"\n    )\n    message_template: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A templatable notification message. Use {braces} to add variables.\"\n            \" Valid variables include:\"\n            f\" {listrepr(sorted(FLOW_RUN_NOTIFICATION_TEMPLATE_KWARGS), sep=', ')}\"\n        ),\n        example=(\n            \"Flow run {flow_run_name} with id {flow_run_id} entered state\"\n            \" {flow_run_state_name}.\"\n        ),\n    )\n\n    @validator(\"message_template\")\n    def validate_message_template_variables(cls, v):\n        if v is not None:\n            try:\n                v.format(**{k: \"test\" for k in FLOW_RUN_NOTIFICATION_TEMPLATE_KWARGS})\n            except KeyError as exc:\n                raise ValueError(f\"Invalid template variable provided: '{exc.args[0]}'\")\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRunPolicy","title":"FlowRunPolicy","text":"

    Bases: PrefectBaseModel

    Defines of how a flow run should retry.

    Source code in prefect/server/schemas/core.py
    class FlowRunPolicy(PrefectBaseModel):\n    \"\"\"Defines of how a flow run should retry.\"\"\"\n\n    # TODO: Determine how to separate between infrastructure and within-process level\n    #       retries\n    max_retries: int = Field(\n        default=0,\n        description=(\n            \"The maximum number of retries. Field is not used. Please use `retries`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retry_delay_seconds: float = Field(\n        default=0,\n        description=(\n            \"The delay between retries. Field is not used. Please use `retry_delay`\"\n            \" instead.\"\n        ),\n        deprecated=True,\n    )\n    retries: Optional[int] = Field(default=None, description=\"The number of retries.\")\n    retry_delay: Optional[int] = Field(\n        default=None, description=\"The delay time between retries, in seconds.\"\n    )\n    pause_keys: Optional[set] = Field(\n        default_factory=set, description=\"Tracks pauses this run has observed.\"\n    )\n    resuming: Optional[bool] = Field(\n        default=False, description=\"Indicates if this run is resuming from a pause.\"\n    )\n\n    @root_validator\n    def populate_deprecated_fields(cls, values):\n        \"\"\"\n        If deprecated fields are provided, populate the corresponding new fields\n        to preserve orchestration behavior.\n        \"\"\"\n        if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n            values[\"retries\"] = values[\"max_retries\"]\n        if (\n            not values.get(\"retry_delay\", None)\n            and values.get(\"retry_delay_seconds\", 0) != 0\n        ):\n            values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n        return values\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRunPolicy.populate_deprecated_fields","title":"populate_deprecated_fields","text":"

    If deprecated fields are provided, populate the corresponding new fields to preserve orchestration behavior.

    Source code in prefect/server/schemas/core.py
    @root_validator\ndef populate_deprecated_fields(cls, values):\n    \"\"\"\n    If deprecated fields are provided, populate the corresponding new fields\n    to preserve orchestration behavior.\n    \"\"\"\n    if not values.get(\"retries\", None) and values.get(\"max_retries\", 0) != 0:\n        values[\"retries\"] = values[\"max_retries\"]\n    if (\n        not values.get(\"retry_delay\", None)\n        and values.get(\"retry_delay_seconds\", 0) != 0\n    ):\n        values[\"retry_delay\"] = values[\"retry_delay_seconds\"]\n    return values\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.FlowRunnerSettings","title":"FlowRunnerSettings","text":"

    Bases: PrefectBaseModel

    An API schema for passing details about the flow runner.

    This schema is agnostic to the types and configuration provided by clients

    Source code in prefect/server/schemas/core.py
    class FlowRunnerSettings(PrefectBaseModel):\n    \"\"\"\n    An API schema for passing details about the flow runner.\n\n    This schema is agnostic to the types and configuration provided by clients\n    \"\"\"\n\n    type: Optional[str] = Field(\n        default=None,\n        description=(\n            \"The type of the flow runner which can be used by the client for\"\n            \" dispatching.\"\n        ),\n    )\n    config: Optional[dict] = Field(\n        default=None, description=\"The configuration for the given flow runner type.\"\n    )\n\n    # The following is required for composite compatibility in the ORM\n\n    def __init__(self, type: str = None, config: dict = None, **kwargs) -> None:\n        # Pydantic does not support positional arguments so they must be converted to\n        # keyword arguments\n        super().__init__(type=type, config=config, **kwargs)\n\n    def __composite_values__(self):\n        return self.type, self.config\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Log","title":"Log","text":"

    Bases: ORMBaseModel

    An ORM representation of log data.

    Source code in prefect/server/schemas/core.py
    class Log(ORMBaseModel):\n    \"\"\"An ORM representation of log data.\"\"\"\n\n    name: str = Field(default=..., description=\"The logger name.\")\n    level: int = Field(default=..., description=\"The log level.\")\n    message: str = Field(default=..., description=\"The log message.\")\n    timestamp: DateTimeTZ = Field(default=..., description=\"The log timestamp.\")\n    flow_run_id: Optional[UUID] = Field(\n        default=None, description=\"The flow run ID associated with the log.\"\n    )\n    task_run_id: Optional[UUID] = Field(\n        default=None, description=\"The task run ID associated with the log.\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.QueueFilter","title":"QueueFilter","text":"

    Bases: PrefectBaseModel

    Filter criteria definition for a work queue.

    Source code in prefect/server/schemas/core.py
    class QueueFilter(PrefectBaseModel):\n    \"\"\"Filter criteria definition for a work queue.\"\"\"\n\n    tags: Optional[List[str]] = Field(\n        default=None,\n        description=\"Only include flow runs with these tags in the work queue.\",\n    )\n    deployment_ids: Optional[List[UUID]] = Field(\n        default=None,\n        description=\"Only include flow runs from these deployments in the work queue.\",\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.SavedSearch","title":"SavedSearch","text":"

    Bases: ORMBaseModel

    An ORM representation of saved search data. Represents a set of filter criteria.

    Source code in prefect/server/schemas/core.py
    class SavedSearch(ORMBaseModel):\n    \"\"\"An ORM representation of saved search data. Represents a set of filter criteria.\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the saved search.\")\n    filters: List[SavedSearchFilter] = Field(\n        default_factory=list, description=\"The filter set for the saved search.\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.SavedSearchFilter","title":"SavedSearchFilter","text":"

    Bases: PrefectBaseModel

    A filter for a saved search model. Intended for use by the Prefect UI.

    Source code in prefect/server/schemas/core.py
    class SavedSearchFilter(PrefectBaseModel):\n    \"\"\"A filter for a saved search model. Intended for use by the Prefect UI.\"\"\"\n\n    object: str = Field(default=..., description=\"The object over which to filter.\")\n    property: str = Field(\n        default=..., description=\"The property of the object on which to filter.\"\n    )\n    type: str = Field(default=..., description=\"The type of the property.\")\n    operation: str = Field(\n        default=...,\n        description=\"The operator to apply to the object. For example, `equals`.\",\n    )\n    value: Any = Field(\n        default=..., description=\"A JSON-compatible value for the filter.\"\n    )\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.TaskRun","title":"TaskRun","text":"

    Bases: ORMBaseModel

    An ORM representation of task run data.

    Source code in prefect/server/schemas/core.py
    class TaskRun(ORMBaseModel):\n    \"\"\"An ORM representation of task run data.\"\"\"\n\n    name: str = Field(default_factory=lambda: generate_slug(2), example=\"my-task-run\")\n    flow_run_id: Optional[UUID] = Field(\n        default=None, description=\"The flow run id of the task run.\"\n    )\n    task_key: str = Field(\n        default=..., description=\"A unique identifier for the task being run.\"\n    )\n    dynamic_key: str = Field(\n        default=...,\n        description=(\n            \"A dynamic key used to differentiate between multiple runs of the same task\"\n            \" within the same flow run.\"\n        ),\n    )\n    cache_key: Optional[str] = Field(\n        default=None,\n        description=(\n            \"An optional cache key. If a COMPLETED state associated with this cache key\"\n            \" is found, the cached COMPLETED state will be used instead of executing\"\n            \" the task run.\"\n        ),\n    )\n    cache_expiration: Optional[DateTimeTZ] = Field(\n        default=None, description=\"Specifies when the cached state should expire.\"\n    )\n    task_version: Optional[str] = Field(\n        default=None, description=\"The version of the task being run.\"\n    )\n    empirical_policy: TaskRunPolicy = Field(\n        default_factory=TaskRunPolicy,\n    )\n    tags: List[str] = Field(\n        default_factory=list,\n        description=\"A list of tags for the task run.\",\n        example=[\"tag-1\", \"tag-2\"],\n    )\n    state_id: Optional[UUID] = Field(\n        default=None, description=\"The id of the current task run state.\"\n    )\n    task_inputs: Dict[str, List[Union[TaskRunResult, Parameter, Constant]]] = Field(\n        default_factory=dict,\n        description=(\n            \"Tracks the source of inputs to a task run. Used for internal bookkeeping.\"\n        ),\n    )\n    state_type: Optional[states.StateType] = Field(\n        default=None, description=\"The type of the current task run state.\"\n    )\n    state_name: Optional[str] = Field(\n        default=None, description=\"The name of the current task run state.\"\n    )\n    run_count: int = Field(\n        default=0, description=\"The number of times the task run has been executed.\"\n    )\n    flow_run_run_count: int = Field(\n        default=0,\n        description=(\n            \"If the parent flow has retried, this indicates the flow retry this run is\"\n            \" associated with.\"\n        ),\n    )\n    expected_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The task run's expected start time.\",\n    )\n\n    # the next scheduled start time will be populated\n    # whenever the run is in a scheduled state\n    next_scheduled_start_time: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"The next time the task run is scheduled to start.\",\n    )\n    start_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual start time.\"\n    )\n    end_time: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The actual end time.\"\n    )\n    total_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=(\n            \"Total run time. If the task run was executed multiple times, the time of\"\n            \" each run will be summed.\"\n        ),\n    )\n    estimated_run_time: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"A real-time estimate of total run time.\",\n    )\n    estimated_start_time_delta: datetime.timedelta = Field(\n        default=datetime.timedelta(0),\n        description=\"The difference between actual and expected start time.\",\n    )\n\n    # relationships\n    # flow_run: FlowRun = None\n    # subflow_runs: List[FlowRun] = Field(default_factory=list)\n    state: Optional[states.State] = Field(\n        default=None, description=\"The current task run state.\"\n    )\n\n    @validator(\"name\", pre=True)\n    def set_name(cls, name):\n        return name or generate_slug(2)\n\n    @validator(\"cache_key\")\n    def validate_cache_key_length(cls, cache_key):\n        if cache_key and len(cache_key) > PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH.value():\n            raise ValueError(\n                \"Cache key exceeded maximum allowed length of\"\n                f\" {PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH.value()} characters.\"\n            )\n        return cache_key\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkPool","title":"WorkPool","text":"

    Bases: ORMBaseModel

    An ORM representation of a work pool

    Source code in prefect/server/schemas/core.py
    class WorkPool(ORMBaseModel):\n    \"\"\"An ORM representation of a work pool\"\"\"\n\n    name: str = Field(\n        description=\"The name of the work pool.\",\n    )\n    description: Optional[str] = Field(\n        default=None, description=\"A description of the work pool.\"\n    )\n    type: str = Field(description=\"The work pool type.\")\n    base_job_template: Dict[str, Any] = Field(\n        default_factory=dict, description=\"The work pool's base job template.\"\n    )\n    is_paused: bool = Field(\n        default=False,\n        description=\"Pausing the work pool stops the delivery of all work.\",\n    )\n    concurrency_limit: Optional[conint(ge=0)] = Field(\n        default=None, description=\"A concurrency limit for the work pool.\"\n    )\n    # this required field has a default of None so that the custom validator\n    # below will be called and produce a more helpful error message\n    default_queue_id: UUID = Field(\n        None, description=\"The id of the pool's default queue.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n\n    @validator(\"default_queue_id\", always=True)\n    def helpful_error_for_missing_default_queue_id(cls, v):\n        \"\"\"\n        Default queue ID is required because all pools must have a default queue\n        ID, but it represents a circular foreign key relationship to a\n        WorkQueue (which can't be created until the work pool exists).\n        Therefore, while this field can *technically* be null, it shouldn't be.\n        This should only be an issue when creating new pools, as reading\n        existing ones will always have this field populated. This custom error\n        message will help users understand that they should use the\n        `actions.WorkPoolCreate` model in that case.\n        \"\"\"\n        if v is None:\n            raise ValueError(\n                \"`default_queue_id` is a required field. If you are \"\n                \"creating a new WorkPool and don't have a queue \"\n                \"ID yet, use the `actions.WorkPoolCreate` model instead.\"\n            )\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkPool.helpful_error_for_missing_default_queue_id","title":"helpful_error_for_missing_default_queue_id","text":"

    Default queue ID is required because all pools must have a default queue ID, but it represents a circular foreign key relationship to a WorkQueue (which can't be created until the work pool exists). Therefore, while this field can technically be null, it shouldn't be. This should only be an issue when creating new pools, as reading existing ones will always have this field populated. This custom error message will help users understand that they should use the actions.WorkPoolCreate model in that case.

    Source code in prefect/server/schemas/core.py
    @validator(\"default_queue_id\", always=True)\ndef helpful_error_for_missing_default_queue_id(cls, v):\n    \"\"\"\n    Default queue ID is required because all pools must have a default queue\n    ID, but it represents a circular foreign key relationship to a\n    WorkQueue (which can't be created until the work pool exists).\n    Therefore, while this field can *technically* be null, it shouldn't be.\n    This should only be an issue when creating new pools, as reading\n    existing ones will always have this field populated. This custom error\n    message will help users understand that they should use the\n    `actions.WorkPoolCreate` model in that case.\n    \"\"\"\n    if v is None:\n        raise ValueError(\n            \"`default_queue_id` is a required field. If you are \"\n            \"creating a new WorkPool and don't have a queue \"\n            \"ID yet, use the `actions.WorkPoolCreate` model instead.\"\n        )\n    return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkQueue","title":"WorkQueue","text":"

    Bases: ORMBaseModel

    An ORM representation of a work queue

    Source code in prefect/server/schemas/core.py
    class WorkQueue(ORMBaseModel):\n    \"\"\"An ORM representation of a work queue\"\"\"\n\n    name: str = Field(default=..., description=\"The name of the work queue.\")\n    description: Optional[str] = Field(\n        default=\"\", description=\"An optional description for the work queue.\"\n    )\n    is_paused: bool = Field(\n        default=False, description=\"Whether or not the work queue is paused.\"\n    )\n    concurrency_limit: Optional[conint(ge=0)] = Field(\n        default=None, description=\"An optional concurrency limit for the work queue.\"\n    )\n    priority: conint(ge=1) = Field(\n        default=1,\n        description=(\n            \"The queue's priority. Lower values are higher priority (1 is the highest).\"\n        ),\n    )\n    # Will be required after a future migration\n    work_pool_id: Optional[UUID] = Field(\n        default=None, description=\"The work pool with which the queue is associated.\"\n    )\n    filter: Optional[QueueFilter] = Field(\n        default=None,\n        description=\"DEPRECATED: Filter criteria for the work queue.\",\n        deprecated=True,\n    )\n    last_polled: Optional[DateTimeTZ] = Field(\n        default=None, description=\"The last time an agent polled this queue for work.\"\n    )\n\n    @validator(\"name\", check_fields=False)\n    def validate_name_characters(cls, v):\n        raise_on_name_with_banned_characters(v)\n        return v\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkQueueHealthPolicy","title":"WorkQueueHealthPolicy","text":"

    Bases: PrefectBaseModel

    Source code in prefect/server/schemas/core.py
    class WorkQueueHealthPolicy(PrefectBaseModel):\n    maximum_late_runs: Optional[int] = Field(\n        default=0,\n        description=(\n            \"The maximum number of late runs in the work queue before it is deemed\"\n            \" unhealthy. Defaults to `0`.\"\n        ),\n    )\n    maximum_seconds_since_last_polled: Optional[int] = Field(\n        default=60,\n        description=(\n            \"The maximum number of time in seconds elapsed since work queue has been\"\n            \" polled before it is deemed unhealthy. Defaults to `60`.\"\n        ),\n    )\n\n    def evaluate_health_status(\n        self, late_runs_count: int, last_polled: Optional[DateTimeTZ] = None\n    ) -> bool:\n        \"\"\"\n        Given empirical information about the state of the work queue, evaluate its health status.\n\n        Args:\n            late_runs: the count of late runs for the work queue.\n            last_polled: the last time the work queue was polled, if available.\n\n        Returns:\n            bool: whether or not the work queue is healthy.\n        \"\"\"\n        healthy = True\n        if (\n            self.maximum_late_runs is not None\n            and late_runs_count > self.maximum_late_runs\n        ):\n            healthy = False\n\n        if self.maximum_seconds_since_last_polled is not None:\n            if (\n                last_polled is None\n                or pendulum.now(\"UTC\").diff(last_polled).in_seconds()\n                > self.maximum_seconds_since_last_polled\n            ):\n                healthy = False\n\n        return healthy\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.WorkQueueHealthPolicy.evaluate_health_status","title":"evaluate_health_status","text":"

    Given empirical information about the state of the work queue, evaluate its health status.

    Parameters:

    Name Type Description Default late_runs

    the count of late runs for the work queue.

    required last_polled Optional[DateTimeTZ]

    the last time the work queue was polled, if available.

    None

    Returns:

    Name Type Description bool bool

    whether or not the work queue is healthy.

    Source code in prefect/server/schemas/core.py
    def evaluate_health_status(\n    self, late_runs_count: int, last_polled: Optional[DateTimeTZ] = None\n) -> bool:\n    \"\"\"\n    Given empirical information about the state of the work queue, evaluate its health status.\n\n    Args:\n        late_runs: the count of late runs for the work queue.\n        last_polled: the last time the work queue was polled, if available.\n\n    Returns:\n        bool: whether or not the work queue is healthy.\n    \"\"\"\n    healthy = True\n    if (\n        self.maximum_late_runs is not None\n        and late_runs_count > self.maximum_late_runs\n    ):\n        healthy = False\n\n    if self.maximum_seconds_since_last_polled is not None:\n        if (\n            last_polled is None\n            or pendulum.now(\"UTC\").diff(last_polled).in_seconds()\n            > self.maximum_seconds_since_last_polled\n        ):\n            healthy = False\n\n    return healthy\n
    "},{"location":"api-ref/server/schemas/core/#prefect.server.schemas.core.Worker","title":"Worker","text":"

    Bases: ORMBaseModel

    An ORM representation of a worker

    Source code in prefect/server/schemas/core.py
    class Worker(ORMBaseModel):\n    \"\"\"An ORM representation of a worker\"\"\"\n\n    name: str = Field(description=\"The name of the worker.\")\n    work_pool_id: UUID = Field(\n        description=\"The work pool with which the queue is associated.\"\n    )\n    last_heartbeat_time: datetime.datetime = Field(\n        None, description=\"The last time the worker process sent a heartbeat.\"\n    )\n    heartbeat_interval_seconds: Optional[int] = Field(\n        default=None,\n        description=(\n            \"The number of seconds to expect between heartbeats sent by the worker.\"\n        ),\n    )\n
    "},{"location":"api-ref/server/schemas/filters/","title":"server.schemas.filters","text":""},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters","title":"prefect.server.schemas.filters","text":"

    Schemas that define Prefect REST API filtering operations.

    Each filter schema includes logic for transforming itself into a SQL where clause.

    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilter","title":"ArtifactCollectionFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter artifact collections. Only artifact collections matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter artifact collections. Only artifact collections matching all criteria will be returned\"\"\"\n\n    latest_id: Optional[ArtifactCollectionFilterLatestId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.id`\"\n    )\n    key: Optional[ArtifactCollectionFilterKey] = Field(\n        default=None, description=\"Filter criteria for `Artifact.key`\"\n    )\n    flow_run_id: Optional[ArtifactCollectionFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.flow_run_id`\"\n    )\n    task_run_id: Optional[ArtifactCollectionFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.task_run_id`\"\n    )\n    type: Optional[ArtifactCollectionFilterType] = Field(\n        default=None, description=\"Filter criteria for `Artifact.type`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.latest_id is not None:\n            filters.append(self.latest_id.as_sql_filter(db))\n        if self.key is not None:\n            filters.append(self.key.as_sql_filter(db))\n        if self.flow_run_id is not None:\n            filters.append(self.flow_run_id.as_sql_filter(db))\n        if self.task_run_id is not None:\n            filters.append(self.task_run_id.as_sql_filter(db))\n        if self.type is not None:\n            filters.append(self.type.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterFlowRunId","title":"ArtifactCollectionFilterFlowRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.flow_run_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterFlowRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.flow_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterFlowRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterKey","title":"ArtifactCollectionFilterKey","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.key.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterKey(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.key`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact keys to include\"\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match artifact keys against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-artifact-%\",\n    )\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If `true`, only include artifacts with a non-null key. If `false`, \"\n            \"only include artifacts with a null key. Should return all rows in \"\n            \"the ArtifactCollection table if specified.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.key.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.ArtifactCollection.key.ilike(f\"%{self.like_}%\"))\n        if self.exists_ is not None:\n            filters.append(\n                db.ArtifactCollection.key.isnot(None)\n                if self.exists_\n                else db.ArtifactCollection.key.is_(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterKey.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterLatestId","title":"ArtifactCollectionFilterLatestId","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.latest_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterLatestId(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.latest_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of artifact ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.latest_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterLatestId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterTaskRunId","title":"ArtifactCollectionFilterTaskRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.task_run_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterTaskRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.task_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterTaskRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterType","title":"ArtifactCollectionFilterType","text":"

    Bases: PrefectFilterBaseModel

    Filter by ArtifactCollection.type.

    Source code in prefect/server/schemas/filters.py
    class ArtifactCollectionFilterType(PrefectFilterBaseModel):\n    \"\"\"Filter by `ArtifactCollection.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to exclude\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.ArtifactCollection.type.in_(self.any_))\n        if self.not_any_ is not None:\n            filters.append(db.ArtifactCollection.type.notin_(self.not_any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactCollectionFilterType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilter","title":"ArtifactFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter artifacts. Only artifacts matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter artifacts. Only artifacts matching all criteria will be returned\"\"\"\n\n    id: Optional[ArtifactFilterId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.id`\"\n    )\n    key: Optional[ArtifactFilterKey] = Field(\n        default=None, description=\"Filter criteria for `Artifact.key`\"\n    )\n    flow_run_id: Optional[ArtifactFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.flow_run_id`\"\n    )\n    task_run_id: Optional[ArtifactFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Artifact.task_run_id`\"\n    )\n    type: Optional[ArtifactFilterType] = Field(\n        default=None, description=\"Filter criteria for `Artifact.type`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.key is not None:\n            filters.append(self.key.as_sql_filter(db))\n        if self.flow_run_id is not None:\n            filters.append(self.flow_run_id.as_sql_filter(db))\n        if self.task_run_id is not None:\n            filters.append(self.task_run_id.as_sql_filter(db))\n        if self.type is not None:\n            filters.append(self.type.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterFlowRunId","title":"ArtifactFilterFlowRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.flow_run_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterFlowRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.flow_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterFlowRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterId","title":"ArtifactFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of artifact ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterKey","title":"ArtifactFilterKey","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.key.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterKey(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.key`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact keys to include\"\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match artifact keys against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-artifact-%\",\n    )\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If `true`, only include artifacts with a non-null key. If `false`, \"\n            \"only include artifacts with a null key.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.key.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Artifact.key.ilike(f\"%{self.like_}%\"))\n        if self.exists_ is not None:\n            filters.append(\n                db.Artifact.key.isnot(None)\n                if self.exists_\n                else db.Artifact.key.is_(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterKey.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterTaskRunId","title":"ArtifactFilterTaskRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.task_run_id.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterTaskRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.task_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterTaskRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterType","title":"ArtifactFilterType","text":"

    Bases: PrefectFilterBaseModel

    Filter by Artifact.type.

    Source code in prefect/server/schemas/filters.py
    class ArtifactFilterType(PrefectFilterBaseModel):\n    \"\"\"Filter by `Artifact.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of artifact types to exclude\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Artifact.type.in_(self.any_))\n        if self.not_any_ is not None:\n            filters.append(db.Artifact.type.notin_(self.not_any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.ArtifactFilterType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilter","title":"BlockDocumentFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter BlockDocuments. Only BlockDocuments matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter BlockDocuments. Only BlockDocuments matching all criteria will be returned\"\"\"\n\n    id: Optional[BlockDocumentFilterId] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.id`\"\n    )\n    is_anonymous: Optional[BlockDocumentFilterIsAnonymous] = Field(\n        # default is to exclude anonymous blocks\n        BlockDocumentFilterIsAnonymous(eq_=False),\n        description=(\n            \"Filter criteria for `BlockDocument.is_anonymous`. \"\n            \"Defaults to excluding anonymous blocks.\"\n        ),\n    )\n    block_type_id: Optional[BlockDocumentFilterBlockTypeId] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.block_type_id`\"\n    )\n    name: Optional[BlockDocumentFilterName] = Field(\n        default=None, description=\"Filter criteria for `BlockDocument.name`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.is_anonymous is not None:\n            filters.append(self.is_anonymous.as_sql_filter(db))\n        if self.block_type_id is not None:\n            filters.append(self.block_type_id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterBlockTypeId","title":"BlockDocumentFilterBlockTypeId","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockDocument.block_type_id.

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilterBlockTypeId(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockDocument.block_type_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block type ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockDocument.block_type_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterBlockTypeId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterId","title":"BlockDocumentFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockDocument.id.

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockDocument.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockDocument.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterIsAnonymous","title":"BlockDocumentFilterIsAnonymous","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockDocument.is_anonymous.

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilterIsAnonymous(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockDocument.is_anonymous`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Filter block documents for only those that are or are not anonymous.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.eq_ is not None:\n            filters.append(db.BlockDocument.is_anonymous.is_(self.eq_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterIsAnonymous.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterName","title":"BlockDocumentFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockDocument.name.

    Source code in prefect/server/schemas/filters.py
    class BlockDocumentFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockDocument.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of block names to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match block names against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-block%\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockDocument.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.BlockDocument.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockDocumentFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilter","title":"BlockSchemaFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter BlockSchemas

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter BlockSchemas\"\"\"\n\n    block_type_id: Optional[BlockSchemaFilterBlockTypeId] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.block_type_id`\"\n    )\n    block_capabilities: Optional[BlockSchemaFilterCapabilities] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.capabilities`\"\n    )\n    id: Optional[BlockSchemaFilterId] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.id`\"\n    )\n    version: Optional[BlockSchemaFilterVersion] = Field(\n        default=None, description=\"Filter criteria for `BlockSchema.version`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.block_type_id is not None:\n            filters.append(self.block_type_id.as_sql_filter(db))\n        if self.block_capabilities is not None:\n            filters.append(self.block_capabilities.as_sql_filter(db))\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.version is not None:\n            filters.append(self.version.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterBlockTypeId","title":"BlockSchemaFilterBlockTypeId","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockSchema.block_type_id.

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilterBlockTypeId(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockSchema.block_type_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of block type ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockSchema.block_type_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterBlockTypeId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterCapabilities","title":"BlockSchemaFilterCapabilities","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockSchema.capabilities

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilterCapabilities(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockSchema.capabilities`\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"write-storage\", \"read-storage\"],\n        description=(\n            \"A list of block capabilities. Block entities will be returned only if an\"\n            \" associated block schema has a superset of the defined capabilities.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.BlockSchema.capabilities, self.all_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterCapabilities.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterId","title":"BlockSchemaFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockSchema.id

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by BlockSchema.id\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockSchema.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterVersion","title":"BlockSchemaFilterVersion","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockSchema.capabilities

    Source code in prefect/server/schemas/filters.py
    class BlockSchemaFilterVersion(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockSchema.capabilities`\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"2.0.0\", \"2.1.0\"],\n        description=\"A list of block schema versions.\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        pass\n\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockSchema.version.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockSchemaFilterVersion.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilter","title":"BlockTypeFilter","text":"

    Bases: PrefectFilterBaseModel

    Filter BlockTypes

    Source code in prefect/server/schemas/filters.py
    class BlockTypeFilter(PrefectFilterBaseModel):\n    \"\"\"Filter BlockTypes\"\"\"\n\n    name: Optional[BlockTypeFilterName] = Field(\n        default=None, description=\"Filter criteria for `BlockType.name`\"\n    )\n\n    slug: Optional[BlockTypeFilterSlug] = Field(\n        default=None, description=\"Filter criteria for `BlockType.slug`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.slug is not None:\n            filters.append(self.slug.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilterName","title":"BlockTypeFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockType.name

    Source code in prefect/server/schemas/filters.py
    class BlockTypeFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockType.name`\"\"\"\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.like_ is not None:\n            filters.append(db.BlockType.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilterSlug","title":"BlockTypeFilterSlug","text":"

    Bases: PrefectFilterBaseModel

    Filter by BlockType.slug

    Source code in prefect/server/schemas/filters.py
    class BlockTypeFilterSlug(PrefectFilterBaseModel):\n    \"\"\"Filter by `BlockType.slug`\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of slugs to match\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.BlockType.slug.in_(self.any_))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.BlockTypeFilterSlug.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilter","title":"DeploymentFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter for deployments. Only deployments matching all criteria will be returned.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter for deployments. Only deployments matching all criteria will be returned.\"\"\"\n\n    id: Optional[DeploymentFilterId] = Field(\n        default=None, description=\"Filter criteria for `Deployment.id`\"\n    )\n    name: Optional[DeploymentFilterName] = Field(\n        default=None, description=\"Filter criteria for `Deployment.name`\"\n    )\n    is_schedule_active: Optional[DeploymentFilterIsScheduleActive] = Field(\n        default=None, description=\"Filter criteria for `Deployment.is_schedule_active`\"\n    )\n    tags: Optional[DeploymentFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Deployment.tags`\"\n    )\n    work_queue_name: Optional[DeploymentFilterWorkQueueName] = Field(\n        default=None, description=\"Filter criteria for `Deployment.work_queue_name`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.is_schedule_active is not None:\n            filters.append(self.is_schedule_active.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n        if self.work_queue_name is not None:\n            filters.append(self.work_queue_name.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterId","title":"DeploymentFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Deployment.id.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Deployment.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of deployment ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Deployment.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterIsScheduleActive","title":"DeploymentFilterIsScheduleActive","text":"

    Bases: PrefectFilterBaseModel

    Filter by Deployment.is_schedule_active.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterIsScheduleActive(PrefectFilterBaseModel):\n    \"\"\"Filter by `Deployment.is_schedule_active`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=\"Only returns where deployment schedule is/is not active\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.eq_ is not None:\n            filters.append(db.Deployment.is_schedule_active.is_(self.eq_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterIsScheduleActive.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterName","title":"DeploymentFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Deployment.name.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Deployment.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of deployment names to include\",\n        example=[\"my-deployment-1\", \"my-deployment-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Deployment.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Deployment.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterTags","title":"DeploymentFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by Deployment.tags.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `Deployment.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Deployments will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include deployments without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.Deployment.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.Deployment.tags == [] if self.is_null_ else db.Deployment.tags != []\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterWorkQueueName","title":"DeploymentFilterWorkQueueName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Deployment.work_queue_name.

    Source code in prefect/server/schemas/filters.py
    class DeploymentFilterWorkQueueName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Deployment.work_queue_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"work_queue_1\", \"work_queue_2\"],\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Deployment.work_queue_name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.DeploymentFilterWorkQueueName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FilterSet","title":"FilterSet","text":"

    Bases: PrefectBaseModel

    A collection of filters for common objects

    Source code in prefect/server/schemas/filters.py
    class FilterSet(PrefectBaseModel):\n    \"\"\"A collection of filters for common objects\"\"\"\n\n    flows: FlowFilter = Field(\n        default_factory=FlowFilter, description=\"Filters that apply to flows\"\n    )\n    flow_runs: FlowRunFilter = Field(\n        default_factory=FlowRunFilter, description=\"Filters that apply to flow runs\"\n    )\n    task_runs: TaskRunFilter = Field(\n        default_factory=TaskRunFilter, description=\"Filters that apply to task runs\"\n    )\n    deployments: DeploymentFilter = Field(\n        default_factory=DeploymentFilter,\n        description=\"Filters that apply to deployments\",\n    )\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FilterSet.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilter","title":"FlowFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter for flows. Only flows matching all criteria will be returned.

    Source code in prefect/server/schemas/filters.py
    class FlowFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter for flows. Only flows matching all criteria will be returned.\"\"\"\n\n    id: Optional[FlowFilterId] = Field(\n        default=None, description=\"Filter criteria for `Flow.id`\"\n    )\n    deployment: Optional[FlowFilterDeployment] = Field(\n        default=None, description=\"Filter criteria for Flow deployments\"\n    )\n    name: Optional[FlowFilterName] = Field(\n        default=None, description=\"Filter criteria for `Flow.name`\"\n    )\n    tags: Optional[FlowFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Flow.tags`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.deployment is not None:\n            filters.append(self.deployment.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterDeployment","title":"FlowFilterDeployment","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by flows by deployment

    Source code in prefect/server/schemas/filters.py
    class FlowFilterDeployment(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by flows by deployment\"\"\"\n\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flows without deployments\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.is_null_ is not None:\n            deployments_subquery = (\n                sa.select(db.Deployment.flow_id).distinct().subquery()\n            )\n\n            if self.is_null_:\n                filters.append(\n                    db.Flow.id.not_in(sa.select(deployments_subquery.c.flow_id))\n                )\n            else:\n                filters.append(\n                    db.Flow.id.in_(sa.select(deployments_subquery.c.flow_id))\n                )\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterDeployment.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterId","title":"FlowFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Flow.id.

    Source code in prefect/server/schemas/filters.py
    class FlowFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Flow.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Flow.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterName","title":"FlowFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Flow.name.

    Source code in prefect/server/schemas/filters.py
    class FlowFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Flow.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of flow names to include\",\n        example=[\"my-flow-1\", \"my-flow-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Flow.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Flow.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterTags","title":"FlowFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by Flow.tags.

    Source code in prefect/server/schemas/filters.py
    class FlowFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `Flow.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Flows will be returned only if their tags are a superset\"\n            \" of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include flows without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.Flow.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(db.Flow.tags == [] if self.is_null_ else db.Flow.tags != [])\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilter","title":"FlowRunFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter flow runs. Only flow runs matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter flow runs. Only flow runs matching all criteria will be returned\"\"\"\n\n    id: Optional[FlowRunFilterId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.id`\"\n    )\n    name: Optional[FlowRunFilterName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.name`\"\n    )\n    tags: Optional[FlowRunFilterTags] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.tags`\"\n    )\n    deployment_id: Optional[FlowRunFilterDeploymentId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.deployment_id`\"\n    )\n    work_queue_name: Optional[FlowRunFilterWorkQueueName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.work_queue_name\"\n    )\n    state: Optional[FlowRunFilterState] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.state`\"\n    )\n    flow_version: Optional[FlowRunFilterFlowVersion] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.flow_version`\"\n    )\n    start_time: Optional[FlowRunFilterStartTime] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.start_time`\"\n    )\n    expected_start_time: Optional[FlowRunFilterExpectedStartTime] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.expected_start_time`\"\n    )\n    next_scheduled_start_time: Optional[FlowRunFilterNextScheduledStartTime] = Field(\n        default=None,\n        description=\"Filter criteria for `FlowRun.next_scheduled_start_time`\",\n    )\n    parent_flow_run_id: Optional[FlowRunFilterParentFlowRunId] = Field(\n        default=None, description=\"Filter criteria for subflows of the given flow runs\"\n    )\n    parent_task_run_id: Optional[FlowRunFilterParentTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.parent_task_run_id`\"\n    )\n    idempotency_key: Optional[FlowRunFilterIdempotencyKey] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.idempotency_key`\"\n    )\n\n    def only_filters_on_id(self):\n        return (\n            self.id is not None\n            and (self.id.any_ and not self.id.not_any_)\n            and self.name is None\n            and self.tags is None\n            and self.deployment_id is None\n            and self.work_queue_name is None\n            and self.state is None\n            and self.flow_version is None\n            and self.start_time is None\n            and self.expected_start_time is None\n            and self.next_scheduled_start_time is None\n            and self.parent_flow_run_id is None\n            and self.parent_task_run_id is None\n            and self.idempotency_key is None\n        )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n        if self.deployment_id is not None:\n            filters.append(self.deployment_id.as_sql_filter(db))\n        if self.work_queue_name is not None:\n            filters.append(self.work_queue_name.as_sql_filter(db))\n        if self.flow_version is not None:\n            filters.append(self.flow_version.as_sql_filter(db))\n        if self.state is not None:\n            filters.append(self.state.as_sql_filter(db))\n        if self.start_time is not None:\n            filters.append(self.start_time.as_sql_filter(db))\n        if self.expected_start_time is not None:\n            filters.append(self.expected_start_time.as_sql_filter(db))\n        if self.next_scheduled_start_time is not None:\n            filters.append(self.next_scheduled_start_time.as_sql_filter(db))\n        if self.parent_flow_run_id is not None:\n            filters.append(self.parent_flow_run_id.as_sql_filter(db))\n        if self.parent_task_run_id is not None:\n            filters.append(self.parent_task_run_id.as_sql_filter(db))\n        if self.idempotency_key is not None:\n            filters.append(self.idempotency_key.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterDeploymentId","title":"FlowRunFilterDeploymentId","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.deployment_id.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterDeploymentId(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.deployment_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run deployment ids to include\"\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without deployment ids\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.deployment_id.in_(self.any_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.deployment_id.is_(None)\n                if self.is_null_\n                else db.FlowRun.deployment_id.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterDeploymentId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterExpectedStartTime","title":"FlowRunFilterExpectedStartTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.expected_start_time.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterExpectedStartTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.expected_start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs scheduled to start at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs scheduled to start at or after this time\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.FlowRun.expected_start_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.FlowRun.expected_start_time >= self.after_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterExpectedStartTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterFlowVersion","title":"FlowRunFilterFlowVersion","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.flow_version.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterFlowVersion(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.flow_version`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run flow_versions to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.flow_version.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterFlowVersion.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterId","title":"FlowRunFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.id.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run ids to include\"\n    )\n    not_any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run ids to exclude\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.id.in_(self.any_))\n        if self.not_any_ is not None:\n            filters.append(db.FlowRun.id.not_in(self.not_any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterIdempotencyKey","title":"FlowRunFilterIdempotencyKey","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.idempotency_key.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterIdempotencyKey(PrefectFilterBaseModel):\n    \"\"\"Filter by FlowRun.idempotency_key.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run idempotency keys to include\"\n    )\n    not_any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run idempotency keys to exclude\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.idempotency_key.in_(self.any_))\n        if self.not_any_ is not None:\n            filters.append(db.FlowRun.idempotency_key.not_in(self.not_any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterIdempotencyKey.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterName","title":"FlowRunFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.name.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of flow run names to include\",\n        example=[\"my-flow-run-1\", \"my-flow-run-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.FlowRun.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterNextScheduledStartTime","title":"FlowRunFilterNextScheduledStartTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.next_scheduled_start_time.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterNextScheduledStartTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.next_scheduled_start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include flow runs with a next_scheduled_start_time or before this\"\n            \" time\"\n        ),\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include flow runs with a next_scheduled_start_time at or after this\"\n            \" time\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.FlowRun.next_scheduled_start_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.FlowRun.next_scheduled_start_time >= self.after_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterNextScheduledStartTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterParentFlowRunId","title":"FlowRunFilterParentFlowRunId","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter for subflows of a given flow run

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterParentFlowRunId(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter for subflows of a given flow run\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of parent flow run ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(\n                db.FlowRun.id.in_(\n                    sa.select(db.FlowRun.id)\n                    .join(\n                        db.TaskRun,\n                        sa.and_(\n                            db.TaskRun.id == db.FlowRun.parent_task_run_id,\n                        ),\n                    )\n                    .where(db.TaskRun.flow_run_id.in_(self.any_))\n                )\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterParentFlowRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterParentTaskRunId","title":"FlowRunFilterParentTaskRunId","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.parent_task_run_id.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterParentTaskRunId(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.parent_task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run parent_task_run_ids to include\"\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without parent_task_run_id\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.parent_task_run_id.in_(self.any_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.parent_task_run_id.is_(None)\n                if self.is_null_\n                else db.FlowRun.parent_task_run_id.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterParentTaskRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStartTime","title":"FlowRunFilterStartTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.start_time.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterStartTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs starting at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include flow runs starting at or after this time\",\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only return flow runs without a start time\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.FlowRun.start_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.FlowRun.start_time >= self.after_)\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.start_time.is_(None)\n                if self.is_null_\n                else db.FlowRun.start_time.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStartTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterState","title":"FlowRunFilterState","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.state_type and FlowRun.state_name.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterState(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.state_type` and `FlowRun.state_name`.\"\"\"\n\n    type: Optional[FlowRunFilterStateType] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.state_type`\"\n    )\n    name: Optional[FlowRunFilterStateName] = Field(\n        default=None, description=\"Filter criteria for `FlowRun.state_name`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.type is not None:\n            filters.extend(self.type._get_filter_list(db))\n        if self.name is not None:\n            filters.extend(self.name._get_filter_list(db))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterState.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStateName","title":"FlowRunFilterStateName","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.state_name.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterStateName(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.state_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of flow run state names to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.state_name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStateName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStateType","title":"FlowRunFilterStateType","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRun.state_type.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterStateType(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRun.state_type`.\"\"\"\n\n    any_: Optional[List[schemas.states.StateType]] = Field(\n        default=None, description=\"A list of flow run state types to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.state_type.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterStateType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterTags","title":"FlowRunFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.tags.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Flow runs will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include flow runs without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.FlowRun.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.tags == [] if self.is_null_ else db.FlowRun.tags != []\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterWorkQueueName","title":"FlowRunFilterWorkQueueName","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by FlowRun.work_queue_name.

    Source code in prefect/server/schemas/filters.py
    class FlowRunFilterWorkQueueName(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `FlowRun.work_queue_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"work_queue_1\", \"work_queue_2\"],\n    )\n    is_null_: Optional[bool] = Field(\n        default=None,\n        description=\"If true, only include flow runs without work queue names\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.FlowRun.work_queue_name.in_(self.any_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.FlowRun.work_queue_name.is_(None)\n                if self.is_null_\n                else db.FlowRun.work_queue_name.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunFilterWorkQueueName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunNotificationPolicyFilter","title":"FlowRunNotificationPolicyFilter","text":"

    Bases: PrefectFilterBaseModel

    Filter FlowRunNotificationPolicies.

    Source code in prefect/server/schemas/filters.py
    class FlowRunNotificationPolicyFilter(PrefectFilterBaseModel):\n    \"\"\"Filter FlowRunNotificationPolicies.\"\"\"\n\n    is_active: Optional[FlowRunNotificationPolicyFilterIsActive] = Field(\n        default=FlowRunNotificationPolicyFilterIsActive(eq_=False),\n        description=\"Filter criteria for `FlowRunNotificationPolicy.is_active`. \",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.is_active is not None:\n            filters.append(self.is_active.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunNotificationPolicyFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunNotificationPolicyFilterIsActive","title":"FlowRunNotificationPolicyFilterIsActive","text":"

    Bases: PrefectFilterBaseModel

    Filter by FlowRunNotificationPolicy.is_active.

    Source code in prefect/server/schemas/filters.py
    class FlowRunNotificationPolicyFilterIsActive(PrefectFilterBaseModel):\n    \"\"\"Filter by `FlowRunNotificationPolicy.is_active`.\"\"\"\n\n    eq_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"Filter notification policies for only those that are or are not active.\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.eq_ is not None:\n            filters.append(db.FlowRunNotificationPolicy.is_active.is_(self.eq_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.FlowRunNotificationPolicyFilterIsActive.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilter","title":"LogFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter logs. Only logs matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class LogFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter logs. Only logs matching all criteria will be returned\"\"\"\n\n    level: Optional[LogFilterLevel] = Field(\n        default=None, description=\"Filter criteria for `Log.level`\"\n    )\n    timestamp: Optional[LogFilterTimestamp] = Field(\n        default=None, description=\"Filter criteria for `Log.timestamp`\"\n    )\n    flow_run_id: Optional[LogFilterFlowRunId] = Field(\n        default=None, description=\"Filter criteria for `Log.flow_run_id`\"\n    )\n    task_run_id: Optional[LogFilterTaskRunId] = Field(\n        default=None, description=\"Filter criteria for `Log.task_run_id`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.level is not None:\n            filters.append(self.level.as_sql_filter(db))\n        if self.timestamp is not None:\n            filters.append(self.timestamp.as_sql_filter(db))\n        if self.flow_run_id is not None:\n            filters.append(self.flow_run_id.as_sql_filter(db))\n        if self.task_run_id is not None:\n            filters.append(self.task_run_id.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterFlowRunId","title":"LogFilterFlowRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.flow_run_id.

    Source code in prefect/server/schemas/filters.py
    class LogFilterFlowRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.flow_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of flow run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Log.flow_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterFlowRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterLevel","title":"LogFilterLevel","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.level.

    Source code in prefect/server/schemas/filters.py
    class LogFilterLevel(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.level`.\"\"\"\n\n    ge_: Optional[int] = Field(\n        default=None,\n        description=\"Include logs with a level greater than or equal to this level\",\n        example=20,\n    )\n\n    le_: Optional[int] = Field(\n        default=None,\n        description=\"Include logs with a level less than or equal to this level\",\n        example=50,\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.ge_ is not None:\n            filters.append(db.Log.level >= self.ge_)\n        if self.le_ is not None:\n            filters.append(db.Log.level <= self.le_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterLevel.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterName","title":"LogFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.name.

    Source code in prefect/server/schemas/filters.py
    class LogFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of log names to include\",\n        example=[\"prefect.logger.flow_runs\", \"prefect.logger.task_runs\"],\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Log.name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterTaskRunId","title":"LogFilterTaskRunId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.task_run_id.

    Source code in prefect/server/schemas/filters.py
    class LogFilterTaskRunId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.task_run_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run IDs to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Log.task_run_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterTaskRunId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterTimestamp","title":"LogFilterTimestamp","text":"

    Bases: PrefectFilterBaseModel

    Filter by Log.timestamp.

    Source code in prefect/server/schemas/filters.py
    class LogFilterTimestamp(PrefectFilterBaseModel):\n    \"\"\"Filter by `Log.timestamp`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include logs with a timestamp at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include logs with a timestamp at or after this time\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.Log.timestamp <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.Log.timestamp >= self.after_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.LogFilterTimestamp.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.Operator","title":"Operator","text":"

    Bases: AutoEnum

    Operators for combining filter criteria.

    Source code in prefect/server/schemas/filters.py
    class Operator(AutoEnum):\n    \"\"\"Operators for combining filter criteria.\"\"\"\n\n    and_ = AutoEnum.auto()\n    or_ = AutoEnum.auto()\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.Operator.auto","title":"auto staticmethod","text":"

    Exposes enum.auto() to avoid requiring a second import to use AutoEnum

    Source code in prefect/utilities/collections.py
    @staticmethod\ndef auto():\n    \"\"\"\n    Exposes `enum.auto()` to avoid requiring a second import to use `AutoEnum`\n    \"\"\"\n    return auto()\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.PrefectFilterBaseModel","title":"PrefectFilterBaseModel","text":"

    Bases: PrefectBaseModel

    Base model for Prefect filters

    Source code in prefect/server/schemas/filters.py
    class PrefectFilterBaseModel(PrefectBaseModel):\n    \"\"\"Base model for Prefect filters\"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    def as_sql_filter(self, db: \"PrefectDBInterface\") -> \"BooleanClauseList\":\n        \"\"\"Generate SQL filter from provided filter parameters. If no filters parameters are available, return a TRUE filter.\"\"\"\n        filters = self._get_filter_list(db)\n        if not filters:\n            return True\n        return sa.and_(*filters)\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        \"\"\"Return a list of boolean filter statements based on filter parameters\"\"\"\n        raise NotImplementedError(\"_get_filter_list must be implemented\")\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.PrefectFilterBaseModel.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.PrefectOperatorFilterBaseModel","title":"PrefectOperatorFilterBaseModel","text":"

    Bases: PrefectFilterBaseModel

    Base model for Prefect filters that combines criteria with a user-provided operator

    Source code in prefect/server/schemas/filters.py
    class PrefectOperatorFilterBaseModel(PrefectFilterBaseModel):\n    \"\"\"Base model for Prefect filters that combines criteria with a user-provided operator\"\"\"\n\n    operator: Operator = Field(\n        default=Operator.and_,\n        description=\"Operator for combining filter criteria. Defaults to 'and_'.\",\n    )\n\n    def as_sql_filter(self, db: \"PrefectDBInterface\") -> \"BooleanClauseList\":\n        filters = self._get_filter_list(db)\n        if not filters:\n            return True\n        return sa.and_(*filters) if self.operator == Operator.and_ else sa.or_(*filters)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.PrefectOperatorFilterBaseModel.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilter","title":"TaskRunFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter task runs. Only task runs matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter task runs. Only task runs matching all criteria will be returned\"\"\"\n\n    id: Optional[TaskRunFilterId] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.id`\"\n    )\n    name: Optional[TaskRunFilterName] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.name`\"\n    )\n    tags: Optional[TaskRunFilterTags] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.tags`\"\n    )\n    state: Optional[TaskRunFilterState] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.state`\"\n    )\n    start_time: Optional[TaskRunFilterStartTime] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.start_time`\"\n    )\n    subflow_runs: Optional[TaskRunFilterSubFlowRuns] = Field(\n        default=None, description=\"Filter criteria for `TaskRun.subflow_run`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n        if self.state is not None:\n            filters.append(self.state.as_sql_filter(db))\n        if self.start_time is not None:\n            filters.append(self.start_time.as_sql_filter(db))\n        if self.subflow_runs is not None:\n            filters.append(self.subflow_runs.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterId","title":"TaskRunFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.id.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of task run ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.TaskRun.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterName","title":"TaskRunFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.name.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of task run names to include\",\n        example=[\"my-task-run-1\", \"my-task-run-2\"],\n    )\n\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A case-insensitive partial match. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', 'sad-Marvin', and 'marvin-robot'.\"\n        ),\n        example=\"marvin\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.TaskRun.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.TaskRun.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStartTime","title":"TaskRunFilterStartTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.start_time.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterStartTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.start_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include task runs starting at or before this time\",\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=\"Only include task runs starting at or after this time\",\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only return task runs without a start time\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.TaskRun.start_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.TaskRun.start_time >= self.after_)\n        if self.is_null_ is not None:\n            filters.append(\n                db.TaskRun.start_time.is_(None)\n                if self.is_null_\n                else db.TaskRun.start_time.is_not(None)\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStartTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterState","title":"TaskRunFilterState","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by TaskRun.type and TaskRun.name.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterState(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `TaskRun.type` and `TaskRun.name`.\"\"\"\n\n    type: Optional[TaskRunFilterStateType]\n    name: Optional[TaskRunFilterStateName]\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.type is not None:\n            filters.extend(self.type._get_filter_list(db))\n        if self.name is not None:\n            filters.extend(self.name._get_filter_list(db))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterState.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStateName","title":"TaskRunFilterStateName","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.state_name.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterStateName(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.state_name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of task run state names to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.TaskRun.state_name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStateName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStateType","title":"TaskRunFilterStateType","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.state_type.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterStateType(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.state_type`.\"\"\"\n\n    any_: Optional[List[schemas.states.StateType]] = Field(\n        default=None, description=\"A list of task run state types to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.TaskRun.state_type.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterStateType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterSubFlowRuns","title":"TaskRunFilterSubFlowRuns","text":"

    Bases: PrefectFilterBaseModel

    Filter by TaskRun.subflow_run.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterSubFlowRuns(PrefectFilterBaseModel):\n    \"\"\"Filter by `TaskRun.subflow_run`.\"\"\"\n\n    exists_: Optional[bool] = Field(\n        default=None,\n        description=(\n            \"If true, only include task runs that are subflow run parents; if false,\"\n            \" exclude parent task runs\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.exists_ is True:\n            filters.append(db.TaskRun.subflow_run.has())\n        elif self.exists_ is False:\n            filters.append(sa.not_(db.TaskRun.subflow_run.has()))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterSubFlowRuns.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterTags","title":"TaskRunFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by TaskRun.tags.

    Source code in prefect/server/schemas/filters.py
    class TaskRunFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `TaskRun.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Task runs will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include task runs without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.TaskRun.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.TaskRun.tags == [] if self.is_null_ else db.TaskRun.tags != []\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.TaskRunFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilter","title":"VariableFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter variables. Only variables matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class VariableFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter variables. Only variables matching all criteria will be returned\"\"\"\n\n    id: Optional[VariableFilterId] = Field(\n        default=None, description=\"Filter criteria for `Variable.id`\"\n    )\n    name: Optional[VariableFilterName] = Field(\n        default=None, description=\"Filter criteria for `Variable.name`\"\n    )\n    value: Optional[VariableFilterValue] = Field(\n        default=None, description=\"Filter criteria for `Variable.value`\"\n    )\n    tags: Optional[VariableFilterTags] = Field(\n        default=None, description=\"Filter criteria for `Variable.tags`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.value is not None:\n            filters.append(self.value.as_sql_filter(db))\n        if self.tags is not None:\n            filters.append(self.tags.as_sql_filter(db))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterId","title":"VariableFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Variable.id.

    Source code in prefect/server/schemas/filters.py
    class VariableFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Variable.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of variable ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Variable.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterName","title":"VariableFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by Variable.name.

    Source code in prefect/server/schemas/filters.py
    class VariableFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `Variable.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of variables names to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match variable names against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my_variable_%\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Variable.name.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Variable.name.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterTags","title":"VariableFilterTags","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by Variable.tags.

    Source code in prefect/server/schemas/filters.py
    class VariableFilterTags(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `Variable.tags`.\"\"\"\n\n    all_: Optional[List[str]] = Field(\n        default=None,\n        example=[\"tag-1\", \"tag-2\"],\n        description=(\n            \"A list of tags. Variables will be returned only if their tags are a\"\n            \" superset of the list\"\n        ),\n    )\n    is_null_: Optional[bool] = Field(\n        default=None, description=\"If true, only include Variables without tags\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        from prefect.server.utilities.database import json_has_all_keys\n\n        filters = []\n        if self.all_ is not None:\n            filters.append(json_has_all_keys(db.Variable.tags, self.all_))\n        if self.is_null_ is not None:\n            filters.append(\n                db.Variable.tags == [] if self.is_null_ else db.Variable.tags != []\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterTags.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterValue","title":"VariableFilterValue","text":"

    Bases: PrefectFilterBaseModel

    Filter by Variable.value.

    Source code in prefect/server/schemas/filters.py
    class VariableFilterValue(PrefectFilterBaseModel):\n    \"\"\"Filter by `Variable.value`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of variables value to include\"\n    )\n    like_: Optional[str] = Field(\n        default=None,\n        description=(\n            \"A string to match variable value against. This can include \"\n            \"SQL wildcard characters like `%` and `_`.\"\n        ),\n        example=\"my-value-%\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Variable.value.in_(self.any_))\n        if self.like_ is not None:\n            filters.append(db.Variable.value.ilike(f\"%{self.like_}%\"))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.VariableFilterValue.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilter","title":"WorkPoolFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter work pools. Only work pools matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class WorkPoolFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter work pools. Only work pools matching all criteria will be returned\"\"\"\n\n    id: Optional[WorkPoolFilterId] = Field(\n        default=None, description=\"Filter criteria for `WorkPool.id`\"\n    )\n    name: Optional[WorkPoolFilterName] = Field(\n        default=None, description=\"Filter criteria for `WorkPool.name`\"\n    )\n    type: Optional[WorkPoolFilterType] = Field(\n        default=None, description=\"Filter criteria for `WorkPool.type`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n        if self.type is not None:\n            filters.append(self.type.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterId","title":"WorkPoolFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkPool.id.

    Source code in prefect/server/schemas/filters.py
    class WorkPoolFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkPool.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of work pool ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkPool.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterName","title":"WorkPoolFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkPool.name.

    Source code in prefect/server/schemas/filters.py
    class WorkPoolFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkPool.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of work pool names to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkPool.name.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterType","title":"WorkPoolFilterType","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkPool.type.

    Source code in prefect/server/schemas/filters.py
    class WorkPoolFilterType(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkPool.type`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None, description=\"A list of work pool types to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkPool.type.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkPoolFilterType.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilter","title":"WorkQueueFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter work queues. Only work queues matching all criteria will be returned

    Source code in prefect/server/schemas/filters.py
    class WorkQueueFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter work queues. Only work queues matching all criteria will be\n    returned\"\"\"\n\n    id: Optional[WorkQueueFilterId] = Field(\n        default=None, description=\"Filter criteria for `WorkQueue.id`\"\n    )\n\n    name: Optional[WorkQueueFilterName] = Field(\n        default=None, description=\"Filter criteria for `WorkQueue.name`\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.id is not None:\n            filters.append(self.id.as_sql_filter(db))\n        if self.name is not None:\n            filters.append(self.name.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilterId","title":"WorkQueueFilterId","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkQueue.id.

    Source code in prefect/server/schemas/filters.py
    class WorkQueueFilterId(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkQueue.id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None,\n        description=\"A list of work queue ids to include\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkQueue.id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilterId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilterName","title":"WorkQueueFilterName","text":"

    Bases: PrefectFilterBaseModel

    Filter by WorkQueue.name.

    Source code in prefect/server/schemas/filters.py
    class WorkQueueFilterName(PrefectFilterBaseModel):\n    \"\"\"Filter by `WorkQueue.name`.\"\"\"\n\n    any_: Optional[List[str]] = Field(\n        default=None,\n        description=\"A list of work queue names to include\",\n        example=[\"wq-1\", \"wq-2\"],\n    )\n\n    startswith_: Optional[List[str]] = Field(\n        default=None,\n        description=(\n            \"A list of case-insensitive starts-with matches. For example, \"\n            \" passing 'marvin' will match \"\n            \"'marvin', and 'Marvin-robot', but not 'sad-marvin'.\"\n        ),\n        example=[\"marvin\", \"Marvin-robot\"],\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.WorkQueue.name.in_(self.any_))\n        if self.startswith_ is not None:\n            filters.append(\n                sa.or_(\n                    *[db.WorkQueue.name.ilike(f\"{item}%\") for item in self.startswith_]\n                )\n            )\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkQueueFilterName.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilter","title":"WorkerFilter","text":"

    Bases: PrefectOperatorFilterBaseModel

    Filter by Worker.last_heartbeat_time.

    Source code in prefect/server/schemas/filters.py
    class WorkerFilter(PrefectOperatorFilterBaseModel):\n    \"\"\"Filter by `Worker.last_heartbeat_time`.\"\"\"\n\n    # worker_config_id: Optional[WorkerFilterWorkPoolId] = Field(\n    #     default=None, description=\"Filter criteria for `Worker.worker_config_id`\"\n    # )\n\n    last_heartbeat_time: Optional[WorkerFilterLastHeartbeatTime] = Field(\n        default=None,\n        description=\"Filter criteria for `Worker.last_heartbeat_time`\",\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n\n        if self.last_heartbeat_time is not None:\n            filters.append(self.last_heartbeat_time.as_sql_filter(db))\n\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilter.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilterLastHeartbeatTime","title":"WorkerFilterLastHeartbeatTime","text":"

    Bases: PrefectFilterBaseModel

    Filter by Worker.last_heartbeat_time.

    Source code in prefect/server/schemas/filters.py
    class WorkerFilterLastHeartbeatTime(PrefectFilterBaseModel):\n    \"\"\"Filter by `Worker.last_heartbeat_time`.\"\"\"\n\n    before_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include processes whose last heartbeat was at or before this time\"\n        ),\n    )\n    after_: Optional[DateTimeTZ] = Field(\n        default=None,\n        description=(\n            \"Only include processes whose last heartbeat was at or after this time\"\n        ),\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.before_ is not None:\n            filters.append(db.Worker.last_heartbeat_time <= self.before_)\n        if self.after_ is not None:\n            filters.append(db.Worker.last_heartbeat_time >= self.after_)\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilterLastHeartbeatTime.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilterWorkPoolId","title":"WorkerFilterWorkPoolId","text":"

    Bases: PrefectFilterBaseModel

    Filter by Worker.worker_config_id.

    Source code in prefect/server/schemas/filters.py
    class WorkerFilterWorkPoolId(PrefectFilterBaseModel):\n    \"\"\"Filter by `Worker.worker_config_id`.\"\"\"\n\n    any_: Optional[List[UUID]] = Field(\n        default=None, description=\"A list of work pool ids to include\"\n    )\n\n    def _get_filter_list(self, db: \"PrefectDBInterface\") -> List:\n        filters = []\n        if self.any_ is not None:\n            filters.append(db.Worker.worker_config_id.in_(self.any_))\n        return filters\n
    "},{"location":"api-ref/server/schemas/filters/#prefect.server.schemas.filters.WorkerFilterWorkPoolId.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/responses/","title":"server.schemas.responses","text":""},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses","title":"prefect.server.schemas.responses","text":"

    Schemas for special responses from the Prefect REST API.

    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.FlowRunResponse","title":"FlowRunResponse","text":"

    Bases: ORMBaseModel

    Source code in prefect/server/schemas/responses.py
    @copy_model_fields\nclass FlowRunResponse(ORMBaseModel):\n    name: str = FieldFrom(schemas.core.FlowRun)\n    flow_id: UUID = FieldFrom(schemas.core.FlowRun)\n    state_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    deployment_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    work_queue_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    work_queue_name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    flow_version: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    parameters: dict = FieldFrom(schemas.core.FlowRun)\n    idempotency_key: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    context: dict = FieldFrom(schemas.core.FlowRun)\n    empirical_policy: FlowRunPolicy = FieldFrom(schemas.core.FlowRun)\n    tags: List[str] = FieldFrom(schemas.core.FlowRun)\n    parent_task_run_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    state_type: Optional[schemas.states.StateType] = FieldFrom(schemas.core.FlowRun)\n    state_name: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    run_count: int = FieldFrom(schemas.core.FlowRun)\n    expected_start_time: Optional[DateTimeTZ] = FieldFrom(schemas.core.FlowRun)\n    next_scheduled_start_time: Optional[DateTimeTZ] = FieldFrom(schemas.core.FlowRun)\n    start_time: Optional[DateTimeTZ] = FieldFrom(schemas.core.FlowRun)\n    end_time: Optional[DateTimeTZ] = FieldFrom(schemas.core.FlowRun)\n    total_run_time: datetime.timedelta = FieldFrom(schemas.core.FlowRun)\n    estimated_run_time: datetime.timedelta = FieldFrom(schemas.core.FlowRun)\n    estimated_start_time_delta: datetime.timedelta = FieldFrom(schemas.core.FlowRun)\n    auto_scheduled: bool = FieldFrom(schemas.core.FlowRun)\n    infrastructure_document_id: Optional[UUID] = FieldFrom(schemas.core.FlowRun)\n    infrastructure_pid: Optional[str] = FieldFrom(schemas.core.FlowRun)\n    created_by: Optional[CreatedBy] = FieldFrom(schemas.core.FlowRun)\n    work_pool_id: Optional[UUID] = Field(\n        default=None,\n        description=\"The id of the flow run's work pool.\",\n    )\n    work_pool_name: Optional[str] = Field(\n        default=None,\n        description=\"The name of the flow run's work pool.\",\n        example=\"my-work-pool\",\n    )\n    state: Optional[schemas.states.State] = FieldFrom(schemas.core.FlowRun)\n\n    @classmethod\n    def from_orm(cls, orm_flow_run: \"prefect.server.database.orm_models.ORMFlowRun\"):\n        response = super().from_orm(orm_flow_run)\n        if orm_flow_run.work_queue:\n            response.work_queue_id = orm_flow_run.work_queue.id\n            response.work_queue_name = orm_flow_run.work_queue.name\n            if orm_flow_run.work_queue.work_pool:\n                response.work_pool_id = orm_flow_run.work_queue.work_pool.id\n                response.work_pool_name = orm_flow_run.work_queue.work_pool.name\n\n        return response\n\n    def __eq__(self, other: Any) -> bool:\n        \"\"\"\n        Check for \"equality\" to another flow run schema\n\n        Estimates times are rolling and will always change with repeated queries for\n        a flow run so we ignore them during equality checks.\n        \"\"\"\n        if isinstance(other, FlowRunResponse):\n            exclude_fields = {\"estimated_run_time\", \"estimated_start_time_delta\"}\n            return self.dict(exclude=exclude_fields) == other.dict(\n                exclude=exclude_fields\n            )\n        return super().__eq__(other)\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.HistoryResponse","title":"HistoryResponse","text":"

    Bases: PrefectBaseModel

    Represents a history of aggregation states over an interval

    Source code in prefect/server/schemas/responses.py
    class HistoryResponse(PrefectBaseModel):\n    \"\"\"Represents a history of aggregation states over an interval\"\"\"\n\n    interval_start: DateTimeTZ = Field(\n        default=..., description=\"The start date of the interval.\"\n    )\n    interval_end: DateTimeTZ = Field(\n        default=..., description=\"The end date of the interval.\"\n    )\n    states: List[HistoryResponseState] = Field(\n        default=..., description=\"A list of state histories during the interval.\"\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.HistoryResponseState","title":"HistoryResponseState","text":"

    Bases: PrefectBaseModel

    Represents a single state's history over an interval.

    Source code in prefect/server/schemas/responses.py
    class HistoryResponseState(PrefectBaseModel):\n    \"\"\"Represents a single state's history over an interval.\"\"\"\n\n    state_type: schemas.states.StateType = Field(\n        default=..., description=\"The state type.\"\n    )\n    state_name: str = Field(default=..., description=\"The state name.\")\n    count_runs: int = Field(\n        default=...,\n        description=\"The number of runs in the specified state during the interval.\",\n    )\n    sum_estimated_run_time: datetime.timedelta = Field(\n        default=...,\n        description=\"The total estimated run time of all runs during the interval.\",\n    )\n    sum_estimated_lateness: datetime.timedelta = Field(\n        default=...,\n        description=(\n            \"The sum of differences between actual and expected start time during the\"\n            \" interval.\"\n        ),\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.OrchestrationResult","title":"OrchestrationResult","text":"

    Bases: PrefectBaseModel

    A container for the output of state orchestration.

    Source code in prefect/server/schemas/responses.py
    class OrchestrationResult(PrefectBaseModel):\n    \"\"\"\n    A container for the output of state orchestration.\n    \"\"\"\n\n    state: Optional[schemas.states.State]\n    status: SetStateStatus\n    details: StateResponseDetails\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.SetStateStatus","title":"SetStateStatus","text":"

    Bases: AutoEnum

    Enumerates return statuses for setting run states.

    Source code in prefect/server/schemas/responses.py
    class SetStateStatus(AutoEnum):\n    \"\"\"Enumerates return statuses for setting run states.\"\"\"\n\n    ACCEPT = AutoEnum.auto()\n    REJECT = AutoEnum.auto()\n    ABORT = AutoEnum.auto()\n    WAIT = AutoEnum.auto()\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.StateAbortDetails","title":"StateAbortDetails","text":"

    Bases: PrefectBaseModel

    Details associated with an ABORT state transition.

    Source code in prefect/server/schemas/responses.py
    class StateAbortDetails(PrefectBaseModel):\n    \"\"\"Details associated with an ABORT state transition.\"\"\"\n\n    type: Literal[\"abort_details\"] = Field(\n        default=\"abort_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition was aborted.\"\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.StateAcceptDetails","title":"StateAcceptDetails","text":"

    Bases: PrefectBaseModel

    Details associated with an ACCEPT state transition.

    Source code in prefect/server/schemas/responses.py
    class StateAcceptDetails(PrefectBaseModel):\n    \"\"\"Details associated with an ACCEPT state transition.\"\"\"\n\n    type: Literal[\"accept_details\"] = Field(\n        default=\"accept_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.StateRejectDetails","title":"StateRejectDetails","text":"

    Bases: PrefectBaseModel

    Details associated with a REJECT state transition.

    Source code in prefect/server/schemas/responses.py
    class StateRejectDetails(PrefectBaseModel):\n    \"\"\"Details associated with a REJECT state transition.\"\"\"\n\n    type: Literal[\"reject_details\"] = Field(\n        default=\"reject_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition was rejected.\"\n    )\n
    "},{"location":"api-ref/server/schemas/responses/#prefect.server.schemas.responses.StateWaitDetails","title":"StateWaitDetails","text":"

    Bases: PrefectBaseModel

    Details associated with a WAIT state transition.

    Source code in prefect/server/schemas/responses.py
    class StateWaitDetails(PrefectBaseModel):\n    \"\"\"Details associated with a WAIT state transition.\"\"\"\n\n    type: Literal[\"wait_details\"] = Field(\n        default=\"wait_details\",\n        description=(\n            \"The type of state transition detail. Used to ensure pydantic does not\"\n            \" coerce into a different type.\"\n        ),\n    )\n    delay_seconds: int = Field(\n        default=...,\n        description=(\n            \"The length of time in seconds the client should wait before transitioning\"\n            \" states.\"\n        ),\n    )\n    reason: Optional[str] = Field(\n        default=None, description=\"The reason why the state transition should wait.\"\n    )\n
    "},{"location":"api-ref/server/schemas/schedules/","title":"server.schemas.schedules","text":""},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules","title":"prefect.server.schemas.schedules","text":"

    Schedule schemas

    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.CronSchedule","title":"CronSchedule","text":"

    Bases: PrefectBaseModel

    Cron schedule

    NOTE: If the timezone is a DST-observing one, then the schedule will adjust itself appropriately. Cron's rules for DST are based on schedule times, not intervals. This means that an hourly cron schedule will fire on every new schedule hour, not every elapsed hour; for example, when clocks are set back this will result in a two-hour pause as the schedule will fire the first time 1am is reached and the first time 2am is reached, 120 minutes later. Longer schedules, such as one that fires at 9am every morning, will automatically adjust for DST.

    Parameters:

    Name Type Description Default cron str

    a valid cron string

    required timezone str

    a valid timezone string in IANA tzdata format (for example, America/New_York).

    required day_or bool

    Control how croniter handles day and day_of_week entries. Defaults to True, matching cron which connects those values using OR. If the switch is set to False, the values are connected using AND. This behaves like fcron and enables you to e.g. define a job that executes each 2nd friday of a month by setting the days of month and the weekday.

    required Source code in prefect/server/schemas/schedules.py
    class CronSchedule(PrefectBaseModel):\n    \"\"\"\n    Cron schedule\n\n    NOTE: If the timezone is a DST-observing one, then the schedule will adjust\n    itself appropriately. Cron's rules for DST are based on schedule times, not\n    intervals. This means that an hourly cron schedule will fire on every new\n    schedule hour, not every elapsed hour; for example, when clocks are set back\n    this will result in a two-hour pause as the schedule will fire *the first\n    time* 1am is reached and *the first time* 2am is reached, 120 minutes later.\n    Longer schedules, such as one that fires at 9am every morning, will\n    automatically adjust for DST.\n\n    Args:\n        cron (str): a valid cron string\n        timezone (str): a valid timezone string in IANA tzdata format (for example,\n            America/New_York).\n        day_or (bool, optional): Control how croniter handles `day` and `day_of_week`\n            entries. Defaults to True, matching cron which connects those values using\n            OR. If the switch is set to False, the values are connected using AND. This\n            behaves like fcron and enables you to e.g. define a job that executes each\n            2nd friday of a month by setting the days of month and the weekday.\n\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    cron: str = Field(default=..., example=\"0 0 * * *\")\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n    day_or: bool = Field(\n        default=True,\n        description=(\n            \"Control croniter behavior for handling day and day_of_week entries.\"\n        ),\n    )\n\n    @validator(\"timezone\")\n    def valid_timezone(cls, v):\n        if v and v not in pendulum.tz.timezones:\n            raise ValueError(\n                f'Invalid timezone: \"{v}\" (specify in IANA tzdata format, for example,'\n                \" America/New_York)\"\n            )\n        return v\n\n    @validator(\"cron\")\n    def valid_cron_string(cls, v):\n        # croniter allows \"random\" and \"hashed\" expressions\n        # which we do not support https://github.com/kiorky/croniter\n        if not croniter.is_valid(v):\n            raise ValueError(f'Invalid cron string: \"{v}\"')\n        elif any(c for c in v.split() if c.casefold() in [\"R\", \"H\", \"r\", \"h\"]):\n            raise ValueError(\n                f'Random and Hashed expressions are unsupported, received: \"{v}\"'\n            )\n        return v\n\n    async def get_dates(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> List[pendulum.DateTime]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to None.  If a timezone-naive datetime is\n                provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): The maximum scheduled date to return. If\n                a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: A list of dates\n        \"\"\"\n        return sorted(self._get_dates_generator(n=n, start=start, end=end))\n\n    def _get_dates_generator(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> Generator[pendulum.DateTime, None, None]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to the current date. If a timezone-naive\n                datetime is provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): No returned date will exceed this date.\n                If a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: a list of dates\n        \"\"\"\n        if start is None:\n            start = pendulum.now(\"UTC\")\n\n        start, end = _prepare_scheduling_start_and_end(start, end, self.timezone)\n\n        if n is None:\n            # if an end was supplied, we do our best to supply all matching dates (up to\n            # MAX_ITERATIONS)\n            if end is not None:\n                n = MAX_ITERATIONS\n            else:\n                n = 1\n\n        elif self.timezone:\n            start = start.in_tz(self.timezone)\n\n        # subtract one second from the start date, so that croniter returns it\n        # as an event (if it meets the cron criteria)\n        start = start.subtract(seconds=1)\n\n        # Respect microseconds by rounding up\n        if start.microsecond > 0:\n            start += datetime.timedelta(seconds=1)\n\n        # croniter's DST logic interferes with all other datetime libraries except pytz\n        start_localized = pytz.timezone(start.tz.name).localize(\n            datetime.datetime(\n                year=start.year,\n                month=start.month,\n                day=start.day,\n                hour=start.hour,\n                minute=start.minute,\n                second=start.second,\n                microsecond=start.microsecond,\n            )\n        )\n        start_naive_tz = start.naive()\n\n        cron = croniter(self.cron, start_naive_tz, day_or=self.day_or)  # type: ignore\n        dates = set()\n        counter = 0\n\n        while True:\n            # croniter does not handle DST properly when the start time is\n            # in and around when the actual shift occurs. To work around this,\n            # we use the naive start time to get the next cron date delta, then\n            # add that time to the original scheduling anchor.\n            next_time = cron.get_next(datetime.datetime)\n            delta = next_time - start_naive_tz\n            next_date = pendulum.instance(start_localized + delta)\n\n            # if the end date was exceeded, exit\n            if end and next_date > end:\n                break\n            # ensure no duplicates; weird things can happen with DST\n            if next_date not in dates:\n                dates.add(next_date)\n                yield next_date\n\n            # if enough dates have been collected or enough attempts were made, exit\n            if len(dates) >= n or counter > MAX_ITERATIONS:\n                break\n\n            counter += 1\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.CronSchedule.get_dates","title":"get_dates async","text":"

    Retrieves dates from the schedule. Up to 1,000 candidate dates are checked following the start date.

    Parameters:

    Name Type Description Default n int

    The number of dates to generate

    None start datetime

    The first returned date will be on or after this date. Defaults to None. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None end datetime

    The maximum scheduled date to return. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None

    Returns:

    Type Description List[DateTime]

    List[pendulum.DateTime]: A list of dates

    Source code in prefect/server/schemas/schedules.py
    async def get_dates(\n    self,\n    n: int = None,\n    start: datetime.datetime = None,\n    end: datetime.datetime = None,\n) -> List[pendulum.DateTime]:\n    \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n    following the start date.\n\n    Args:\n        n (int): The number of dates to generate\n        start (datetime.datetime, optional): The first returned date will be on or\n            after this date. Defaults to None.  If a timezone-naive datetime is\n            provided, it is assumed to be in the schedule's timezone.\n        end (datetime.datetime, optional): The maximum scheduled date to return. If\n            a timezone-naive datetime is provided, it is assumed to be in the\n            schedule's timezone.\n\n    Returns:\n        List[pendulum.DateTime]: A list of dates\n    \"\"\"\n    return sorted(self._get_dates_generator(n=n, start=start, end=end))\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.IntervalSchedule","title":"IntervalSchedule","text":"

    Bases: PrefectBaseModel

    A schedule formed by adding interval increments to an anchor_date. If no anchor_date is supplied, the current UTC time is used. If a timezone-naive datetime is provided for anchor_date, it is assumed to be in the schedule's timezone (or UTC). Even if supplied with an IANA timezone, anchor dates are always stored as UTC offsets, so a timezone can be provided to determine localization behaviors like DST boundary handling. If none is provided it will be inferred from the anchor date.

    NOTE: If the IntervalSchedule anchor_date or timezone is provided in a DST-observing timezone, then the schedule will adjust itself appropriately. Intervals greater than 24 hours will follow DST conventions, while intervals of less than 24 hours will follow UTC intervals. For example, an hourly schedule will fire every UTC hour, even across DST boundaries. When clocks are set back, this will result in two runs that appear to both be scheduled for 1am local time, even though they are an hour apart in UTC time. For longer intervals, like a daily schedule, the interval schedule will adjust for DST boundaries so that the clock-hour remains constant. This means that a daily schedule that always fires at 9am will observe DST and continue to fire at 9am in the local time zone.

    Parameters:

    Name Type Description Default interval timedelta

    an interval to schedule on

    required anchor_date DateTimeTZ

    an anchor date to schedule increments against; if not provided, the current timestamp will be used

    required timezone str

    a valid timezone string

    required Source code in prefect/server/schemas/schedules.py
    class IntervalSchedule(PrefectBaseModel):\n    \"\"\"\n    A schedule formed by adding `interval` increments to an `anchor_date`. If no\n    `anchor_date` is supplied, the current UTC time is used.  If a\n    timezone-naive datetime is provided for `anchor_date`, it is assumed to be\n    in the schedule's timezone (or UTC). Even if supplied with an IANA timezone,\n    anchor dates are always stored as UTC offsets, so a `timezone` can be\n    provided to determine localization behaviors like DST boundary handling. If\n    none is provided it will be inferred from the anchor date.\n\n    NOTE: If the `IntervalSchedule` `anchor_date` or `timezone` is provided in a\n    DST-observing timezone, then the schedule will adjust itself appropriately.\n    Intervals greater than 24 hours will follow DST conventions, while intervals\n    of less than 24 hours will follow UTC intervals. For example, an hourly\n    schedule will fire every UTC hour, even across DST boundaries. When clocks\n    are set back, this will result in two runs that *appear* to both be\n    scheduled for 1am local time, even though they are an hour apart in UTC\n    time. For longer intervals, like a daily schedule, the interval schedule\n    will adjust for DST boundaries so that the clock-hour remains constant. This\n    means that a daily schedule that always fires at 9am will observe DST and\n    continue to fire at 9am in the local time zone.\n\n    Args:\n        interval (datetime.timedelta): an interval to schedule on\n        anchor_date (DateTimeTZ, optional): an anchor date to schedule increments against;\n            if not provided, the current timestamp will be used\n        timezone (str, optional): a valid timezone string\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n        exclude_none = True\n\n    interval: datetime.timedelta\n    anchor_date: DateTimeTZ = None\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n\n    @validator(\"interval\")\n    def interval_must_be_positive(cls, v):\n        if v.total_seconds() <= 0:\n            raise ValueError(\"The interval must be positive\")\n        return v\n\n    @validator(\"anchor_date\", always=True)\n    def default_anchor_date(cls, v):\n        if v is None:\n            return pendulum.now(\"UTC\")\n        return pendulum.instance(v)\n\n    @validator(\"timezone\", always=True)\n    def default_timezone(cls, v, *, values, **kwargs):\n        # if was provided, make sure its a valid IANA string\n        if v and v not in pendulum.tz.timezones:\n            raise ValueError(f'Invalid timezone: \"{v}\"')\n\n        # otherwise infer the timezone from the anchor date\n        elif v is None and values.get(\"anchor_date\"):\n            tz = values[\"anchor_date\"].tz.name\n            if tz in pendulum.tz.timezones:\n                return tz\n            # sometimes anchor dates have \"timezones\" that are UTC offsets\n            # like \"-04:00\". This happens when parsing ISO8601 strings.\n            # In this case we, the correct inferred localization is \"UTC\".\n            else:\n                return \"UTC\"\n\n        return v\n\n    async def get_dates(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> List[pendulum.DateTime]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to None.  If a timezone-naive datetime is\n                provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): The maximum scheduled date to return. If\n                a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: A list of dates\n        \"\"\"\n        return sorted(self._get_dates_generator(n=n, start=start, end=end))\n\n    def _get_dates_generator(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> Generator[pendulum.DateTime, None, None]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to None.  If a timezone-naive datetime is\n                provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): The maximum scheduled date to return. If\n                a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: a list of dates\n        \"\"\"\n        if n is None:\n            # if an end was supplied, we do our best to supply all matching dates (up to\n            # MAX_ITERATIONS)\n            if end is not None:\n                n = MAX_ITERATIONS\n            else:\n                n = 1\n\n        if start is None:\n            start = pendulum.now(\"UTC\")\n\n        anchor_tz = self.anchor_date.in_tz(self.timezone)\n        start, end = _prepare_scheduling_start_and_end(start, end, self.timezone)\n\n        # compute the offset between the anchor date and the start date to jump to the\n        # next date\n        offset = (start - anchor_tz).total_seconds() / self.interval.total_seconds()\n        next_date = anchor_tz.add(seconds=self.interval.total_seconds() * int(offset))\n\n        # break the interval into `days` and `seconds` because pendulum\n        # will handle DST boundaries properly if days are provided, but not\n        # if we add `total seconds`. Therefore, `next_date + self.interval`\n        # fails while `next_date.add(days=days, seconds=seconds)` works.\n        interval_days = self.interval.days\n        interval_seconds = self.interval.total_seconds() - (\n            interval_days * 24 * 60 * 60\n        )\n\n        # daylight saving time boundaries can create a situation where the next date is\n        # before the start date, so we advance it if necessary\n        while next_date < start:\n            next_date = next_date.add(days=interval_days, seconds=interval_seconds)\n\n        counter = 0\n        dates = set()\n\n        while True:\n            # if the end date was exceeded, exit\n            if end and next_date > end:\n                break\n\n            # ensure no duplicates; weird things can happen with DST\n            if next_date not in dates:\n                dates.add(next_date)\n                yield next_date\n\n            # if enough dates have been collected or enough attempts were made, exit\n            if len(dates) >= n or counter > MAX_ITERATIONS:\n                break\n\n            counter += 1\n\n            next_date = next_date.add(days=interval_days, seconds=interval_seconds)\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.IntervalSchedule.get_dates","title":"get_dates async","text":"

    Retrieves dates from the schedule. Up to 1,000 candidate dates are checked following the start date.

    Parameters:

    Name Type Description Default n int

    The number of dates to generate

    None start datetime

    The first returned date will be on or after this date. Defaults to None. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None end datetime

    The maximum scheduled date to return. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None

    Returns:

    Type Description List[DateTime]

    List[pendulum.DateTime]: A list of dates

    Source code in prefect/server/schemas/schedules.py
    async def get_dates(\n    self,\n    n: int = None,\n    start: datetime.datetime = None,\n    end: datetime.datetime = None,\n) -> List[pendulum.DateTime]:\n    \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n    following the start date.\n\n    Args:\n        n (int): The number of dates to generate\n        start (datetime.datetime, optional): The first returned date will be on or\n            after this date. Defaults to None.  If a timezone-naive datetime is\n            provided, it is assumed to be in the schedule's timezone.\n        end (datetime.datetime, optional): The maximum scheduled date to return. If\n            a timezone-naive datetime is provided, it is assumed to be in the\n            schedule's timezone.\n\n    Returns:\n        List[pendulum.DateTime]: A list of dates\n    \"\"\"\n    return sorted(self._get_dates_generator(n=n, start=start, end=end))\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.RRuleSchedule","title":"RRuleSchedule","text":"

    Bases: PrefectBaseModel

    RRule schedule, based on the iCalendar standard (RFC 5545) as implemented in dateutils.rrule.

    RRules are appropriate for any kind of calendar-date manipulation, including irregular intervals, repetition, exclusions, week day or day-of-month adjustments, and more.

    Note that as a calendar-oriented standard, RRuleSchedules are sensitive to to the initial timezone provided. A 9am daily schedule with a daylight saving time-aware start date will maintain a local 9am time through DST boundaries; a 9am daily schedule with a UTC start date will maintain a 9am UTC time.

    Parameters:

    Name Type Description Default rrule str

    a valid RRule string

    required timezone str

    a valid timezone string

    required Source code in prefect/server/schemas/schedules.py
    class RRuleSchedule(PrefectBaseModel):\n    \"\"\"\n    RRule schedule, based on the iCalendar standard\n    ([RFC 5545](https://datatracker.ietf.org/doc/html/rfc5545)) as\n    implemented in `dateutils.rrule`.\n\n    RRules are appropriate for any kind of calendar-date manipulation, including\n    irregular intervals, repetition, exclusions, week day or day-of-month\n    adjustments, and more.\n\n    Note that as a calendar-oriented standard, `RRuleSchedules` are sensitive to\n    to the initial timezone provided. A 9am daily schedule with a daylight saving\n    time-aware start date will maintain a local 9am time through DST boundaries;\n    a 9am daily schedule with a UTC start date will maintain a 9am UTC time.\n\n    Args:\n        rrule (str): a valid RRule string\n        timezone (str, optional): a valid timezone string\n    \"\"\"\n\n    class Config:\n        extra = \"forbid\"\n\n    rrule: str\n    timezone: Optional[str] = Field(default=None, example=\"America/New_York\")\n\n    @validator(\"rrule\")\n    def validate_rrule_str(cls, v):\n        # attempt to parse the rrule string as an rrule object\n        # this will error if the string is invalid\n        try:\n            dateutil.rrule.rrulestr(v, cache=True)\n        except ValueError as exc:\n            # rrules errors are a mix of cryptic and informative\n            # so reraise to be clear that the string was invalid\n            raise ValueError(f'Invalid RRule string \"{v}\": {exc}')\n        if len(v) > MAX_RRULE_LENGTH:\n            raise ValueError(\n                f'Invalid RRule string \"{v[:40]}...\"\\n'\n                f\"Max length is {MAX_RRULE_LENGTH}, got {len(v)}\"\n            )\n        return v\n\n    @classmethod\n    def from_rrule(cls, rrule: dateutil.rrule.rrule):\n        if isinstance(rrule, dateutil.rrule.rrule):\n            if rrule._dtstart.tzinfo is not None:\n                timezone = rrule._dtstart.tzinfo.name\n            else:\n                timezone = \"UTC\"\n            return RRuleSchedule(rrule=str(rrule), timezone=timezone)\n        elif isinstance(rrule, dateutil.rrule.rruleset):\n            dtstarts = [rr._dtstart for rr in rrule._rrule if rr._dtstart is not None]\n            unique_dstarts = set(pendulum.instance(d).in_tz(\"UTC\") for d in dtstarts)\n            unique_timezones = set(d.tzinfo for d in dtstarts if d.tzinfo is not None)\n\n            if len(unique_timezones) > 1:\n                raise ValueError(\n                    f\"rruleset has too many dtstart timezones: {unique_timezones}\"\n                )\n\n            if len(unique_dstarts) > 1:\n                raise ValueError(f\"rruleset has too many dtstarts: {unique_dstarts}\")\n\n            if unique_dstarts and unique_timezones:\n                timezone = dtstarts[0].tzinfo.name\n            else:\n                timezone = \"UTC\"\n\n            rruleset_string = \"\"\n            if rrule._rrule:\n                rruleset_string += \"\\n\".join(str(r) for r in rrule._rrule)\n            if rrule._exrule:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"\\n\".join(str(r) for r in rrule._exrule).replace(\n                    \"RRULE\", \"EXRULE\"\n                )\n            if rrule._rdate:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"RDATE:\" + \",\".join(\n                    rd.strftime(\"%Y%m%dT%H%M%SZ\") for rd in rrule._rdate\n                )\n            if rrule._exdate:\n                rruleset_string += \"\\n\" if rruleset_string else \"\"\n                rruleset_string += \"EXDATE:\" + \",\".join(\n                    exd.strftime(\"%Y%m%dT%H%M%SZ\") for exd in rrule._exdate\n                )\n            return RRuleSchedule(rrule=rruleset_string, timezone=timezone)\n        else:\n            raise ValueError(f\"Invalid RRule object: {rrule}\")\n\n    def to_rrule(self) -> dateutil.rrule.rrule:\n        \"\"\"\n        Since rrule doesn't properly serialize/deserialize timezones, we localize dates\n        here\n        \"\"\"\n        rrule = dateutil.rrule.rrulestr(\n            self.rrule,\n            dtstart=DEFAULT_ANCHOR_DATE,\n            cache=True,\n        )\n        timezone = dateutil.tz.gettz(self.timezone)\n        if isinstance(rrule, dateutil.rrule.rrule):\n            kwargs = dict(dtstart=rrule._dtstart.replace(tzinfo=timezone))\n            if rrule._until:\n                kwargs.update(\n                    until=rrule._until.replace(tzinfo=timezone),\n                )\n            return rrule.replace(**kwargs)\n        elif isinstance(rrule, dateutil.rrule.rruleset):\n            # update rrules\n            localized_rrules = []\n            for rr in rrule._rrule:\n                kwargs = dict(dtstart=rr._dtstart.replace(tzinfo=timezone))\n                if rr._until:\n                    kwargs.update(\n                        until=rr._until.replace(tzinfo=timezone),\n                    )\n                localized_rrules.append(rr.replace(**kwargs))\n            rrule._rrule = localized_rrules\n\n            # update exrules\n            localized_exrules = []\n            for exr in rrule._exrule:\n                kwargs = dict(dtstart=exr._dtstart.replace(tzinfo=timezone))\n                if exr._until:\n                    kwargs.update(\n                        until=exr._until.replace(tzinfo=timezone),\n                    )\n                localized_exrules.append(exr.replace(**kwargs))\n            rrule._exrule = localized_exrules\n\n            # update rdates\n            localized_rdates = []\n            for rd in rrule._rdate:\n                localized_rdates.append(rd.replace(tzinfo=timezone))\n            rrule._rdate = localized_rdates\n\n            # update exdates\n            localized_exdates = []\n            for exd in rrule._exdate:\n                localized_exdates.append(exd.replace(tzinfo=timezone))\n            rrule._exdate = localized_exdates\n\n            return rrule\n\n    @validator(\"timezone\", always=True)\n    def valid_timezone(cls, v):\n        if v and v not in pytz.all_timezones_set:\n            raise ValueError(f'Invalid timezone: \"{v}\"')\n        elif v is None:\n            return \"UTC\"\n        return v\n\n    async def get_dates(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> List[pendulum.DateTime]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to None.  If a timezone-naive datetime is\n                provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): The maximum scheduled date to return. If\n                a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: A list of dates\n        \"\"\"\n        return sorted(self._get_dates_generator(n=n, start=start, end=end))\n\n    def _get_dates_generator(\n        self,\n        n: int = None,\n        start: datetime.datetime = None,\n        end: datetime.datetime = None,\n    ) -> Generator[pendulum.DateTime, None, None]:\n        \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n        following the start date.\n\n        Args:\n            n (int): The number of dates to generate\n            start (datetime.datetime, optional): The first returned date will be on or\n                after this date. Defaults to the current date. If a timezone-naive\n                datetime is provided, it is assumed to be in the schedule's timezone.\n            end (datetime.datetime, optional): No returned date will exceed this date.\n                If a timezone-naive datetime is provided, it is assumed to be in the\n                schedule's timezone.\n\n        Returns:\n            List[pendulum.DateTime]: a list of dates\n        \"\"\"\n        if start is None:\n            start = pendulum.now(\"UTC\")\n\n        start, end = _prepare_scheduling_start_and_end(start, end, self.timezone)\n\n        if n is None:\n            # if an end was supplied, we do our best to supply all matching dates (up\n            # to MAX_ITERATIONS)\n            if end is not None:\n                n = MAX_ITERATIONS\n            else:\n                n = 1\n\n        dates = set()\n        counter = 0\n\n        # pass count = None to account for discrepancies with duplicates around DST\n        # boundaries\n        for next_date in self.to_rrule().xafter(start, count=None, inc=True):\n            next_date = pendulum.instance(next_date).in_tz(self.timezone)\n\n            # if the end date was exceeded, exit\n            if end and next_date > end:\n                break\n\n            # ensure no duplicates; weird things can happen with DST\n            if next_date not in dates:\n                dates.add(next_date)\n                yield next_date\n\n            # if enough dates have been collected or enough attempts were made, exit\n            if len(dates) >= n or counter > MAX_ITERATIONS:\n                break\n\n            counter += 1\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.RRuleSchedule.get_dates","title":"get_dates async","text":"

    Retrieves dates from the schedule. Up to 1,000 candidate dates are checked following the start date.

    Parameters:

    Name Type Description Default n int

    The number of dates to generate

    None start datetime

    The first returned date will be on or after this date. Defaults to None. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None end datetime

    The maximum scheduled date to return. If a timezone-naive datetime is provided, it is assumed to be in the schedule's timezone.

    None

    Returns:

    Type Description List[DateTime]

    List[pendulum.DateTime]: A list of dates

    Source code in prefect/server/schemas/schedules.py
    async def get_dates(\n    self,\n    n: int = None,\n    start: datetime.datetime = None,\n    end: datetime.datetime = None,\n) -> List[pendulum.DateTime]:\n    \"\"\"Retrieves dates from the schedule. Up to 1,000 candidate dates are checked\n    following the start date.\n\n    Args:\n        n (int): The number of dates to generate\n        start (datetime.datetime, optional): The first returned date will be on or\n            after this date. Defaults to None.  If a timezone-naive datetime is\n            provided, it is assumed to be in the schedule's timezone.\n        end (datetime.datetime, optional): The maximum scheduled date to return. If\n            a timezone-naive datetime is provided, it is assumed to be in the\n            schedule's timezone.\n\n    Returns:\n        List[pendulum.DateTime]: A list of dates\n    \"\"\"\n    return sorted(self._get_dates_generator(n=n, start=start, end=end))\n
    "},{"location":"api-ref/server/schemas/schedules/#prefect.server.schemas.schedules.RRuleSchedule.to_rrule","title":"to_rrule","text":"

    Since rrule doesn't properly serialize/deserialize timezones, we localize dates here

    Source code in prefect/server/schemas/schedules.py
    def to_rrule(self) -> dateutil.rrule.rrule:\n    \"\"\"\n    Since rrule doesn't properly serialize/deserialize timezones, we localize dates\n    here\n    \"\"\"\n    rrule = dateutil.rrule.rrulestr(\n        self.rrule,\n        dtstart=DEFAULT_ANCHOR_DATE,\n        cache=True,\n    )\n    timezone = dateutil.tz.gettz(self.timezone)\n    if isinstance(rrule, dateutil.rrule.rrule):\n        kwargs = dict(dtstart=rrule._dtstart.replace(tzinfo=timezone))\n        if rrule._until:\n            kwargs.update(\n                until=rrule._until.replace(tzinfo=timezone),\n            )\n        return rrule.replace(**kwargs)\n    elif isinstance(rrule, dateutil.rrule.rruleset):\n        # update rrules\n        localized_rrules = []\n        for rr in rrule._rrule:\n            kwargs = dict(dtstart=rr._dtstart.replace(tzinfo=timezone))\n            if rr._until:\n                kwargs.update(\n                    until=rr._until.replace(tzinfo=timezone),\n                )\n            localized_rrules.append(rr.replace(**kwargs))\n        rrule._rrule = localized_rrules\n\n        # update exrules\n        localized_exrules = []\n        for exr in rrule._exrule:\n            kwargs = dict(dtstart=exr._dtstart.replace(tzinfo=timezone))\n            if exr._until:\n                kwargs.update(\n                    until=exr._until.replace(tzinfo=timezone),\n                )\n            localized_exrules.append(exr.replace(**kwargs))\n        rrule._exrule = localized_exrules\n\n        # update rdates\n        localized_rdates = []\n        for rd in rrule._rdate:\n            localized_rdates.append(rd.replace(tzinfo=timezone))\n        rrule._rdate = localized_rdates\n\n        # update exdates\n        localized_exdates = []\n        for exd in rrule._exdate:\n            localized_exdates.append(exd.replace(tzinfo=timezone))\n        rrule._exdate = localized_exdates\n\n        return rrule\n
    "},{"location":"api-ref/server/schemas/sorting/","title":"server.schemas.sorting","text":""},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting","title":"prefect.server.schemas.sorting","text":"

    Schemas for sorting Prefect REST API objects.

    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.ArtifactCollectionSort","title":"ArtifactCollectionSort","text":"

    Bases: AutoEnum

    Defines artifact collection sorting options.

    Source code in prefect/server/schemas/sorting.py
    class ArtifactCollectionSort(AutoEnum):\n    \"\"\"Defines artifact collection sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    ID_DESC = AutoEnum.auto()\n    KEY_DESC = AutoEnum.auto()\n    KEY_ASC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort artifact collections\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.ArtifactCollection.created.desc(),\n            \"UPDATED_DESC\": db.ArtifactCollection.updated.desc(),\n            \"ID_DESC\": db.ArtifactCollection.id.desc(),\n            \"KEY_DESC\": db.ArtifactCollection.key.desc(),\n            \"KEY_ASC\": db.ArtifactCollection.key.asc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.ArtifactCollectionSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort artifact collections

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort artifact collections\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.ArtifactCollection.created.desc(),\n        \"UPDATED_DESC\": db.ArtifactCollection.updated.desc(),\n        \"ID_DESC\": db.ArtifactCollection.id.desc(),\n        \"KEY_DESC\": db.ArtifactCollection.key.desc(),\n        \"KEY_ASC\": db.ArtifactCollection.key.asc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.ArtifactSort","title":"ArtifactSort","text":"

    Bases: AutoEnum

    Defines artifact sorting options.

    Source code in prefect/server/schemas/sorting.py
    class ArtifactSort(AutoEnum):\n    \"\"\"Defines artifact sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    ID_DESC = AutoEnum.auto()\n    KEY_DESC = AutoEnum.auto()\n    KEY_ASC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort artifacts\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.Artifact.created.desc(),\n            \"UPDATED_DESC\": db.Artifact.updated.desc(),\n            \"ID_DESC\": db.Artifact.id.desc(),\n            \"KEY_DESC\": db.Artifact.key.desc(),\n            \"KEY_ASC\": db.Artifact.key.asc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.ArtifactSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort artifacts

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort artifacts\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.Artifact.created.desc(),\n        \"UPDATED_DESC\": db.Artifact.updated.desc(),\n        \"ID_DESC\": db.Artifact.id.desc(),\n        \"KEY_DESC\": db.Artifact.key.desc(),\n        \"KEY_ASC\": db.Artifact.key.asc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.BlockDocumentSort","title":"BlockDocumentSort","text":"

    Bases: AutoEnum

    Defines block document sorting options.

    Source code in prefect/server/schemas/sorting.py
    class BlockDocumentSort(AutoEnum):\n    \"\"\"Defines block document sorting options.\"\"\"\n\n    NAME_DESC = \"NAME_DESC\"\n    NAME_ASC = \"NAME_ASC\"\n    BLOCK_TYPE_AND_NAME_ASC = \"BLOCK_TYPE_AND_NAME_ASC\"\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort block documents\"\"\"\n        sort_mapping = {\n            \"NAME_DESC\": db.BlockDocument.name.desc(),\n            \"NAME_ASC\": db.BlockDocument.name.asc(),\n            \"BLOCK_TYPE_AND_NAME_ASC\": sa.text(\"block_type_name asc, name asc\"),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.BlockDocumentSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort block documents

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort block documents\"\"\"\n    sort_mapping = {\n        \"NAME_DESC\": db.BlockDocument.name.desc(),\n        \"NAME_ASC\": db.BlockDocument.name.asc(),\n        \"BLOCK_TYPE_AND_NAME_ASC\": sa.text(\"block_type_name asc, name asc\"),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.DeploymentSort","title":"DeploymentSort","text":"

    Bases: AutoEnum

    Defines deployment sorting options.

    Source code in prefect/server/schemas/sorting.py
    class DeploymentSort(AutoEnum):\n    \"\"\"Defines deployment sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort deployments\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.Deployment.created.desc(),\n            \"UPDATED_DESC\": db.Deployment.updated.desc(),\n            \"NAME_ASC\": db.Deployment.name.asc(),\n            \"NAME_DESC\": db.Deployment.name.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.DeploymentSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort deployments

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort deployments\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.Deployment.created.desc(),\n        \"UPDATED_DESC\": db.Deployment.updated.desc(),\n        \"NAME_ASC\": db.Deployment.name.asc(),\n        \"NAME_DESC\": db.Deployment.name.desc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.FlowRunSort","title":"FlowRunSort","text":"

    Bases: AutoEnum

    Defines flow run sorting options.

    Source code in prefect/server/schemas/sorting.py
    class FlowRunSort(AutoEnum):\n    \"\"\"Defines flow run sorting options.\"\"\"\n\n    ID_DESC = AutoEnum.auto()\n    START_TIME_ASC = AutoEnum.auto()\n    START_TIME_DESC = AutoEnum.auto()\n    EXPECTED_START_TIME_ASC = AutoEnum.auto()\n    EXPECTED_START_TIME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n    NEXT_SCHEDULED_START_TIME_ASC = AutoEnum.auto()\n    END_TIME_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        from sqlalchemy.sql.functions import coalesce\n\n        \"\"\"Return an expression used to sort flow runs\"\"\"\n        sort_mapping = {\n            \"ID_DESC\": db.FlowRun.id.desc(),\n            \"START_TIME_ASC\": coalesce(\n                db.FlowRun.start_time, db.FlowRun.expected_start_time\n            ).asc(),\n            \"START_TIME_DESC\": coalesce(\n                db.FlowRun.start_time, db.FlowRun.expected_start_time\n            ).desc(),\n            \"EXPECTED_START_TIME_ASC\": db.FlowRun.expected_start_time.asc(),\n            \"EXPECTED_START_TIME_DESC\": db.FlowRun.expected_start_time.desc(),\n            \"NAME_ASC\": db.FlowRun.name.asc(),\n            \"NAME_DESC\": db.FlowRun.name.desc(),\n            \"NEXT_SCHEDULED_START_TIME_ASC\": db.FlowRun.next_scheduled_start_time.asc(),\n            \"END_TIME_DESC\": db.FlowRun.end_time.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.FlowSort","title":"FlowSort","text":"

    Bases: AutoEnum

    Defines flow sorting options.

    Source code in prefect/server/schemas/sorting.py
    class FlowSort(AutoEnum):\n    \"\"\"Defines flow sorting options.\"\"\"\n\n    CREATED_DESC = AutoEnum.auto()\n    UPDATED_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort flows\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.Flow.created.desc(),\n            \"UPDATED_DESC\": db.Flow.updated.desc(),\n            \"NAME_ASC\": db.Flow.name.asc(),\n            \"NAME_DESC\": db.Flow.name.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.FlowSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort flows

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort flows\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.Flow.created.desc(),\n        \"UPDATED_DESC\": db.Flow.updated.desc(),\n        \"NAME_ASC\": db.Flow.name.asc(),\n        \"NAME_DESC\": db.Flow.name.desc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.LogSort","title":"LogSort","text":"

    Bases: AutoEnum

    Defines log sorting options.

    Source code in prefect/server/schemas/sorting.py
    class LogSort(AutoEnum):\n    \"\"\"Defines log sorting options.\"\"\"\n\n    TIMESTAMP_ASC = AutoEnum.auto()\n    TIMESTAMP_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort task runs\"\"\"\n        sort_mapping = {\n            \"TIMESTAMP_ASC\": db.Log.timestamp.asc(),\n            \"TIMESTAMP_DESC\": db.Log.timestamp.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.LogSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort task runs

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort task runs\"\"\"\n    sort_mapping = {\n        \"TIMESTAMP_ASC\": db.Log.timestamp.asc(),\n        \"TIMESTAMP_DESC\": db.Log.timestamp.desc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.TaskRunSort","title":"TaskRunSort","text":"

    Bases: AutoEnum

    Defines task run sorting options.

    Source code in prefect/server/schemas/sorting.py
    class TaskRunSort(AutoEnum):\n    \"\"\"Defines task run sorting options.\"\"\"\n\n    ID_DESC = AutoEnum.auto()\n    EXPECTED_START_TIME_ASC = AutoEnum.auto()\n    EXPECTED_START_TIME_DESC = AutoEnum.auto()\n    NAME_ASC = AutoEnum.auto()\n    NAME_DESC = AutoEnum.auto()\n    NEXT_SCHEDULED_START_TIME_ASC = AutoEnum.auto()\n    END_TIME_DESC = AutoEnum.auto()\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort task runs\"\"\"\n        sort_mapping = {\n            \"ID_DESC\": db.TaskRun.id.desc(),\n            \"EXPECTED_START_TIME_ASC\": db.TaskRun.expected_start_time.asc(),\n            \"EXPECTED_START_TIME_DESC\": db.TaskRun.expected_start_time.desc(),\n            \"NAME_ASC\": db.TaskRun.name.asc(),\n            \"NAME_DESC\": db.TaskRun.name.desc(),\n            \"NEXT_SCHEDULED_START_TIME_ASC\": db.TaskRun.next_scheduled_start_time.asc(),\n            \"END_TIME_DESC\": db.TaskRun.end_time.desc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.TaskRunSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort task runs

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort task runs\"\"\"\n    sort_mapping = {\n        \"ID_DESC\": db.TaskRun.id.desc(),\n        \"EXPECTED_START_TIME_ASC\": db.TaskRun.expected_start_time.asc(),\n        \"EXPECTED_START_TIME_DESC\": db.TaskRun.expected_start_time.desc(),\n        \"NAME_ASC\": db.TaskRun.name.asc(),\n        \"NAME_DESC\": db.TaskRun.name.desc(),\n        \"NEXT_SCHEDULED_START_TIME_ASC\": db.TaskRun.next_scheduled_start_time.asc(),\n        \"END_TIME_DESC\": db.TaskRun.end_time.desc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.VariableSort","title":"VariableSort","text":"

    Bases: AutoEnum

    Defines variables sorting options.

    Source code in prefect/server/schemas/sorting.py
    class VariableSort(AutoEnum):\n    \"\"\"Defines variables sorting options.\"\"\"\n\n    CREATED_DESC = \"CREATED_DESC\"\n    UPDATED_DESC = \"UPDATED_DESC\"\n    NAME_DESC = \"NAME_DESC\"\n    NAME_ASC = \"NAME_ASC\"\n\n    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n        \"\"\"Return an expression used to sort variables\"\"\"\n        sort_mapping = {\n            \"CREATED_DESC\": db.Variable.created.desc(),\n            \"UPDATED_DESC\": db.Variable.updated.desc(),\n            \"NAME_DESC\": db.Variable.name.desc(),\n            \"NAME_ASC\": db.Variable.name.asc(),\n        }\n        return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/sorting/#prefect.server.schemas.sorting.VariableSort.as_sql_sort","title":"as_sql_sort","text":"

    Return an expression used to sort variables

    Source code in prefect/server/schemas/sorting.py
    def as_sql_sort(self, db: \"PrefectDBInterface\") -> \"ColumnElement\":\n    \"\"\"Return an expression used to sort variables\"\"\"\n    sort_mapping = {\n        \"CREATED_DESC\": db.Variable.created.desc(),\n        \"UPDATED_DESC\": db.Variable.updated.desc(),\n        \"NAME_DESC\": db.Variable.name.desc(),\n        \"NAME_ASC\": db.Variable.name.asc(),\n    }\n    return sort_mapping[self.value]\n
    "},{"location":"api-ref/server/schemas/states/","title":"server.schemas.states","text":""},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states","title":"prefect.server.schemas.states","text":"

    State schemas.

    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.State","title":"State","text":"

    Bases: StateBaseModel, Generic[R]

    Represents the state of a run.

    Source code in prefect/server/schemas/states.py
    class State(StateBaseModel, Generic[R]):\n    \"\"\"Represents the state of a run.\"\"\"\n\n    class Config:\n        orm_mode = True\n\n    type: StateType\n    name: Optional[str] = Field(default=None)\n    timestamp: DateTimeTZ = Field(default_factory=lambda: pendulum.now(\"UTC\"))\n    message: Optional[str] = Field(default=None, example=\"Run started\")\n    data: Optional[Any] = Field(\n        default=None,\n        description=(\n            \"Data associated with the state, e.g. a result. \"\n            \"Content must be storable as JSON.\"\n        ),\n    )\n    state_details: StateDetails = Field(default_factory=StateDetails)\n\n    @classmethod\n    def from_orm_without_result(\n        cls,\n        orm_state: Union[\n            \"prefect.server.database.orm_models.ORMFlowRunState\",\n            \"prefect.server.database.orm_models.ORMTaskRunState\",\n        ],\n        with_data: Optional[Any] = None,\n    ):\n        \"\"\"\n        During orchestration, ORM states can be instantiated prior to inserting results\n        into the artifact table and the `data` field will not be eagerly loaded. In\n        these cases, sqlalchemy will attempt to lazily load the the relationship, which\n        will fail when called within a synchronous pydantic method.\n\n        This method will construct a `State` object from an ORM model without a loaded\n        artifact and attach data passed using the `with_data` argument to the `data`\n        field.\n        \"\"\"\n\n        field_keys = cls.schema()[\"properties\"].keys()\n        state_data = {\n            field: getattr(orm_state, field, None)\n            for field in field_keys\n            if field != \"data\"\n        }\n        state_data[\"data\"] = with_data\n        return cls(**state_data)\n\n    @validator(\"name\", always=True)\n    def default_name_from_type(cls, v, *, values, **kwargs):\n        \"\"\"If a name is not provided, use the type\"\"\"\n\n        # if `type` is not in `values` it means the `type` didn't pass its own\n        # validation check and an error will be raised after this function is called\n        if v is None and values.get(\"type\"):\n            v = \" \".join([v.capitalize() for v in values.get(\"type\").value.split(\"_\")])\n        return v\n\n    @root_validator\n    def default_scheduled_start_time(cls, values):\n        \"\"\"\n        TODO: This should throw an error instead of setting a default but is out of\n              scope for https://github.com/PrefectHQ/orion/pull/174/ and can be rolled\n              into work refactoring state initialization\n        \"\"\"\n        if values.get(\"type\") == StateType.SCHEDULED:\n            state_details = values.setdefault(\n                \"state_details\", cls.__fields__[\"state_details\"].get_default()\n            )\n            if not state_details.scheduled_time:\n                state_details.scheduled_time = pendulum.now(\"utc\")\n        return values\n\n    def is_scheduled(self) -> bool:\n        return self.type == StateType.SCHEDULED\n\n    def is_pending(self) -> bool:\n        return self.type == StateType.PENDING\n\n    def is_running(self) -> bool:\n        return self.type == StateType.RUNNING\n\n    def is_completed(self) -> bool:\n        return self.type == StateType.COMPLETED\n\n    def is_failed(self) -> bool:\n        return self.type == StateType.FAILED\n\n    def is_crashed(self) -> bool:\n        return self.type == StateType.CRASHED\n\n    def is_cancelled(self) -> bool:\n        return self.type == StateType.CANCELLED\n\n    def is_cancelling(self) -> bool:\n        return self.type == StateType.CANCELLING\n\n    def is_final(self) -> bool:\n        return self.type in TERMINAL_STATES\n\n    def is_paused(self) -> bool:\n        return self.type == StateType.PAUSED\n\n    def copy(self, *, update: dict = None, reset_fields: bool = False, **kwargs):\n        \"\"\"\n        Copying API models should return an object that could be inserted into the\n        database again. The 'timestamp' is reset using the default factory.\n        \"\"\"\n        update = update or {}\n        update.setdefault(\"timestamp\", self.__fields__[\"timestamp\"].get_default())\n        return super().copy(reset_fields=reset_fields, update=update, **kwargs)\n\n    def result(self, raise_on_failure: bool = True, fetch: Optional[bool] = None):\n        # Backwards compatible `result` handling on the server-side schema\n        from prefect.states import State\n\n        warnings.warn(\n            (\n                \"`result` is no longer supported by\"\n                \" `prefect.server.schemas.states.State` and will be removed in a future\"\n                \" release. When result retrieval is needed, use `prefect.states.State`.\"\n            ),\n            DeprecationWarning,\n            stacklevel=2,\n        )\n\n        state = State.parse_obj(self)\n        return state.result(raise_on_failure=raise_on_failure, fetch=fetch)\n\n    def to_state_create(self):\n        # Backwards compatibility for `to_state_create`\n        from prefect.client.schemas import State\n\n        warnings.warn(\n            (\n                \"Use of `prefect.server.schemas.states.State` from the client is\"\n                \" deprecated and support will be removed in a future release. Use\"\n                \" `prefect.states.State` instead.\"\n            ),\n            DeprecationWarning,\n            stacklevel=2,\n        )\n\n        state = State.parse_obj(self)\n        return state.to_state_create()\n\n    def __repr__(self) -> str:\n        \"\"\"\n        Generates a complete state representation appropriate for introspection\n        and debugging, including the result:\n\n        `MyCompletedState(message=\"my message\", type=COMPLETED, result=...)`\n        \"\"\"\n        from prefect.deprecated.data_documents import DataDocument\n\n        if isinstance(self.data, DataDocument):\n            result = self.data.decode()\n        else:\n            result = self.data\n\n        display = dict(\n            message=repr(self.message),\n            type=str(self.type.value),\n            result=repr(result),\n        )\n\n        return f\"{self.name}({', '.join(f'{k}={v}' for k, v in display.items())})\"\n\n    def __str__(self) -> str:\n        \"\"\"\n        Generates a simple state representation appropriate for logging:\n\n        `MyCompletedState(\"my message\", type=COMPLETED)`\n        \"\"\"\n\n        display = []\n\n        if self.message:\n            display.append(repr(self.message))\n\n        if self.type.value.lower() != self.name.lower():\n            display.append(f\"type={self.type.value}\")\n\n        return f\"{self.name}({', '.join(display)})\"\n\n    def __hash__(self) -> int:\n        return hash(\n            (\n                getattr(self.state_details, \"flow_run_id\", None),\n                getattr(self.state_details, \"task_run_id\", None),\n                self.timestamp,\n                self.type,\n            )\n        )\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.State.from_orm_without_result","title":"from_orm_without_result classmethod","text":"

    During orchestration, ORM states can be instantiated prior to inserting results into the artifact table and the data field will not be eagerly loaded. In these cases, sqlalchemy will attempt to lazily load the the relationship, which will fail when called within a synchronous pydantic method.

    This method will construct a State object from an ORM model without a loaded artifact and attach data passed using the with_data argument to the data field.

    Source code in prefect/server/schemas/states.py
    @classmethod\ndef from_orm_without_result(\n    cls,\n    orm_state: Union[\n        \"prefect.server.database.orm_models.ORMFlowRunState\",\n        \"prefect.server.database.orm_models.ORMTaskRunState\",\n    ],\n    with_data: Optional[Any] = None,\n):\n    \"\"\"\n    During orchestration, ORM states can be instantiated prior to inserting results\n    into the artifact table and the `data` field will not be eagerly loaded. In\n    these cases, sqlalchemy will attempt to lazily load the the relationship, which\n    will fail when called within a synchronous pydantic method.\n\n    This method will construct a `State` object from an ORM model without a loaded\n    artifact and attach data passed using the `with_data` argument to the `data`\n    field.\n    \"\"\"\n\n    field_keys = cls.schema()[\"properties\"].keys()\n    state_data = {\n        field: getattr(orm_state, field, None)\n        for field in field_keys\n        if field != \"data\"\n    }\n    state_data[\"data\"] = with_data\n    return cls(**state_data)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.State.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.StateBaseModel","title":"StateBaseModel","text":"

    Bases: IDBaseModel

    Source code in prefect/server/schemas/states.py
    class StateBaseModel(IDBaseModel):\n    def orm_dict(\n        self, *args, shallow: bool = False, json_compatible: bool = False, **kwargs\n    ) -> dict:\n        \"\"\"\n        This method is used as a convenience method for constructing fixtues by first\n        building a `State` schema object and converting it into an ORM-compatible\n        format. Because the `data` field is not writable on ORM states, this method\n        omits the `data` field entirely for the purposes of constructing an ORM model.\n        If state data is required, an artifact must be created separately.\n        \"\"\"\n\n        schema_dict = self.dict(\n            *args, shallow=shallow, json_compatible=json_compatible, **kwargs\n        )\n        # remove the data field in order to construct a state ORM model\n        schema_dict.pop(\"data\", None)\n        return schema_dict\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.StateBaseModel.json","title":"json","text":"

    Returns a representation of the model as JSON.

    If include_secrets=True, then SecretStr and SecretBytes objects are fully revealed. Otherwise they are obfuscated.

    Source code in prefect/server/utilities/schemas/bases.py
    def json(self, *args, include_secrets: bool = False, **kwargs) -> str:\n    \"\"\"\n    Returns a representation of the model as JSON.\n\n    If `include_secrets=True`, then `SecretStr` and `SecretBytes` objects are\n    fully revealed. Otherwise they are obfuscated.\n\n    \"\"\"\n    if include_secrets:\n        if \"encoder\" in kwargs:\n            raise ValueError(\n                \"Alternative encoder provided; can not set encoder for\"\n                \" SecretFields.\"\n            )\n        kwargs[\"encoder\"] = partial(\n            custom_pydantic_encoder,\n            {SecretField: lambda v: v.get_secret_value() if v else None},\n        )\n    return super().json(*args, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.StateType","title":"StateType","text":"

    Bases: AutoEnum

    Enumeration of state types.

    Source code in prefect/server/schemas/states.py
    class StateType(AutoEnum):\n    \"\"\"Enumeration of state types.\"\"\"\n\n    SCHEDULED = AutoEnum.auto()\n    PENDING = AutoEnum.auto()\n    RUNNING = AutoEnum.auto()\n    COMPLETED = AutoEnum.auto()\n    FAILED = AutoEnum.auto()\n    CANCELLED = AutoEnum.auto()\n    CRASHED = AutoEnum.auto()\n    PAUSED = AutoEnum.auto()\n    CANCELLING = AutoEnum.auto()\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.StateType.auto","title":"auto staticmethod","text":"

    Exposes enum.auto() to avoid requiring a second import to use AutoEnum

    Source code in prefect/utilities/collections.py
    @staticmethod\ndef auto():\n    \"\"\"\n    Exposes `enum.auto()` to avoid requiring a second import to use `AutoEnum`\n    \"\"\"\n    return auto()\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.AwaitingRetry","title":"AwaitingRetry","text":"

    Convenience function for creating AwaitingRetry states.

    Returns:

    Name Type Description State State

    a AwaitingRetry state

    Source code in prefect/server/schemas/states.py
    def AwaitingRetry(\n    scheduled_time: datetime.datetime = None, cls: Type[State] = State, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `AwaitingRetry` states.\n\n    Returns:\n        State: a AwaitingRetry state\n    \"\"\"\n    return Scheduled(\n        cls=cls, scheduled_time=scheduled_time, name=\"AwaitingRetry\", **kwargs\n    )\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Cancelled","title":"Cancelled","text":"

    Convenience function for creating Cancelled states.

    Returns:

    Name Type Description State State

    a Cancelled state

    Source code in prefect/server/schemas/states.py
    def Cancelled(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Cancelled` states.\n\n    Returns:\n        State: a Cancelled state\n    \"\"\"\n    return cls(type=StateType.CANCELLED, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Cancelling","title":"Cancelling","text":"

    Convenience function for creating Cancelling states.

    Returns:

    Name Type Description State State

    a Cancelling state

    Source code in prefect/server/schemas/states.py
    def Cancelling(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Cancelling` states.\n\n    Returns:\n        State: a Cancelling state\n    \"\"\"\n    return cls(type=StateType.CANCELLING, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Completed","title":"Completed","text":"

    Convenience function for creating Completed states.

    Returns:

    Name Type Description State State

    a Completed state

    Source code in prefect/server/schemas/states.py
    def Completed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Completed` states.\n\n    Returns:\n        State: a Completed state\n    \"\"\"\n    return cls(type=StateType.COMPLETED, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Crashed","title":"Crashed","text":"

    Convenience function for creating Crashed states.

    Returns:

    Name Type Description State State

    a Crashed state

    Source code in prefect/server/schemas/states.py
    def Crashed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Crashed` states.\n\n    Returns:\n        State: a Crashed state\n    \"\"\"\n    return cls(type=StateType.CRASHED, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Failed","title":"Failed","text":"

    Convenience function for creating Failed states.

    Returns:

    Name Type Description State State

    a Failed state

    Source code in prefect/server/schemas/states.py
    def Failed(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Failed` states.\n\n    Returns:\n        State: a Failed state\n    \"\"\"\n    return cls(type=StateType.FAILED, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Late","title":"Late","text":"

    Convenience function for creating Late states.

    Returns:

    Name Type Description State State

    a Late state

    Source code in prefect/server/schemas/states.py
    def Late(\n    scheduled_time: datetime.datetime = None, cls: Type[State] = State, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `Late` states.\n\n    Returns:\n        State: a Late state\n    \"\"\"\n    return Scheduled(cls=cls, scheduled_time=scheduled_time, name=\"Late\", **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Paused","title":"Paused","text":"

    Convenience function for creating Paused states.

    Returns:

    Name Type Description State State

    a Paused state

    Source code in prefect/server/schemas/states.py
    def Paused(\n    cls: Type[State] = State,\n    timeout_seconds: int = None,\n    pause_expiration_time: datetime.datetime = None,\n    reschedule: bool = False,\n    pause_key: str = None,\n    **kwargs,\n) -> State:\n    \"\"\"Convenience function for creating `Paused` states.\n\n    Returns:\n        State: a Paused state\n    \"\"\"\n    state_details = StateDetails.parse_obj(kwargs.pop(\"state_details\", {}))\n\n    if state_details.pause_timeout:\n        raise ValueError(\"An extra pause timeout was provided in state_details\")\n\n    if pause_expiration_time is not None and timeout_seconds is not None:\n        raise ValueError(\n            \"Cannot supply both a pause_expiration_time and timeout_seconds\"\n        )\n\n    if pause_expiration_time is None and timeout_seconds is None:\n        pass\n    else:\n        state_details.pause_timeout = pause_expiration_time or (\n            pendulum.now(\"UTC\") + pendulum.Duration(seconds=timeout_seconds)\n        )\n\n    state_details.pause_reschedule = reschedule\n    state_details.pause_key = pause_key\n\n    return cls(type=StateType.PAUSED, state_details=state_details, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Pending","title":"Pending","text":"

    Convenience function for creating Pending states.

    Returns:

    Name Type Description State State

    a Pending state

    Source code in prefect/server/schemas/states.py
    def Pending(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Pending` states.\n\n    Returns:\n        State: a Pending state\n    \"\"\"\n    return cls(type=StateType.PENDING, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Retrying","title":"Retrying","text":"

    Convenience function for creating Retrying states.

    Returns:

    Name Type Description State State

    a Retrying state

    Source code in prefect/server/schemas/states.py
    def Retrying(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Retrying` states.\n\n    Returns:\n        State: a Retrying state\n    \"\"\"\n    return cls(type=StateType.RUNNING, name=\"Retrying\", **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Running","title":"Running","text":"

    Convenience function for creating Running states.

    Returns:

    Name Type Description State State

    a Running state

    Source code in prefect/server/schemas/states.py
    def Running(cls: Type[State] = State, **kwargs) -> State:\n    \"\"\"Convenience function for creating `Running` states.\n\n    Returns:\n        State: a Running state\n    \"\"\"\n    return cls(type=StateType.RUNNING, **kwargs)\n
    "},{"location":"api-ref/server/schemas/states/#prefect.server.schemas.states.Scheduled","title":"Scheduled","text":"

    Convenience function for creating Scheduled states.

    Returns:

    Name Type Description State State

    a Scheduled state

    Source code in prefect/server/schemas/states.py
    def Scheduled(\n    scheduled_time: datetime.datetime = None, cls: Type[State] = State, **kwargs\n) -> State:\n    \"\"\"Convenience function for creating `Scheduled` states.\n\n    Returns:\n        State: a Scheduled state\n    \"\"\"\n    # NOTE: `scheduled_time` must come first for backwards compatibility\n\n    state_details = StateDetails.parse_obj(kwargs.pop(\"state_details\", {}))\n    if scheduled_time is None:\n        scheduled_time = pendulum.now(\"UTC\")\n    elif state_details.scheduled_time:\n        raise ValueError(\"An extra scheduled_time was provided in state_details\")\n    state_details.scheduled_time = scheduled_time\n\n    return cls(type=StateType.SCHEDULED, state_details=state_details, **kwargs)\n
    "},{"location":"api-ref/server/services/late_runs/","title":"server.services.late_runs","text":""},{"location":"api-ref/server/services/late_runs/#prefect.server.services.late_runs","title":"prefect.server.services.late_runs","text":"

    The MarkLateRuns service. Responsible for putting flow runs in a Late state if they are not started on time. The threshold for a late run can be configured by changing PREFECT_API_SERVICES_LATE_RUNS_AFTER_SECONDS.

    "},{"location":"api-ref/server/services/late_runs/#prefect.server.services.late_runs.MarkLateRuns","title":"MarkLateRuns","text":"

    Bases: LoopService

    A simple loop service responsible for identifying flow runs that are \"late\".

    A flow run is defined as \"late\" if has not scheduled within a certain amount of time after its scheduled start time. The exact amount is configurable in Prefect REST API Settings.

    Source code in prefect/server/services/late_runs.py
    class MarkLateRuns(LoopService):\n    \"\"\"\n    A simple loop service responsible for identifying flow runs that are \"late\".\n\n    A flow run is defined as \"late\" if has not scheduled within a certain amount\n    of time after its scheduled start time. The exact amount is configurable in\n    Prefect REST API Settings.\n    \"\"\"\n\n    def __init__(self, loop_seconds: float = None, **kwargs):\n        super().__init__(\n            loop_seconds=loop_seconds\n            or PREFECT_API_SERVICES_LATE_RUNS_LOOP_SECONDS.value(),\n            **kwargs,\n        )\n\n        # mark runs late if they are this far past their expected start time\n        self.mark_late_after: datetime.timedelta = (\n            PREFECT_API_SERVICES_LATE_RUNS_AFTER_SECONDS.value()\n        )\n\n        # query for this many runs to mark as late at once\n        self.batch_size = 400\n\n    @inject_db\n    async def run_once(self, db: PrefectDBInterface):\n        \"\"\"\n        Mark flow runs as late by:\n\n        - Querying for flow runs in a scheduled state that are Scheduled to start in the past\n        - For any runs past the \"late\" threshold, setting the flow run state to a new `Late` state\n        \"\"\"\n        scheduled_to_start_before = pendulum.now(\"UTC\").subtract(\n            seconds=self.mark_late_after.total_seconds()\n        )\n\n        while True:\n            async with db.session_context(begin_transaction=True) as session:\n                query = self._get_select_late_flow_runs_query(\n                    scheduled_to_start_before=scheduled_to_start_before, db=db\n                )\n\n                result = await session.execute(query)\n                runs = result.all()\n\n                # mark each run as late\n                for run in runs:\n                    await self._mark_flow_run_as_late(session=session, flow_run=run)\n\n                # if no runs were found, exit the loop\n                if len(runs) < self.batch_size:\n                    break\n\n        self.logger.info(\"Finished monitoring for late runs.\")\n\n    @inject_db\n    def _get_select_late_flow_runs_query(\n        self, scheduled_to_start_before: datetime.datetime, db: PrefectDBInterface\n    ):\n        \"\"\"\n        Returns a sqlalchemy query for late flow runs.\n\n        Args:\n            scheduled_to_start_before: the maximum next scheduled start time of\n                scheduled flow runs to consider in the returned query\n        \"\"\"\n        query = (\n            sa.select(\n                db.FlowRun.id,\n                db.FlowRun.next_scheduled_start_time,\n            )\n            .where(\n                # The next scheduled start time is in the past, including the mark late\n                # after buffer\n                (db.FlowRun.next_scheduled_start_time <= scheduled_to_start_before),\n                db.FlowRun.state_type == states.StateType.SCHEDULED,\n                db.FlowRun.state_name == \"Scheduled\",\n            )\n            .limit(self.batch_size)\n        )\n        return query\n\n    async def _mark_flow_run_as_late(\n        self, session: AsyncSession, flow_run: PrefectDBInterface.FlowRun\n    ) -> None:\n        \"\"\"\n        Mark a flow run as late.\n\n        Pass-through method for overrides.\n        \"\"\"\n        await models.flow_runs.set_flow_run_state(\n            session=session,\n            flow_run_id=flow_run.id,\n            state=states.Late(scheduled_time=flow_run.next_scheduled_start_time),\n            force=True,\n        )\n
    "},{"location":"api-ref/server/services/late_runs/#prefect.server.services.late_runs.MarkLateRuns.run_once","title":"run_once async","text":"

    Mark flow runs as late by:

    • Querying for flow runs in a scheduled state that are Scheduled to start in the past
    • For any runs past the \"late\" threshold, setting the flow run state to a new Late state
    Source code in prefect/server/services/late_runs.py
    @inject_db\nasync def run_once(self, db: PrefectDBInterface):\n    \"\"\"\n    Mark flow runs as late by:\n\n    - Querying for flow runs in a scheduled state that are Scheduled to start in the past\n    - For any runs past the \"late\" threshold, setting the flow run state to a new `Late` state\n    \"\"\"\n    scheduled_to_start_before = pendulum.now(\"UTC\").subtract(\n        seconds=self.mark_late_after.total_seconds()\n    )\n\n    while True:\n        async with db.session_context(begin_transaction=True) as session:\n            query = self._get_select_late_flow_runs_query(\n                scheduled_to_start_before=scheduled_to_start_before, db=db\n            )\n\n            result = await session.execute(query)\n            runs = result.all()\n\n            # mark each run as late\n            for run in runs:\n                await self._mark_flow_run_as_late(session=session, flow_run=run)\n\n            # if no runs were found, exit the loop\n            if len(runs) < self.batch_size:\n                break\n\n    self.logger.info(\"Finished monitoring for late runs.\")\n
    "},{"location":"api-ref/server/services/loop_service/","title":"server.services.loop_service","text":""},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service","title":"prefect.server.services.loop_service","text":"

    The base class for all Prefect REST API loop services.

    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.LoopService","title":"LoopService","text":"

    Loop services are relatively lightweight maintenance routines that need to run periodically.

    This class makes it straightforward to design and integrate them. Users only need to define the run_once coroutine to describe the behavior of the service on each loop.

    Source code in prefect/server/services/loop_service.py
    class LoopService:\n    \"\"\"\n    Loop services are relatively lightweight maintenance routines that need to run periodically.\n\n    This class makes it straightforward to design and integrate them. Users only need to\n    define the `run_once` coroutine to describe the behavior of the service on each loop.\n    \"\"\"\n\n    loop_seconds = 60\n\n    def __init__(self, loop_seconds: float = None, handle_signals: bool = True):\n        \"\"\"\n        Args:\n            loop_seconds (float): if provided, overrides the loop interval\n                otherwise specified as a class variable\n            handle_signals (bool): if True (default), SIGINT and SIGTERM are\n                gracefully intercepted and shut down the running service.\n        \"\"\"\n        if loop_seconds:\n            self.loop_seconds = loop_seconds  # seconds between runs\n        self._should_stop = False  # flag for whether the service should stop running\n        self._is_running = False  # flag for whether the service is running\n        self.name = type(self).__name__\n        self.logger = get_logger(f\"server.services.{self.name.lower()}\")\n\n        if handle_signals:\n            _register_signal(signal.SIGINT, self._stop)\n            _register_signal(signal.SIGTERM, self._stop)\n\n    @inject_db\n    async def _on_start(self, db: PrefectDBInterface) -> None:\n        \"\"\"\n        Called prior to running the service\n        \"\"\"\n        # reset the _should_stop flag\n        self._should_stop = False\n        # set the _is_running flag\n        self._is_running = True\n\n    async def _on_stop(self) -> None:\n        \"\"\"\n        Called after running the service\n        \"\"\"\n        # reset the _is_running flag\n        self._is_running = False\n\n    async def start(self, loops=None) -> None:\n        \"\"\"\n        Run the service `loops` time. Pass loops=None to run forever.\n\n        Args:\n            loops (int, optional): the number of loops to run before exiting.\n        \"\"\"\n\n        await self._on_start()\n\n        i = 0\n        while not self._should_stop:\n            start_time = pendulum.now(\"UTC\")\n\n            try:\n                self.logger.debug(f\"About to run {self.name}...\")\n                await self.run_once()\n\n            except NotImplementedError as exc:\n                raise exc from None\n\n            # if an error is raised, log and continue\n            except Exception as exc:\n                self.logger.error(f\"Unexpected error in: {repr(exc)}\", exc_info=True)\n\n            end_time = pendulum.now(\"UTC\")\n\n            # if service took longer than its loop interval, log a warning\n            # that the interval might be too short\n            if (end_time - start_time).total_seconds() > self.loop_seconds:\n                self.logger.warning(\n                    f\"{self.name} took {(end_time-start_time).total_seconds()} seconds\"\n                    \" to run, which is longer than its loop interval of\"\n                    f\" {self.loop_seconds} seconds.\"\n                )\n\n            # check if early stopping was requested\n            i += 1\n            if loops is not None and i == loops:\n                self.logger.debug(f\"{self.name} exiting after {loops} loop(s).\")\n                await self.stop(block=False)\n\n            # next run is every \"loop seconds\" after each previous run *started*.\n            # note that if the loop took unexpectedly long, the \"next_run\" time\n            # might be in the past, which will result in an instant start\n            next_run = max(\n                start_time.add(seconds=self.loop_seconds), pendulum.now(\"UTC\")\n            )\n            self.logger.debug(f\"Finished running {self.name}. Next run at {next_run}\")\n\n            # check the `_should_stop` flag every 1 seconds until the next run time is reached\n            while pendulum.now(\"UTC\") < next_run and not self._should_stop:\n                await asyncio.sleep(\n                    min(1, (next_run - pendulum.now(\"UTC\")).total_seconds())\n                )\n\n        await self._on_stop()\n\n    async def stop(self, block=True) -> None:\n        \"\"\"\n        Gracefully stops a running LoopService and optionally blocks until the\n        service stops.\n\n        Args:\n            block (bool): if True, blocks until the service is\n                finished running. Otherwise it requests a stop and returns but\n                the service may still be running a final loop.\n\n        \"\"\"\n        self._stop()\n\n        if block:\n            # if block=True, sleep until the service stops running,\n            # but no more than `loop_seconds` to avoid a deadlock\n            with anyio.move_on_after(self.loop_seconds):\n                while self._is_running:\n                    await asyncio.sleep(0.1)\n\n            # if the service is still running after `loop_seconds`, something's wrong\n            if self._is_running:\n                self.logger.warning(\n                    f\"`stop(block=True)` was called on {self.name} but more than one\"\n                    f\" loop interval ({self.loop_seconds} seconds) has passed. This\"\n                    \" usually means something is wrong. If `stop()` was called from\"\n                    \" inside the loop service, use `stop(block=False)` instead.\"\n                )\n\n    def _stop(self, *_) -> None:\n        \"\"\"\n        Private, synchronous method for setting the `_should_stop` flag. Takes arbitrary\n        arguments so it can be used as a signal handler.\n        \"\"\"\n        self._should_stop = True\n\n    async def run_once(self) -> None:\n        \"\"\"\n        Represents one loop of the service.\n\n        Users should override this method.\n\n        To actually run the service once, call `LoopService().start(loops=1)`\n        instead of `LoopService().run_once()`, because this method will not invoke setup\n        and teardown methods properly.\n        \"\"\"\n        raise NotImplementedError(\"LoopService subclasses must implement this method.\")\n
    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.LoopService.run_once","title":"run_once async","text":"

    Represents one loop of the service.

    Users should override this method.

    To actually run the service once, call LoopService().start(loops=1) instead of LoopService().run_once(), because this method will not invoke setup and teardown methods properly.

    Source code in prefect/server/services/loop_service.py
    async def run_once(self) -> None:\n    \"\"\"\n    Represents one loop of the service.\n\n    Users should override this method.\n\n    To actually run the service once, call `LoopService().start(loops=1)`\n    instead of `LoopService().run_once()`, because this method will not invoke setup\n    and teardown methods properly.\n    \"\"\"\n    raise NotImplementedError(\"LoopService subclasses must implement this method.\")\n
    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.LoopService.start","title":"start async","text":"

    Run the service loops time. Pass loops=None to run forever.

    Parameters:

    Name Type Description Default loops int

    the number of loops to run before exiting.

    None Source code in prefect/server/services/loop_service.py
    async def start(self, loops=None) -> None:\n    \"\"\"\n    Run the service `loops` time. Pass loops=None to run forever.\n\n    Args:\n        loops (int, optional): the number of loops to run before exiting.\n    \"\"\"\n\n    await self._on_start()\n\n    i = 0\n    while not self._should_stop:\n        start_time = pendulum.now(\"UTC\")\n\n        try:\n            self.logger.debug(f\"About to run {self.name}...\")\n            await self.run_once()\n\n        except NotImplementedError as exc:\n            raise exc from None\n\n        # if an error is raised, log and continue\n        except Exception as exc:\n            self.logger.error(f\"Unexpected error in: {repr(exc)}\", exc_info=True)\n\n        end_time = pendulum.now(\"UTC\")\n\n        # if service took longer than its loop interval, log a warning\n        # that the interval might be too short\n        if (end_time - start_time).total_seconds() > self.loop_seconds:\n            self.logger.warning(\n                f\"{self.name} took {(end_time-start_time).total_seconds()} seconds\"\n                \" to run, which is longer than its loop interval of\"\n                f\" {self.loop_seconds} seconds.\"\n            )\n\n        # check if early stopping was requested\n        i += 1\n        if loops is not None and i == loops:\n            self.logger.debug(f\"{self.name} exiting after {loops} loop(s).\")\n            await self.stop(block=False)\n\n        # next run is every \"loop seconds\" after each previous run *started*.\n        # note that if the loop took unexpectedly long, the \"next_run\" time\n        # might be in the past, which will result in an instant start\n        next_run = max(\n            start_time.add(seconds=self.loop_seconds), pendulum.now(\"UTC\")\n        )\n        self.logger.debug(f\"Finished running {self.name}. Next run at {next_run}\")\n\n        # check the `_should_stop` flag every 1 seconds until the next run time is reached\n        while pendulum.now(\"UTC\") < next_run and not self._should_stop:\n            await asyncio.sleep(\n                min(1, (next_run - pendulum.now(\"UTC\")).total_seconds())\n            )\n\n    await self._on_stop()\n
    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.LoopService.stop","title":"stop async","text":"

    Gracefully stops a running LoopService and optionally blocks until the service stops.

    Parameters:

    Name Type Description Default block bool

    if True, blocks until the service is finished running. Otherwise it requests a stop and returns but the service may still be running a final loop.

    True Source code in prefect/server/services/loop_service.py
    async def stop(self, block=True) -> None:\n    \"\"\"\n    Gracefully stops a running LoopService and optionally blocks until the\n    service stops.\n\n    Args:\n        block (bool): if True, blocks until the service is\n            finished running. Otherwise it requests a stop and returns but\n            the service may still be running a final loop.\n\n    \"\"\"\n    self._stop()\n\n    if block:\n        # if block=True, sleep until the service stops running,\n        # but no more than `loop_seconds` to avoid a deadlock\n        with anyio.move_on_after(self.loop_seconds):\n            while self._is_running:\n                await asyncio.sleep(0.1)\n\n        # if the service is still running after `loop_seconds`, something's wrong\n        if self._is_running:\n            self.logger.warning(\n                f\"`stop(block=True)` was called on {self.name} but more than one\"\n                f\" loop interval ({self.loop_seconds} seconds) has passed. This\"\n                \" usually means something is wrong. If `stop()` was called from\"\n                \" inside the loop service, use `stop(block=False)` instead.\"\n            )\n
    "},{"location":"api-ref/server/services/loop_service/#prefect.server.services.loop_service.run_multiple_services","title":"run_multiple_services async","text":"

    Only one signal handler can be active at a time, so this function takes a list of loop services and runs all of them with a global signal handler.

    Source code in prefect/server/services/loop_service.py
    async def run_multiple_services(loop_services: List[LoopService]):\n    \"\"\"\n    Only one signal handler can be active at a time, so this function takes a list\n    of loop services and runs all of them with a global signal handler.\n    \"\"\"\n\n    def stop_all_services(self, *_):\n        for service in loop_services:\n            service._stop()\n\n    signal.signal(signal.SIGINT, stop_all_services)\n    signal.signal(signal.SIGTERM, stop_all_services)\n    await asyncio.gather(*[service.start() for service in loop_services])\n
    "},{"location":"api-ref/server/services/scheduler/","title":"server.services.scheduler","text":""},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler","title":"prefect.server.services.scheduler","text":"

    The Scheduler service.

    "},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler.RecentDeploymentsScheduler","title":"RecentDeploymentsScheduler","text":"

    Bases: Scheduler

    A scheduler that only schedules deployments that were updated very recently. This scheduler can run on a tight loop and ensure that runs from newly-created or updated deployments are rapidly scheduled without having to wait for the \"main\" scheduler to complete its loop.

    Note that scheduling is idempotent, so its ok for this scheduler to attempt to schedule the same deployments as the main scheduler. It's purpose is to accelerate scheduling for any deployments that users are interacting with.

    Source code in prefect/server/services/scheduler.py
    class RecentDeploymentsScheduler(Scheduler):\n    \"\"\"\n    A scheduler that only schedules deployments that were updated very recently.\n    This scheduler can run on a tight loop and ensure that runs from\n    newly-created or updated deployments are rapidly scheduled without having to\n    wait for the \"main\" scheduler to complete its loop.\n\n    Note that scheduling is idempotent, so its ok for this scheduler to attempt\n    to schedule the same deployments as the main scheduler. It's purpose is to\n    accelerate scheduling for any deployments that users are interacting with.\n    \"\"\"\n\n    # this scheduler runs on a tight loop\n    loop_seconds = 5\n\n    @inject_db\n    def _get_select_deployments_to_schedule_query(self, db: PrefectDBInterface):\n        \"\"\"\n        Returns a sqlalchemy query for selecting deployments to schedule\n        \"\"\"\n        query = (\n            sa.select(db.Deployment.id)\n            .where(\n                db.Deployment.is_schedule_active.is_(True),\n                db.Deployment.schedule.is_not(None),\n                # use a slightly larger window than the loop interval to pick up\n                # any deployments that were created *while* the scheduler was\n                # last running (assuming the scheduler takes less than one\n                # second to run). Scheduling is idempotent so picking up schedules\n                # multiple times is not a concern.\n                db.Deployment.updated\n                >= pendulum.now(\"UTC\").subtract(seconds=self.loop_seconds + 1),\n            )\n            .order_by(db.Deployment.id)\n            .limit(self.deployment_batch_size)\n        )\n        return query\n
    "},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler.Scheduler","title":"Scheduler","text":"

    Bases: LoopService

    A loop service that schedules flow runs from deployments.

    Source code in prefect/server/services/scheduler.py
    class Scheduler(LoopService):\n    \"\"\"\n    A loop service that schedules flow runs from deployments.\n    \"\"\"\n\n    # the main scheduler takes its loop interval from\n    # PREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS\n    loop_seconds = None\n\n    def __init__(self, loop_seconds: float = None, **kwargs):\n        super().__init__(\n            loop_seconds=(\n                loop_seconds\n                or self.loop_seconds\n                or PREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS.value()\n            ),\n            **kwargs,\n        )\n        self.deployment_batch_size: int = (\n            PREFECT_API_SERVICES_SCHEDULER_DEPLOYMENT_BATCH_SIZE.value()\n        )\n        self.max_runs: int = PREFECT_API_SERVICES_SCHEDULER_MAX_RUNS.value()\n        self.min_runs: int = PREFECT_API_SERVICES_SCHEDULER_MIN_RUNS.value()\n        self.max_scheduled_time: datetime.timedelta = (\n            PREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME.value()\n        )\n        self.min_scheduled_time: datetime.timedelta = (\n            PREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME.value()\n        )\n        self.insert_batch_size = (\n            PREFECT_API_SERVICES_SCHEDULER_INSERT_BATCH_SIZE.value()\n        )\n\n    @inject_db\n    async def run_once(self, db: PrefectDBInterface):\n        \"\"\"\n        Schedule flow runs by:\n\n        - Querying for deployments with active schedules\n        - Generating the next set of flow runs based on each deployments schedule\n        - Inserting all scheduled flow runs into the database\n\n        All inserted flow runs are committed to the database at the termination of the\n        loop.\n        \"\"\"\n        total_inserted_runs = 0\n\n        last_id = None\n        while True:\n            async with db.session_context(begin_transaction=False) as session:\n                query = self._get_select_deployments_to_schedule_query()\n\n                # use cursor based pagination\n                if last_id:\n                    query = query.where(db.Deployment.id > last_id)\n\n                result = await session.execute(query)\n                deployment_ids = result.scalars().unique().all()\n\n                # collect runs across all deployments\n                try:\n                    runs_to_insert = await self._collect_flow_runs(\n                        session=session, deployment_ids=deployment_ids\n                    )\n                except TryAgain:\n                    continue\n\n            # bulk insert the runs based on batch size setting\n            for batch in batched_iterable(runs_to_insert, self.insert_batch_size):\n                async with db.session_context(begin_transaction=True) as session:\n                    inserted_runs = await self._insert_scheduled_flow_runs(\n                        session=session, runs=batch\n                    )\n                    total_inserted_runs += len(inserted_runs)\n\n            # if this is the last page of deployments, exit the loop\n            if len(deployment_ids) < self.deployment_batch_size:\n                break\n            else:\n                # record the last deployment ID\n                last_id = deployment_ids[-1]\n\n        self.logger.info(f\"Scheduled {total_inserted_runs} runs.\")\n\n    @inject_db\n    def _get_select_deployments_to_schedule_query(self, db: PrefectDBInterface):\n        \"\"\"\n        Returns a sqlalchemy query for selecting deployments to schedule.\n\n        The query gets the IDs of any deployments with:\n\n            - an active schedule\n            - EITHER:\n                - fewer than `min_runs` auto-scheduled runs\n                - OR the max scheduled time is less than `max_scheduled_time` in the future\n        \"\"\"\n        now = pendulum.now(\"UTC\")\n        query = (\n            sa.select(db.Deployment.id)\n            .select_from(db.Deployment)\n            # TODO: on Postgres, this could be replaced with a lateral join that\n            # sorts by `next_scheduled_start_time desc` and limits by\n            # `self.min_runs` for a ~ 50% speedup. At the time of writing,\n            # performance of this universal query appears to be fast enough that\n            # this optimization is not worth maintaining db-specific queries\n            .join(\n                db.FlowRun,\n                # join on matching deployments, only picking up future scheduled runs\n                sa.and_(\n                    db.Deployment.id == db.FlowRun.deployment_id,\n                    db.FlowRun.state_type == StateType.SCHEDULED,\n                    db.FlowRun.next_scheduled_start_time >= now,\n                    db.FlowRun.auto_scheduled.is_(True),\n                ),\n                isouter=True,\n            )\n            .where(\n                db.Deployment.is_schedule_active.is_(True),\n                db.Deployment.schedule.is_not(None),\n            )\n            .group_by(db.Deployment.id)\n            # having EITHER fewer than three runs OR runs not scheduled far enough out\n            .having(\n                sa.or_(\n                    sa.func.count(db.FlowRun.next_scheduled_start_time) < self.min_runs,\n                    sa.func.max(db.FlowRun.next_scheduled_start_time)\n                    < now + self.min_scheduled_time,\n                )\n            )\n            .order_by(db.Deployment.id)\n            .limit(self.deployment_batch_size)\n        )\n        return query\n\n    async def _collect_flow_runs(\n        self,\n        session: sa.orm.Session,\n        deployment_ids: List[UUID],\n    ) -> List[Dict]:\n        runs_to_insert = []\n        for deployment_id in deployment_ids:\n            now = pendulum.now(\"UTC\")\n            # guard against erroneously configured schedules\n            try:\n                runs_to_insert.extend(\n                    await self._generate_scheduled_flow_runs(\n                        session=session,\n                        deployment_id=deployment_id,\n                        start_time=now,\n                        end_time=now + self.max_scheduled_time,\n                        min_time=self.min_scheduled_time,\n                        min_runs=self.min_runs,\n                        max_runs=self.max_runs,\n                    )\n                )\n            except Exception:\n                self.logger.exception(\n                    f\"Error scheduling deployment {deployment_id!r}.\",\n                )\n            finally:\n                connection = await session.connection()\n                if connection.invalidated:\n                    # If the error we handled above was the kind of database error that\n                    # causes underlying transaction to rollback and the connection to\n                    # become invalidated, rollback this session.  Errors that may cause\n                    # this are connection drops, database restarts, and things of the\n                    # sort.\n                    #\n                    # This rollback _does not rollback a transaction_, since that has\n                    # actually already happened due to the error above.  It brings the\n                    # Python session in sync with underlying connection so that when we\n                    # exec the outer with block, the context manager will not attempt to\n                    # commit the session.\n                    #\n                    # Then, raise TryAgain to break out of these nested loops, back to\n                    # the outer loop, where we'll begin a new transaction with\n                    # session.begin() in the next loop iteration.\n                    await session.rollback()\n                    raise TryAgain()\n        return runs_to_insert\n\n    @inject_db\n    async def _generate_scheduled_flow_runs(\n        self,\n        session: sa.orm.Session,\n        deployment_id: UUID,\n        start_time: datetime.datetime,\n        end_time: datetime.datetime,\n        min_time: datetime.timedelta,\n        min_runs: int,\n        max_runs: int,\n        db: PrefectDBInterface,\n    ) -> List[Dict]:\n        \"\"\"\n        Given a `deployment_id` and schedule params, generates a list of flow run\n        objects and associated scheduled states that represent scheduled flow runs.\n\n        Pass-through method for overrides.\n\n\n        Args:\n            session: a database session\n            deployment_id: the id of the deployment to schedule\n            start_time: the time from which to start scheduling runs\n            end_time: runs will be scheduled until at most this time\n            min_time: runs will be scheduled until at least this far in the future\n            min_runs: a minimum amount of runs to schedule\n            max_runs: a maximum amount of runs to schedule\n\n        This function will generate the minimum number of runs that satisfy the min\n        and max times, and the min and max counts. Specifically, the following order\n        will be respected:\n\n            - Runs will be generated starting on or after the `start_time`\n            - No more than `max_runs` runs will be generated\n            - No runs will be generated after `end_time` is reached\n            - At least `min_runs` runs will be generated\n            - Runs will be generated until at least `start_time + min_time` is reached\n\n        \"\"\"\n        return await models.deployments._generate_scheduled_flow_runs(\n            session=session,\n            deployment_id=deployment_id,\n            start_time=start_time,\n            end_time=end_time,\n            min_time=min_time,\n            min_runs=min_runs,\n            max_runs=max_runs,\n        )\n\n    @inject_db\n    async def _insert_scheduled_flow_runs(\n        self,\n        session: sa.orm.Session,\n        runs: List[Dict],\n        db: PrefectDBInterface,\n    ) -> List[UUID]:\n        \"\"\"\n        Given a list of flow runs to schedule, as generated by\n        `_generate_scheduled_flow_runs`, inserts them into the database. Note this is a\n        separate method to facilitate batch operations on many scheduled runs.\n\n        Pass-through method for overrides.\n        \"\"\"\n        return await models.deployments._insert_scheduled_flow_runs(\n            session=session, runs=runs\n        )\n
    "},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler.Scheduler.run_once","title":"run_once async","text":"

    Schedule flow runs by:

    • Querying for deployments with active schedules
    • Generating the next set of flow runs based on each deployments schedule
    • Inserting all scheduled flow runs into the database

    All inserted flow runs are committed to the database at the termination of the loop.

    Source code in prefect/server/services/scheduler.py
    @inject_db\nasync def run_once(self, db: PrefectDBInterface):\n    \"\"\"\n    Schedule flow runs by:\n\n    - Querying for deployments with active schedules\n    - Generating the next set of flow runs based on each deployments schedule\n    - Inserting all scheduled flow runs into the database\n\n    All inserted flow runs are committed to the database at the termination of the\n    loop.\n    \"\"\"\n    total_inserted_runs = 0\n\n    last_id = None\n    while True:\n        async with db.session_context(begin_transaction=False) as session:\n            query = self._get_select_deployments_to_schedule_query()\n\n            # use cursor based pagination\n            if last_id:\n                query = query.where(db.Deployment.id > last_id)\n\n            result = await session.execute(query)\n            deployment_ids = result.scalars().unique().all()\n\n            # collect runs across all deployments\n            try:\n                runs_to_insert = await self._collect_flow_runs(\n                    session=session, deployment_ids=deployment_ids\n                )\n            except TryAgain:\n                continue\n\n        # bulk insert the runs based on batch size setting\n        for batch in batched_iterable(runs_to_insert, self.insert_batch_size):\n            async with db.session_context(begin_transaction=True) as session:\n                inserted_runs = await self._insert_scheduled_flow_runs(\n                    session=session, runs=batch\n                )\n                total_inserted_runs += len(inserted_runs)\n\n        # if this is the last page of deployments, exit the loop\n        if len(deployment_ids) < self.deployment_batch_size:\n            break\n        else:\n            # record the last deployment ID\n            last_id = deployment_ids[-1]\n\n    self.logger.info(f\"Scheduled {total_inserted_runs} runs.\")\n
    "},{"location":"api-ref/server/services/scheduler/#prefect.server.services.scheduler.TryAgain","title":"TryAgain","text":"

    Bases: Exception

    Internal control-flow exception used to retry the Scheduler's main loop

    Source code in prefect/server/services/scheduler.py
    class TryAgain(Exception):\n    \"\"\"Internal control-flow exception used to retry the Scheduler's main loop\"\"\"\n
    "},{"location":"api-ref/server/utilities/database/","title":"server.utilities.database","text":""},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database","title":"prefect.server.utilities.database","text":"

    Utilities for interacting with Prefect REST API database and ORM layer.

    Prefect supports both SQLite and Postgres. Many of these utilities allow the Prefect REST API to seamlessly switch between the two.

    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.GenerateUUID","title":"GenerateUUID","text":"

    Bases: FunctionElement

    Platform-independent UUID default generator. Note the actual functionality for this class is specified in the compiles-decorated functions below

    Source code in prefect/server/utilities/database.py
    class GenerateUUID(FunctionElement):\n    \"\"\"\n    Platform-independent UUID default generator.\n    Note the actual functionality for this class is specified in the\n    `compiles`-decorated functions below\n    \"\"\"\n\n    name = \"uuid_default\"\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.JSON","title":"JSON","text":"

    Bases: TypeDecorator

    JSON type that returns SQLAlchemy's dialect-specific JSON types, where possible. Uses generic JSON otherwise.

    The \"base\" type is postgresql.JSONB to expose useful methods prior to SQL compilation

    Source code in prefect/server/utilities/database.py
    class JSON(TypeDecorator):\n    \"\"\"\n    JSON type that returns SQLAlchemy's dialect-specific JSON types, where\n    possible. Uses generic JSON otherwise.\n\n    The \"base\" type is postgresql.JSONB to expose useful methods prior\n    to SQL compilation\n    \"\"\"\n\n    impl = postgresql.JSONB\n    cache_ok = True\n\n    def load_dialect_impl(self, dialect):\n        if dialect.name == \"postgresql\":\n            return dialect.type_descriptor(postgresql.JSONB(none_as_null=True))\n        elif dialect.name == \"sqlite\":\n            return dialect.type_descriptor(sqlite.JSON(none_as_null=True))\n        else:\n            return dialect.type_descriptor(sa.JSON(none_as_null=True))\n\n    def process_bind_param(self, value, dialect):\n        \"\"\"Prepares the given value to be used as a JSON field in a parameter binding\"\"\"\n        if not value:\n            return value\n\n        # PostgreSQL does not support the floating point extrema values `NaN`,\n        # `-Infinity`, or `Infinity`\n        # https://www.postgresql.org/docs/current/datatype-json.html#JSON-TYPE-MAPPING-TABLE\n        #\n        # SQLite supports storing and retrieving full JSON values that include\n        # `NaN`, `-Infinity`, or `Infinity`, but any query that requires SQLite to parse\n        # the value (like `json_extract`) will fail.\n        #\n        # Replace any `NaN`, `-Infinity`, or `Infinity` values with `None` in the\n        # returned value.  See more about `parse_constant` at\n        # https://docs.python.org/3/library/json.html#json.load.\n        return json.loads(json.dumps(value), parse_constant=lambda c: None)\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.Pydantic","title":"Pydantic","text":"

    Bases: TypeDecorator

    A pydantic type that converts inserted parameters to json and converts read values to the pydantic type.

    Source code in prefect/server/utilities/database.py
    class Pydantic(TypeDecorator):\n    \"\"\"\n    A pydantic type that converts inserted parameters to\n    json and converts read values to the pydantic type.\n    \"\"\"\n\n    impl = JSON\n    cache_ok = True\n\n    def __init__(self, pydantic_type, sa_column_type=None):\n        super().__init__()\n        self._pydantic_type = pydantic_type\n        if sa_column_type is not None:\n            self.impl = sa_column_type\n\n    def process_bind_param(self, value, dialect):\n        if value is None:\n            return None\n        # parse the value to ensure it complies with the schema\n        # (this will raise validation errors if not)\n        value = pydantic.parse_obj_as(self._pydantic_type, value)\n        # sqlalchemy requires the bind parameter's value to be a python-native\n        # collection of JSON-compatible objects. we achieve that by dumping the\n        # value to a json string using the pydantic JSON encoder and re-parsing\n        # it into a python-native form.\n        return json.loads(json.dumps(value, default=pydantic.json.pydantic_encoder))\n\n    def process_result_value(self, value, dialect):\n        if value is not None:\n            # load the json object into a fully hydrated typed object\n            return pydantic.parse_obj_as(self._pydantic_type, value)\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.Timestamp","title":"Timestamp","text":"

    Bases: TypeDecorator

    TypeDecorator that ensures that timestamps have a timezone.

    For SQLite, all timestamps are converted to UTC (since they are stored as naive timestamps without timezones) and recovered as UTC.

    Source code in prefect/server/utilities/database.py
    class Timestamp(TypeDecorator):\n    \"\"\"TypeDecorator that ensures that timestamps have a timezone.\n\n    For SQLite, all timestamps are converted to UTC (since they are stored\n    as naive timestamps without timezones) and recovered as UTC.\n    \"\"\"\n\n    impl = sa.TIMESTAMP(timezone=True)\n    cache_ok = True\n\n    def load_dialect_impl(self, dialect):\n        if dialect.name == \"postgresql\":\n            return dialect.type_descriptor(postgresql.TIMESTAMP(timezone=True))\n        elif dialect.name == \"sqlite\":\n            return dialect.type_descriptor(\n                sqlite.DATETIME(\n                    # SQLite is very particular about datetimes, and performs all comparisons\n                    # as alphanumeric comparisons without regard for actual timestamp\n                    # semantics or timezones. Therefore, it's important to have uniform\n                    # and sortable datetime representations. The default is an ISO8601-compatible\n                    # string with NO time zone and a space (\" \") delimiter between the date\n                    # and the time. The below settings can be used to add a \"T\" delimiter but\n                    # will require all other sqlite datetimes to be set similarly, including\n                    # the custom default value for datetime columns and any handwritten SQL\n                    # formed with `strftime()`.\n                    #\n                    # store with \"T\" separator for time\n                    # storage_format=(\n                    #     \"%(year)04d-%(month)02d-%(day)02d\"\n                    #     \"T%(hour)02d:%(minute)02d:%(second)02d.%(microsecond)06d\"\n                    # ),\n                    # handle ISO 8601 with \"T\" or \" \" as the time separator\n                    # regexp=r\"(\\d+)-(\\d+)-(\\d+)[T ](\\d+):(\\d+):(\\d+).(\\d+)\",\n                )\n            )\n        else:\n            return dialect.type_descriptor(sa.TIMESTAMP(timezone=True))\n\n    def process_bind_param(self, value, dialect):\n        if value is None:\n            return None\n        else:\n            if value.tzinfo is None:\n                raise ValueError(\"Timestamps must have a timezone.\")\n            elif dialect.name == \"sqlite\":\n                return pendulum.instance(value).in_timezone(\"UTC\")\n            else:\n                return value\n\n    def process_result_value(self, value, dialect):\n        # retrieve timestamps in their native timezone (or UTC)\n        if value is not None:\n            return pendulum.instance(value).in_timezone(\"utc\")\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.UUID","title":"UUID","text":"

    Bases: TypeDecorator

    Platform-independent UUID type.

    Uses PostgreSQL's UUID type, otherwise uses CHAR(36), storing as stringified hex values with hyphens.

    Source code in prefect/server/utilities/database.py
    class UUID(TypeDecorator):\n    \"\"\"\n    Platform-independent UUID type.\n\n    Uses PostgreSQL's UUID type, otherwise uses\n    CHAR(36), storing as stringified hex values with\n    hyphens.\n    \"\"\"\n\n    impl = TypeEngine\n    cache_ok = True\n\n    def load_dialect_impl(self, dialect):\n        if dialect.name == \"postgresql\":\n            return dialect.type_descriptor(postgresql.UUID())\n        else:\n            return dialect.type_descriptor(CHAR(36))\n\n    def process_bind_param(self, value, dialect):\n        if value is None:\n            return None\n        elif dialect.name == \"postgresql\":\n            return str(value)\n        elif isinstance(value, uuid.UUID):\n            return str(value)\n        else:\n            return str(uuid.UUID(value))\n\n    def process_result_value(self, value, dialect):\n        if value is None:\n            return value\n        else:\n            if not isinstance(value, uuid.UUID):\n                value = uuid.UUID(value)\n            return value\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.date_add","title":"date_add","text":"

    Bases: FunctionElement

    Platform-independent way to add a date and an interval.

    Source code in prefect/server/utilities/database.py
    class date_add(FunctionElement):\n    \"\"\"\n    Platform-independent way to add a date and an interval.\n    \"\"\"\n\n    type = Timestamp()\n    name = \"date_add\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, dt, interval):\n        self.dt = dt\n        self.interval = interval\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.date_diff","title":"date_diff","text":"

    Bases: FunctionElement

    Platform-independent difference of dates. Computes d1 - d2.

    Source code in prefect/server/utilities/database.py
    class date_diff(FunctionElement):\n    \"\"\"\n    Platform-independent difference of dates. Computes d1 - d2.\n    \"\"\"\n\n    type = sa.Interval()\n    name = \"date_diff\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, d1, d2):\n        self.d1 = d1\n        self.d2 = d2\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.interval_add","title":"interval_add","text":"

    Bases: FunctionElement

    Platform-independent way to add two intervals.

    Source code in prefect/server/utilities/database.py
    class interval_add(FunctionElement):\n    \"\"\"\n    Platform-independent way to add two intervals.\n    \"\"\"\n\n    type = sa.Interval()\n    name = \"interval_add\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, i1, i2):\n        self.i1 = i1\n        self.i2 = i2\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.json_contains","title":"json_contains","text":"

    Bases: FunctionElement

    Platform independent json_contains operator, tests if the left expression contains the right expression.

    On postgres this is equivalent to the @> containment operator. https://www.postgresql.org/docs/current/functions-json.html

    Source code in prefect/server/utilities/database.py
    class json_contains(FunctionElement):\n    \"\"\"\n    Platform independent json_contains operator, tests if the\n    `left` expression contains the `right` expression.\n\n    On postgres this is equivalent to the @> containment operator.\n    https://www.postgresql.org/docs/current/functions-json.html\n    \"\"\"\n\n    type = BOOLEAN\n    name = \"json_contains\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, left, right):\n        self.left = left\n        self.right = right\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.json_has_all_keys","title":"json_has_all_keys","text":"

    Bases: FunctionElement

    Platform independent json_has_all_keys operator.

    On postgres this is equivalent to the ?& existence operator. https://www.postgresql.org/docs/current/functions-json.html

    Source code in prefect/server/utilities/database.py
    class json_has_all_keys(FunctionElement):\n    \"\"\"Platform independent json_has_all_keys operator.\n\n    On postgres this is equivalent to the ?& existence operator.\n    https://www.postgresql.org/docs/current/functions-json.html\n    \"\"\"\n\n    type = BOOLEAN\n    name = \"json_has_all_keys\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, json_expr, values: List):\n        self.json_expr = json_expr\n        if isinstance(values, list) and not all(isinstance(v, str) for v in values):\n            raise ValueError(\n                \"json_has_all_key values must be strings if provided as a literal list\"\n            )\n        self.values = values\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.json_has_any_key","title":"json_has_any_key","text":"

    Bases: FunctionElement

    Platform independent json_has_any_key operator.

    On postgres this is equivalent to the ?| existence operator. https://www.postgresql.org/docs/current/functions-json.html

    Source code in prefect/server/utilities/database.py
    class json_has_any_key(FunctionElement):\n    \"\"\"\n    Platform independent json_has_any_key operator.\n\n    On postgres this is equivalent to the ?| existence operator.\n    https://www.postgresql.org/docs/current/functions-json.html\n    \"\"\"\n\n    type = BOOLEAN\n    name = \"json_has_any_key\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = False\n\n    def __init__(self, json_expr, values: List):\n        self.json_expr = json_expr\n        if not all(isinstance(v, str) for v in values):\n            raise ValueError(\"json_has_any_key values must be strings\")\n        self.values = values\n        super().__init__()\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.now","title":"now","text":"

    Bases: FunctionElement

    Platform-independent \"now\" generator.

    Source code in prefect/server/utilities/database.py
    class now(FunctionElement):\n    \"\"\"\n    Platform-independent \"now\" generator.\n    \"\"\"\n\n    type = Timestamp()\n    name = \"now\"\n    # see https://docs.sqlalchemy.org/en/14/core/compiler.html#enabling-caching-support-for-custom-constructs\n    inherit_cache = True\n
    "},{"location":"api-ref/server/utilities/database/#prefect.server.utilities.database.get_dialect","title":"get_dialect","text":"

    Get the dialect of a session, engine, or connection url.

    Primary use case is figuring out whether the Prefect REST API is communicating with SQLite or Postgres.

    Example
    import prefect.settings\nfrom prefect.server.utilities.database import get_dialect\n\ndialect = get_dialect(PREFECT_API_DATABASE_CONNECTION_URL.value())\nif dialect == \"sqlite\":\n    print(\"Using SQLite!\")\nelse:\n    print(\"Using Postgres!\")\n
    Source code in prefect/server/utilities/database.py
    def get_dialect(\n    obj: Union[str, sa.orm.Session, sa.engine.Engine],\n) -> sa.engine.Dialect:\n    \"\"\"\n    Get the dialect of a session, engine, or connection url.\n\n    Primary use case is figuring out whether the Prefect REST API is communicating with\n    SQLite or Postgres.\n\n    Example:\n        ```python\n        import prefect.settings\n        from prefect.server.utilities.database import get_dialect\n\n        dialect = get_dialect(PREFECT_API_DATABASE_CONNECTION_URL.value())\n        if dialect == \"sqlite\":\n            print(\"Using SQLite!\")\n        else:\n            print(\"Using Postgres!\")\n        ```\n    \"\"\"\n    if isinstance(obj, sa.orm.Session):\n        url = obj.bind.url\n    elif isinstance(obj, sa.engine.Engine):\n        url = obj.url\n    else:\n        url = sa.engine.url.make_url(obj)\n\n    return url.get_dialect()\n
    "},{"location":"api-ref/server/utilities/schemas/","title":"server.utilities.schemas","text":""},{"location":"api-ref/server/utilities/schemas/#prefect.server.utilities.schemas","title":"prefect.server.utilities.schemas","text":""},{"location":"api-ref/server/utilities/server/","title":"server.utilities.server","text":""},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server","title":"prefect.server.utilities.server","text":"

    Utilities for the Prefect REST API server.

    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.PrefectAPIRoute","title":"PrefectAPIRoute","text":"

    Bases: APIRoute

    A FastAPIRoute class which attaches an async stack to requests that exits before a response is returned.

    Requests already have request.scope['fastapi_astack'] which is an async stack for the full scope of the request. This stack is used for managing contexts of FastAPI dependencies. If we want to close a dependency before the request is complete (i.e. before returning a response to the user), we need a stack with a different scope. This extension adds this stack at request.state.response_scoped_stack.

    Source code in prefect/server/utilities/server.py
    class PrefectAPIRoute(APIRoute):\n    \"\"\"\n    A FastAPIRoute class which attaches an async stack to requests that exits before\n    a response is returned.\n\n    Requests already have `request.scope['fastapi_astack']` which is an async stack for\n    the full scope of the request. This stack is used for managing contexts of FastAPI\n    dependencies. If we want to close a dependency before the request is complete\n    (i.e. before returning a response to the user), we need a stack with a different\n    scope. This extension adds this stack at `request.state.response_scoped_stack`.\n    \"\"\"\n\n    def get_route_handler(self) -> Callable[[Request], Coroutine[Any, Any, Response]]:\n        default_handler = super().get_route_handler()\n\n        async def handle_response_scoped_depends(request: Request) -> Response:\n            # Create a new stack scoped to exit before the response is returned\n            async with AsyncExitStack() as stack:\n                request.state.response_scoped_stack = stack\n                response = await default_handler(request)\n\n            return response\n\n        return handle_response_scoped_depends\n
    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.PrefectRouter","title":"PrefectRouter","text":"

    Bases: APIRouter

    A base class for Prefect REST API routers.

    Source code in prefect/server/utilities/server.py
    class PrefectRouter(APIRouter):\n    \"\"\"\n    A base class for Prefect REST API routers.\n    \"\"\"\n\n    def __init__(self, **kwargs: Any) -> None:\n        kwargs.setdefault(\"route_class\", PrefectAPIRoute)\n        super().__init__(**kwargs)\n\n    def add_api_route(\n        self, path: str, endpoint: Callable[..., Any], **kwargs: Any\n    ) -> None:\n        \"\"\"\n        Add an API route.\n\n        For routes that return content and have not specified a `response_model`,\n        use return type annotation to infer the response model.\n\n        For routes that return No-Content status codes, explicitly set\n        a `response_class` to ensure nothing is returned in the response body.\n        \"\"\"\n        if kwargs.get(\"status_code\") == status.HTTP_204_NO_CONTENT:\n            # any routes that return No-Content status codes must\n            # explicitly set a response_class that will handle status codes\n            # and not return anything in the body\n            kwargs[\"response_class\"] = Response\n        if kwargs.get(\"response_model\") is None:\n            kwargs[\"response_model\"] = get_type_hints(endpoint).get(\"return\")\n        return super().add_api_route(path, endpoint, **kwargs)\n
    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.PrefectRouter.add_api_route","title":"add_api_route","text":"

    Add an API route.

    For routes that return content and have not specified a response_model, use return type annotation to infer the response model.

    For routes that return No-Content status codes, explicitly set a response_class to ensure nothing is returned in the response body.

    Source code in prefect/server/utilities/server.py
    def add_api_route(\n    self, path: str, endpoint: Callable[..., Any], **kwargs: Any\n) -> None:\n    \"\"\"\n    Add an API route.\n\n    For routes that return content and have not specified a `response_model`,\n    use return type annotation to infer the response model.\n\n    For routes that return No-Content status codes, explicitly set\n    a `response_class` to ensure nothing is returned in the response body.\n    \"\"\"\n    if kwargs.get(\"status_code\") == status.HTTP_204_NO_CONTENT:\n        # any routes that return No-Content status codes must\n        # explicitly set a response_class that will handle status codes\n        # and not return anything in the body\n        kwargs[\"response_class\"] = Response\n    if kwargs.get(\"response_model\") is None:\n        kwargs[\"response_model\"] = get_type_hints(endpoint).get(\"return\")\n    return super().add_api_route(path, endpoint, **kwargs)\n
    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.method_paths_from_routes","title":"method_paths_from_routes","text":"

    Generate a set of strings describing the given routes in the format:

    For example, \"GET /logs/\"

    Source code in prefect/server/utilities/server.py
    def method_paths_from_routes(routes: Iterable[APIRoute]) -> Set[str]:\n    \"\"\"\n    Generate a set of strings describing the given routes in the format: <method> <path>\n\n    For example, \"GET /logs/\"\n    \"\"\"\n    method_paths = set()\n    for route in routes:\n        for method in route.methods:\n            method_paths.add(f\"{method} {route.path}\")\n\n    return method_paths\n
    "},{"location":"api-ref/server/utilities/server/#prefect.server.utilities.server.response_scoped_dependency","title":"response_scoped_dependency","text":"

    Ensure that this dependency closes before the response is returned to the client. By default, FastAPI closes dependencies after sending the response.

    Uses an async stack that is exited before the response is returned. This is particularly useful for database sessions which must be committed before the client can do more work.

    Do not use a response-scoped dependency within a FastAPI background task.

    Background tasks run after FastAPI sends the response, so a response-scoped dependency will already be closed. Use a normal FastAPI dependency instead.

    Parameters:

    Name Type Description Default dependency Callable

    An async callable. FastAPI dependencies may still be used.

    required

    Returns:

    Type Description

    A wrapped dependency which will push the dependency context manager onto

    a stack when called.

    Source code in prefect/server/utilities/server.py
    def response_scoped_dependency(dependency: Callable):\n    \"\"\"\n    Ensure that this dependency closes before the response is returned to the client. By\n    default, FastAPI closes dependencies after sending the response.\n\n    Uses an async stack that is exited before the response is returned. This is\n    particularly useful for database sessions which must be committed before the client\n    can do more work.\n\n    NOTE: Do not use a response-scoped dependency within a FastAPI background task.\n          Background tasks run after FastAPI sends the response, so a response-scoped\n          dependency will already be closed. Use a normal FastAPI dependency instead.\n\n    Args:\n        dependency: An async callable. FastAPI dependencies may still be used.\n\n    Returns:\n        A wrapped `dependency` which will push the `dependency` context manager onto\n        a stack when called.\n    \"\"\"\n    signature = inspect.signature(dependency)\n\n    async def wrapper(*args, request: Request, **kwargs):\n        # Replicate FastAPI behavior of auto-creating a context manager\n        if inspect.isasyncgenfunction(dependency):\n            context_manager = asynccontextmanager(dependency)\n        else:\n            context_manager = dependency\n\n        # Ensure request is provided if requested\n        if \"request\" in signature.parameters:\n            kwargs[\"request\"] = request\n\n        # Enter the route handler provided stack that is closed before responding,\n        # return the value yielded by the wrapped dependency\n        return await request.state.response_scoped_stack.enter_async_context(\n            context_manager(*args, **kwargs)\n        )\n\n    # Ensure that the signature includes `request: Request` to ensure that FastAPI will\n    # inject the request as a dependency; maintain the old signature so those depends\n    # work\n    request_parameter = inspect.signature(wrapper).parameters[\"request\"]\n    functools.update_wrapper(wrapper, dependency)\n\n    if \"request\" not in signature.parameters:\n        new_parameters = signature.parameters.copy()\n        new_parameters[\"request\"] = request_parameter\n        wrapper.__signature__ = signature.replace(\n            parameters=tuple(new_parameters.values())\n        )\n\n    return wrapper\n
    "},{"location":"cloud/","title":"Welcome to Prefect Cloud","text":"

    Prefect Cloud is a hosted workflow application framework that provides all the capabilities of Prefect server plus additional features, such as:

    • automations, events, and webhooks so you can create event-driven workflows
    • workspaces, RBAC, SSO, audit logs and related user management tools for collaboration
    • push work pools for running flows on serverless infrastructure without a worker
    • error summaries powered by Marvin AI to help you resolve errors faster

    Getting Started with Prefect Cloud

    Ready to jump right in and start running with Prefect Cloud? See the Quickstart and follow the instructions on the Cloud tabs to write and deploy your first Prefect Cloud-monitored flow run.

    Prefect Cloud includes all the features in the open-source Prefect server plus the following:

    Prefect Cloud features

    • User accounts \u2014 personal accounts for working in Prefect Cloud.
    • Workspaces \u2014 isolated environments to organize your flows, deployments, and flow runs.
    • Automations \u2014 configure triggers, actions, and notifications in response to real-time monitoring events.
    • Email notifications \u2014 send email alerts from Prefect's server based on automation triggers.
    • Service accounts \u2014 configure API access for running workers or executing flow runs on remote infrastructure.
    • Custom role-based access controls (RBAC) \u2014 assign users granular permissions to perform certain activities within an account or a workspace.
    • Single Sign-on (SSO) \u2014 authentication using your identity provider.
    • Audit Log \u2014 a record of user activities to monitor security and compliance.
    • Collaboration \u2014 invite other people to your account.
    • Error summaries \u2014 (enabled by Marvin AI) distill the error logs of Failed and Crashed flow runs into actionable information.
    • Push work pools \u2014 run flows on your serverless infrastructure without running a worker.
    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#user-accounts","title":"User accounts","text":"

    When you sign up for Prefect Cloud, an account and a user profile are automatically provisioned for you.

    Your profile is the place where you'll manage settings related to yourself as a user, including:

    • Profile, including profile handle and image
    • API keys
    • Preferences, including timezone and color mode

    As an account Admin, you will also have access to account settings from the Account Settings page, such as:

    • Members
    • Workspaces
    • Roles

    As an account Admin you can create a workspace and invite other individuals to your workspace.

    Upgrading from a Prefect Cloud Free tier plan to a Pro or Enterprise tier plan enables additional functionality for adding workspaces, managing teams, and running higher volume workloads.

    Workspace Admins have the ability to use single sign-on (SSO), set role-based access controls (RBAC), view Audit Logs, and configure service accounts.

    Enterprise add custom roles, object-level access control lists, teams, and Directory Sync/SCIM provisioning for SSO.

    Prefect Cloud plans for teams of every size

    See the Prefect Cloud plans for details on Pro and Enterprise account tiers.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#workspaces","title":"Workspaces","text":"

    A workspace is an isolated environment within Prefect Cloud for your flows, deployments, and block configuration. See the Workspaces documentation for more information about configuring and using workspaces.

    Each workspace keeps track of its own:

    • Flow runs and task runs executed in an environment that is syncing with the workspace
    • Flows associated with flow runs and deployments observed by the Prefect Cloud API
    • Deployments
    • Work pools
    • Blocks and storage
    • Automations

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#events","title":"Events","text":"

    Prefect Cloud allows you to see your events. Events provide information about the state of your workflows, and can be used to trigger automations.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#automations","title":"Automations","text":"

    Prefect Cloud automations provide additional notification capabilities beyond those in a self-hosted open-source Prefect server. Automations also enable you to create event-driven workflows.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#incidents","title":"Incidents","text":"

    Prefect Cloud's incidents help teams identify, rectify, and document issues in mission-critical workflows. Incidents are formal declarations of disruptions to a workspace. With automations), activity in that workspace can be paused when an incident is created and resumed when it is resolved.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#error-summaries","title":"Error summaries","text":"

    Prefect Cloud error summaries, enabled by Marvin AI, distill the error logs of Failed and Crashed flow runs into actionable information. To enable this feature and others powered by Marvin AI, visit the Settings page for your account.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#service-accounts","title":"Service accounts","text":"

    Service accounts enable you to create Prefect Cloud API keys that are not associated with a user account. Service accounts are typically used to configure API access for running workers or executing flow runs on remote infrastructure. See the service accounts documentation for more information about creating and managing service accounts.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#roles-and-custom-permissions","title":"Roles and custom permissions","text":"

    Role-based access controls (RBAC) enable you to assign users a role with permissions to perform certain activities within an account or a workspace. See the role-based access controls (RBAC) documentation for more information about managing user roles in a Prefect Cloud account.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#single-sign-on-sso","title":"Single Sign-on (SSO)","text":"

    Prefect Cloud's Pro and Enterprise plans offer single sign-on (SSO) authentication integration with your team\u2019s identity provider. SSO integration can bet set up with identity providers that support OIDC and SAML. Directory Sync and SCIM provisioning is also available with Enterprise plans.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#audit-log","title":"Audit log","text":"

    Prefect Cloud's Pro and Enterprise plans offer Audit Logs for compliance and security. Audit logs provide a chronological record of activities performed by users in an account.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#prefect-cloud-rest-api","title":"Prefect Cloud REST API","text":"

    The Prefect REST API is used for communicating data from Prefect clients to Prefect Cloud or a local Prefect server for orchestration and monitoring. This API is mainly consumed by Prefect clients like the Prefect Python Client or the Prefect UI.

    Prefect Cloud REST API interactive documentation

    Prefect Cloud REST API documentation is available at https://app.prefect.cloud/api/docs.

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/#start-using-prefect-cloud","title":"Start using Prefect Cloud","text":"

    To create an account or sign in with an existing Prefect Cloud account, go to https://app.prefect.cloud/.

    Then follow the steps in the UI to deploy your first Prefect Cloud-monitored flow run. For more details, see the Prefect Quickstart and follow the instructions on the Cloud tabs.

    Need help?

    Get your questions answered by a Prefect Product Advocate! Book a Meeting

    ","tags":["UI","dashboard","orchestration","Prefect Cloud","accounts","teams","workspaces","PaaS"],"boost":2},{"location":"cloud/cloud-quickstart/","title":"Getting Started with Prefect Cloud","text":"

    Get started with Prefect Cloud in just a few steps:

    1. Sign in or register a Prefect Cloud account.
    2. Create a workspace for your account.
    3. Install Prefect in your local environment.
    4. Log into Prefect Cloud from a local terminal session.
    5. Run a flow locally and view flow run execution in Prefect Cloud.
    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#sign-in-or-register","title":"Sign in or register","text":"

    To sign in with an existing account or register an account, go to https://app.prefect.cloud/.

    You can create an account with any of the following:

    • Google account
    • Microsoft account
    • GitHub account
    • Email

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#create-a-workspace","title":"Create a workspace","text":"

    A workspace is an isolated environment within Prefect Cloud for your flows and deployments. You can use workspaces to organize or compartmentalize your workflows.

    When you register a new account, you'll be prompted to provide a name and description for your workspace.

    Note that the Owner setting applies only to users who are members of Prefect Cloud accounts and have permission to create workspaces within account.

    Select Create to create the workspace. If you change your mind, select Edit from the options menu to modify the workspace details or to delete it.

    The Workspace Settings page for your new workspace displays the commands that enable you to install Prefect and log into Prefect Cloud in a local execution environment.

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#install-prefect","title":"Install Prefect","text":"

    Configure a local execution environment to use Prefect Cloud as the API server for flow runs. In other words, \"log in\" to Prefect Cloud from a local environment where you want to run a flow.

    Open a new terminal session.

    Install Prefect in the environment in which you want to execute flow runs.

    pip install -U prefect\n

    Installation requirements

    Prefect requires Python 3.8 or later. If you have any questions about Prefect installations requirements or dependencies in your preferred development environment, check out the Installation documentation.

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#log-into-prefect-cloud-from-a-terminal","title":"Log into Prefect Cloud from a terminal","text":"

    Use the prefect cloud login Prefect CLI command to log into Prefect Cloud from your environment.

    prefect cloud login\n

    The prefect cloud login command, used on its own, provides an interactive login experience. Using this command, you may log in with either an API key or through a browser.

    ? How would you like to authenticate? [Use arrows to move; enter to select]\n> Log in with a webb browser                                                \n  Paste an API key                                                         \nOpening browser...\nWaiting for response...\nAuthenticated with Prefect Cloud! Using workspace 'jeffdc/prod'.\n

    If you choose to log in via the browser, Prefect opens a new tab in your default browser and enables you to log in and authenticate the session.

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#run-a-flow-with-prefect-cloud","title":"Run a flow with Prefect Cloud","text":"

    You're all set to run a flow locally, orchestrated with Prefect Cloud.

    In your local environment, where you configured the previous steps, create a file named quickstart_flow.py with the following contents:

    from prefect import flow\n\n@flow(log_prints=True)\ndef quickstart_flow():\n    print(\"Local quickstart flow is running!\")\n\nif __name__ == \"__main__\":\n    quickstart_flow()\n

    Now run quickstart_flow.py. You'll see log messages like this in your terminal, indicating that the flow is running correctly:

    17:18:09.863 | INFO    | prefect.engine - Created flow run 'fragrant-quetzal' for flow 'quickstart-flow'\n17:18:09.864 | INFO    | Flow run 'fragrant-quetzal' - View at https://app.prefect.cloud/account/my_workspace_id/workspace/my_flow_id/flow-runs/flow-run/my_flow_run_id\n17:18:10.010 | INFO    | Flow run 'fragrant-quetzal' - Local quickstart flow is running!\n17:18:10.144 | INFO    | Flow run 'fragrant-quetzal' - Finished in state Completed()\n

    Go to the Flow Runs pages in your workspace in Prefect Cloud. You'll see the flow run results right there in Prefect Cloud!

    Prefect Cloud automatically tracks any flow runs in a local execution environment logged into Prefect Cloud.

    Select the name of the flow run to see details about this run.

    Congratulations! You successfully ran a local flow and, because you're logged into Prefect Cloud, the local flow run results were captured by Prefect Cloud.

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/cloud-quickstart/#next-steps","title":"Next steps","text":"

    If you're new to Prefect, learn more about writing and running flows in the Prefect Flows First Steps tutorial. If you're already familiar with flows, try creating a deployment and triggering flow runs with Prefect Cloud by following the Deployments tutorial.

    Want to learn more about the features available in Prefect Cloud? Start with the Prefect Cloud Overview.

    If you ran into any issues getting your first flow run with Prefect Cloud working, please join our community to ask questions or provide feedback:

    Prefect's Slack Community is helpful, friendly, and fast growing - come say hi!

    ","tags":["UI","dashboard","Prefect Cloud","quickstart","workspaces","tutorial","getting started"],"boost":2},{"location":"cloud/connecting/","title":"Connecting & Troubleshooting Prefect Cloud","text":"

    To create flow runs in a local or remote execution environment and use either Prefect Cloud or a Prefect server as the backend API server, you need to

    • Configure the execution environment with the location of the API.
    • Authenticate with the API, either by logging in or providing a valid API key (Prefect Cloud only).
    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#log-into-prefect-cloud-from-a-terminal","title":"Log into Prefect Cloud from a terminal","text":"

    Configure a local execution environment to use Prefect Cloud as the API server for flow runs. In other words, \"log in\" to Prefect Cloud from a local environment where you want to run a flow.

    1. Open a new terminal session.
    2. Install Prefect in the environment in which you want to execute flow runs.
    $ pip install -U prefect\n
    1. Use the prefect cloud login Prefect CLI command to log into Prefect Cloud from your environment.
    $ prefect cloud login\n

    The prefect cloud login command, used on its own, provides an interactive login experience. Using this command, you can log in with either an API key or through a browser.

    $ prefect cloud login\n? How would you like to authenticate? [Use arrows to move; enter to select]\n> Log in with a web browser\n    Paste an API key\nPaste your authentication key:\n? Which workspace would you like to use? [Use arrows to move; enter to select]\n> prefect/terry-prefect-workspace\n    g-gadflow/g-workspace\nAuthenticated with Prefect Cloud! Using workspace 'prefect/terry-prefect-workspace'.\n

    You can also log in by providing a Prefect Cloud API key that you create.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#change-workspaces","title":"Change workspaces","text":"

    If you need to change which workspace you're syncing with, use the prefect cloud workspace set Prefect CLI command while logged in, passing the account handle and workspace name.

    $ prefect cloud workspace set --workspace \"prefect/my-workspace\"\n

    If no workspace is provided, you will be prompted to select one.

    Workspace Settings also shows you the prefect cloud workspace set Prefect CLI command you can use to sync a local execution environment with a given workspace.

    You may also use the prefect cloud login command with the --workspace or -w option to set the current workspace.

    $ prefect cloud login --workspace \"prefect/my-workspace\"\n
    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#manually-configure-prefect-api-settings","title":"Manually configure Prefect API settings","text":"

    You can also manually configure the PREFECT_API_URL setting to specify the Prefect Cloud API.

    For Prefect Cloud, you can configure the PREFECT_API_URL and PREFECT_API_KEY settings to authenticate with Prefect Cloud by using an account ID, workspace ID, and API key.

    $ prefect config set PREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]\"\n$ prefect config set PREFECT_API_KEY=\"[API-KEY]\"\n

    When you're in a Prefect Cloud workspace, you can copy the PREFECT_API_URL value directly from the page URL.

    In this example, we configured PREFECT_API_URL and PREFECT_API_KEY in the default profile. You can use prefect profile CLI commands to create settings profiles for different configurations. For example, you could have a \"cloud\" profile configured to use the Prefect Cloud API URL and API key, and another \"local\" profile for local development using a local Prefect API server started with prefect server start. See Settings for details.

    Environment variables

    You can also set PREFECT_API_URL and PREFECT_API_KEY as you would any other environment variable. See Overriding defaults with environment variables for more information.

    See the Flow orchestration with Prefect tutorial for examples.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#install-requirements-in-execution-environments","title":"Install requirements in execution environments","text":"

    In local and remote execution environments \u2014 such as VMs and containers \u2014 you must make sure any flow requirements or dependencies have been installed before creating a flow run.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#troubleshooting-prefect-cloud","title":"Troubleshooting Prefect Cloud","text":"

    This section provides tips that may be helpful if you run into problems using Prefect Cloud.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#prefect-cloud-and-proxies","title":"Prefect Cloud and proxies","text":"

    Proxies intermediate network requests between a server and a client.

    To communicate with Prefect Cloud, the Prefect client library makes HTTPS requests. These requests are made using the httpx Python library. httpx respects accepted proxy environment variables, so the Prefect client is able to communicate through proxies.

    To enable communication via proxies, simply set the HTTPS_PROXY and SSL_CERT_FILE environment variables as appropriate in your execution environment and things should \u201cjust work.\u201d

    See the Using Prefect Cloud with proxies topic in Prefect Discourse for examples of proxy configuration.

    URLs that should be whitelisted for outbound-communication in a secure environment include the UI, the API, Authentication, and the current OCSP server:

    • app.prefect.cloud
    • api.prefect.cloud
    • auth.workos.com
    • api.github.com
    • github.com
    • ocsp.pki.goog/s/gts1d4/OxYEb8XcYmo
    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#prefect-cloud-access-via-api","title":"Prefect Cloud access via API","text":"

    If the Prefect Cloud API key, environment variable settings, or account login for your execution environment are not configured correctly, you may experience errors or unexpected flow run results when using Prefect CLI commands, running flows, or observing flow run results in Prefect Cloud.

    Use the prefect config view CLI command to make sure your execution environment is correctly configured to access Prefect Cloud.

    $ prefect config view\nPREFECT_PROFILE='cloud'\nPREFECT_API_KEY='pnu_xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx' (from profile)\nPREFECT_API_URL='https://api.prefect.cloud/api/accounts/...' (from profile)\n

    Make sure PREFECT_API_URL is configured to use https://api.prefect.cloud/api/....

    Make sure PREFECT_API_KEY is configured to use a valid API key.

    You can use the prefect cloud workspace ls CLI command to view or set the active workspace.

    $ prefect cloud workspace ls\n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503   Available Workspaces: \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502   g-gadflow/g-workspace \u2502\n\u2502    * prefect/workinonit \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n    * active workspace\n

    You can also check that the account and workspace IDs specified in the URL for PREFECT_API_URL match those shown in the URL bar for your Prefect Cloud workspace.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/connecting/#prefect-cloud-login-errors","title":"Prefect Cloud login errors","text":"

    If you're having difficulty logging in to Prefect Cloud, the following troubleshooting steps may resolve the issue, or will provide more information when sharing your case to the support channel.

    • Are you logging into Prefect Cloud 2? Prefect Cloud 1 and Prefect Cloud 2 use separate accounts. Make sure to use the right Prefect Cloud 2 URL: https://app.prefect.cloud/
    • Do you already have a Prefect Cloud account? If you\u2019re having difficulty accepting an invitation, try creating an account first using the email associated with the invitation, then accept the invitation.
    • Are you using a single sign-on (SSO) provider, social authentication (Google, Microsoft, or GitHub) or just using an emailed link?

    Other tips to help with login difficulties:

    • Hard refresh your browser with Cmd+Shift+R.
    • Try in a different browser. We actively test against the following browsers:
    • Chrome
    • Edge
    • Firefox
    • Safari
    • Clear recent browser history/cookies

    None of this worked?

    Email us at help@prefect.io and provide answers to the questions above in your email to make it faster to troubleshoot and unblock you. Make sure you add the email address with which you were trying to log in, your Prefect Cloud account name, and, if applicable, the organization to which it belongs.

    ","tags":["Prefect Cloud","API keys","configuration","workers","troubleshooting","connecting"],"boost":2},{"location":"cloud/incidents/","title":"Incidents","text":"","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#overview","title":"Overview","text":"

    Incidents in Prefect Cloud is an advanced feature designed to optimize the management of workflow disruptions. It serves as a proactive tool for data-driven teams, helping them identify, rectify, and document issues in mission-critical workflows. This system enhances operational efficiency by automating the incident management process and providing a centralized platform for collaboration and compliance.

    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#what-are-incidents","title":"What are incidents?","text":"

    Incidents are formal declarations of disruptions to a workspace. With automations, activity in that workspace can be paused when an incident is created and resumed when it is resolved.

    Incidents vary in nature and severity, ranging from minor glitches to critical system failures. Prefect Cloud now enables users to effectively and automatically track and manage these incidents, ensuring minimal impact on operational continuity.

    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#why-use-incident-management","title":"Why use incident management?","text":"
    1. Automated detection and reporting: Incidents can be automatically identified based on specific triggers or manually reported by team members, facilitating prompt response.

    2. Collaborative problem-solving: The platform fosters collaboration, allowing team members to share insights, discuss resolutions, and track contributions.

    3. Comprehensive impact assessment: Users gain insights into the incident's influence on workflows, helping in prioritizing response efforts.

    4. Compliance with incident management processes: Detailed documentation and reporting features support compliance with incident management systems.

    5. Enhanced operational transparency: The system provides a transparent view of both ongoing and resolved incidents, promoting accountability and continuous improvement.

    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#how-to-use-incident-management-in-prefect-cloud","title":"How to use incident management in Prefect Cloud","text":"","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#creating-an-incident","title":"Creating an incident","text":"

    There are several ways to create an incident:

    1. From the Incidents page:

      • Click on the + button.
      • Fill in required fields and attach any Prefect resources related to your incident.
    2. From a flow run, work pool, or block:

      • Initiate an incident directly from a failed flow run, automatically linking it as a resource, by clicking on the menu button and selecting \"Declare an incident\".
    3. Via an automation:

      • Set up incident creation as an automated response to selected triggers.
    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#incident-automations","title":"Incident automations","text":"

    Automations can be used for triggering an incident and for selecting actions to take when an incident is triggered. For example, a work pool status change could trigger the declaration of an incident, or a critical level incident could trigger a notification action.

    To automatically take action when an incident is declared, set up a custom trigger that listens for declaration events.

    {\n  \"match\": {\n    \"prefect.resource.id\": \"prefect-cloud.incident.*\"\n  },\n  \"expect\": [\n    \"prefect-cloud.incident.declared\"\n  ],\n  \"posture\": \"Reactive\",\n  \"threshold\": 1,\n  \"within\": 0\n}\n

    Building custom triggers

    To get started with incident automations, you only need to specify two fields in your trigger:

    • match: The resource emitting your event of interest. You can match on specific resource IDs, use wildcards to match on all resources of a given type, and even match on other resource attributes, like prefect.resource.name.

    • expect: The event type to listen for. For example, you could listen for any (or all) of the following event types:

      • prefect-cloud.incident.declared
      • prefect-cloud.incident.resolved
      • prefect-cloud.incident.updated.severity

    See Event Triggers for more information on custom triggers, and check out your Event Feed to see the event types emitted by your incidents and other resources (i.e. events that you can react to).

    When an incident is declared, any actions you configure such as pausing work pools or sending notifications, will execute immediately.

    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#managing-an-incident","title":"Managing an incident","text":"
    • Monitor active incidents: View real-time status, severity, and impact.
    • Adjust incident details: Update status, severity, and other relevant information.
    • Collaborate: Add comments and insights; these will display with user identifiers and timestamps.
    • Impact assessment: Evaluate how the incident affects ongoing and future workflows.
    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#resolving-and-documenting-incidents","title":"Resolving and documenting incidents","text":"
    • Resolution: Update the incident status to reflect resolution steps taken.
    • Documentation: Ensure all actions, comments, and changes are logged for future reference.
    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/incidents/#incident-reporting","title":"Incident reporting","text":"
    • Generate a detailed timeline of the incident: actions taken, updates to severity and resolution - suitable for compliance and retrospective analysis.
    ","tags":["UI","flow runs","triggers","Prefect Cloud"],"boost":2},{"location":"cloud/rate-limits/","title":"API Rate Limits & Retention Periods","text":"

    API rate limits restrict the number of requests that a single client can make in a given time period. They ensure Prefect Cloud's stability, so that when you make an API call, you always get a response.

    Prefect Cloud rate limits are subject to change

    The following rate limits are in effect currently, but are subject to change. Contact Prefect support at help@prefect.io if you have questions about current rate limits.

    Prefect Cloud enforces the following rate limits:

    • Flow and task creation rate limits
    • Log service rate limits
    ","tags":["API","Prefect Cloud","rate limits"],"boost":2},{"location":"cloud/rate-limits/#flow-flow-run-and-task-run-rate-limits","title":"Flow, flow run, and task run rate limits","text":"

    Prefect Cloud limits the flow_runs, task_runs, and flows endpoints and their subroutes at the following levels:

    • 400 per minute for personal accounts
    • 2,000 per minute for Pro accounts

    The Prefect Cloud API will return a 429 response with an appropriate Retry-After header if these limits are triggered.

    ","tags":["API","Prefect Cloud","rate limits"],"boost":2},{"location":"cloud/rate-limits/#log-service-rate-limits","title":"Log service rate limits","text":"

    Prefect Cloud limits the number of logs accepted:

    • 700 logs per minute for personal accounts
    • 10,000 logs per minute for Pro accounts

    The Prefect Cloud API will return a 429 response if these limits are triggered.

    ","tags":["API","Prefect Cloud","rate limits"],"boost":2},{"location":"cloud/rate-limits/#flow-run-retention","title":"Flow run retention","text":"

    Prefect Cloud feature

    The Flow Run Retention Policy setting is only applicable in Prefect Cloud.

    Flow runs in Prefect Cloud are retained according to the Flow Run Retention Policy set by your account tier. The policy setting applies to all workspaces owned by the account.

    The flow run retention policy represents the number of days each flow run is available in the Prefect Cloud UI, and via the Prefect CLI and API after it ends. Once a flow run reaches a terminal state (detailed in the chart here), it will be retained until the end of the flow run retention period.

    Flow Run Retention Policy keys on terminal state

    Note that, because Flow Run Retention Policy keys on terminal state, if two flows start at the same time, but reach a terminal state at different times, they will be removed at different times according to when they each reached their respective terminal states.

    This retention policy applies to all details about a flow run, including its task runs. Subflow runs follow the retention policy independently from their parent flow runs, and are removed based on the time each subflow run reaches a terminal state.

    If you or your organization have needs that require a tailored retention period, contact the Prefect Sales team.

    ","tags":["API","Prefect Cloud","rate limits"],"boost":2},{"location":"cloud/workspaces/","title":"Workspaces","text":"

    A workspace is a discrete environment within Prefect Cloud for your workflows and blocks. Workspaces are available to Prefect Cloud accounts only.

    Workspaces can be used to organize and compartmentalize your workflows. For example, you can use separate workspaces to isolate dev, staging, and prod environments, or to provide separation between different teams.

    When you first log into Prefect Cloud, you will be prompted to create your own initial workspace. After creating your workspace, you'll be able to view flow runs, flows, deployments, and other workspace-specific features in the Prefect Cloud UI.

    Select a workspace name in the navigation menu to see all workspaces you can access.

    Your list of available workspaces may include:

    • Your own account's workspace.
    • Workspaces in an account to which you've been invited and have been given access as an Admin or Member.

    Workspace-specific features

    Each workspace keeps track of its own:

    • Flow runs and task runs executed in an environment that is syncing with the workspace
    • Flows associated with flow runs or deployments observed by the Prefect Cloud API
    • Deployments
    • Work pools
    • Blocks and Storage
    • Automations

    Your user permissions within workspaces may vary. Account admins can assign roles and permissions at the workspace level.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#create-a-workspace","title":"Create a workspace","text":"

    On the Account Workspaces dropdown or the Workspaces page select the + icon to create a new workspace.

    You'll be prompted to configure:

    • The Workspace Owner from the dropdown account menu options.
    • The Workspace Name must be unique within the account.
    • An optional description for the workspace.

    Select Create to create the new workspace. The number of available workspaces varies by Prefect Cloud plan. See Pricing if you need additional workspaces or users.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#workspace-settings","title":"Workspace settings","text":"

    Within a workspace, select Settings -> General to view or edit workspace details.

    On this page you can edit workspace details or delete the workspace.

    Deleting a workspace

    Deleting a workspace deletes all deployments, flow run history, work pools, and notifications configured in workspace.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#workspace-access","title":"Workspace access","text":"

    Within a Prefect Cloud Pro or Enterprise tier account, Workspace Owners can invite other people to be members and provision service accounts to a workspace. In addition to giving the user access to the workspace, a Workspace Owner assigns a workspace role to the user. The role specifies the scope of permissions for the user within the workspace.

    As a Workspace Owner, select Workspaces -> Sharing to manage members and service accounts for the workspace.

    If you've previously invited individuals to your account or provisioned service accounts, you'll see them listed here.

    To invite someone to an account, select the Members + icon. You can select from a list of existing account members.

    Select a Role for the user. This will be the initial role for the user within the workspace. A workspace Owner can change this role at any time.

    Select Send to initiate the invitation.

    To add a service account to a workspace, select the Service Accounts + icon. You can select from a list of configured service accounts. Select a Workspace Role for the service account. This will be the initial role for the service account within the workspace. A workspace Owner can change this role at any time. Select Share to finalize adding the service account.

    To remove a workspace member or service account, select Remove from the menu on the right side of the user or service account information on this page.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#workspace-transfer","title":"Workspace transfer","text":"

    Workspace transfer enables you to move an existing workspace from one account to another.

    Workspace transfer retains existing workspace configuration and flow run history, including blocks, deployments, notifications, work pools, and logs.

    Workspace transfer permissions

    Workspace transfer must be initiated or approved by a user with admin privileges for the workspace to be transferred.

    To initiate a workspace transfer between personal accounts, contact support@prefect.io.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/workspaces/#transfer-a-workspace","title":"Transfer a workspace","text":"

    To transfer a workspace, select Settings -> General within the workspace. Then, from the three dot menu in the upper right of the page, select Transfer.

    The Transfer Workspace page shows the workspace to be transferred on the left. Select the target account for the workspace on the right.

    Workspace transfer impact on accounts

    Workspace transfer may impact resource usage and costs for source and target accounts.

    When you transfer a workspace, users, API keys, and service accounts may lose access to the workspace. Audit log will no longer track activity on the workspace. Flow runs ending outside of the destination account\u2019s flow run retention period will be removed. You may also need to update Prefect CLI profiles and execution environment settings to access the workspace's new location.

    You may also incur new charges in the target account to accommodate the transferred workspace.

    The Transfer Workspace page outlines the impacts of transferring the selected workspace to the selected target. Please review these notes carefully before selecting Transfer to transfer the workspace.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/users/","title":"User accounts","text":"

    Sign up for a Prefect Cloud account at app.prefect.cloud.

    An individual user can be invited to become a member of other accounts.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#user-settings","title":"User settings","text":"

    Users can access their personal settings in the profile menu, including:

    • Profile: View and editing basic information, such as name.
    • API keys: Create and view API keys for connecting to Prefect Cloud from the CLI or other environments.
    • Preferences: Manage settings, such as color mode and default time zone.
    • Feature previews: Enable or disable feature previews.
    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#account-roles","title":"Account roles","text":"

    Users who are part of an account can hold the role of Admin or Member. Admins can invite other users to join the account and manage the account's workspaces and teams.

    Admins on Pro and Enterprise tier Prefect Cloud accounts can grant members of the account roles in a workspace, such as Runner or Viewer. Custom roles are available on Enterprise tier accounts.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#api-keys","title":"API keys","text":"

    API keys enable you to authenticate an environment to work with Prefect Cloud.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#service-accounts","title":"Service accounts","text":"

    Service accounts enable you to create a Prefect Cloud API key that is not associated with a user account.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#single-sign-on-sso","title":"Single sign-on (SSO)","text":"

    Pro and Enterprise plans offer single sign-on (SSO) integration with your team\u2019s identity provider. Enterprise tier accounts provide additional options with directory sync and SCIM provisioning.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#audit-log","title":"Audit log","text":"

    Audit logs provide a chronological record of activities performed by Prefect Cloud users who are members of an account.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#object-level-access-control-lists-acls","title":"Object-level access control lists (ACLs)","text":"

    Prefect Cloud's Enterprise plan offers object-level access control lists to restrict access to specific users and service accounts within a workspace.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/#teams","title":"Teams","text":"

    Users of Enterprise tier Prefect Cloud accounts can be added to Teams to simplify access control governance.

    ","tags":["Prefect Cloud","Users"],"boost":2},{"location":"cloud/users/api-keys/","title":"Manage Prefect Cloud API Keys","text":"

    API keys enable you to authenticate a local environment to work with Prefect Cloud.

    If you run prefect cloud login from your CLI, you'll have the choice to authenticate through your browser or by pasting an API key.

    If you choose to authenticate through your browser, you'll be directed to an authorization page. After you grant approval to connect, you'll be redirected to the CLI and the API key will be saved to your local Prefect profile.

    If you choose to authenticate by pasting an API key, you'll need to create an API key in the Prefect Cloud UI first.

    ","tags":["Prefect Cloud","API keys","configuration"],"boost":2},{"location":"cloud/users/api-keys/#create-an-api-key","title":"Create an API key","text":"

    To create an API key, select the account icon at the bottom-left corner of the UI.

    Select API Keys. The page displays a list of previously generated keys and lets you create new API keys or delete keys.

    Select the + button to create a new API key. Provide a name for the key and an expiration date.

    Note that API keys cannot be revealed again in the UI after you generate them, so copy the key to a secure location.

    ","tags":["Prefect Cloud","API keys","configuration"],"boost":2},{"location":"cloud/users/api-keys/#log-into-prefect-cloud-with-an-api-key","title":"Log into Prefect Cloud with an API Key","text":"
    prefect cloud login -k '<my-api-key>'\n
    ","tags":["Prefect Cloud","API keys","configuration"],"boost":2},{"location":"cloud/users/api-keys/#service-account-api-keys","title":"Service account API keys","text":"

    Service accounts are a feature of Prefect Cloud Pro and Enterprise tier plans that enable you to create a Prefect Cloud API key that is not associated with a user account.

    Service accounts are typically used to configure API access for running workers or executing flow runs on remote infrastructure. Events and logs for flow runs in those environments are then associated with the service account rather than a user, and API access may be managed or revoked by configuring or removing the service account without disrupting user access.

    See the service accounts documentation for more information about creating and managing service accounts in Prefect Cloud.

    ","tags":["Prefect Cloud","API keys","configuration"],"boost":2},{"location":"cloud/users/audit-log/","title":"Audit Log","text":"

    Prefect Cloud's Pro and Enterprise plans offer enhanced compliance and transparency tools with Audit Log. Audit logs provide a chronological record of activities performed by members in your account, allowing you to monitor detailed Prefect Cloud actions for security and compliance purposes.

    Audit logs enable you to identify who took what action, when, and using what resources within your Prefect Cloud account. In conjunction with appropriate tools and procedures, audit logs can assist in detecting potential security violations and investigating application errors.

    Audit logs can be used to identify changes in:

    • Access to workspaces
    • User login activity
    • User API key creation and removal
    • Workspace creation and removal
    • Account member invitations and removal
    • Service account creation, API key rotation, and removal
    • Billing payment method for self-serve pricing tiers

    See the Prefect Cloud plan information to learn more about options for supporting audit logs.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","audit logs","compliance"],"boost":2},{"location":"cloud/users/audit-log/#viewing-audit-logs","title":"Viewing audit logs","text":"

    From your Pro or Enterprise account settings page, select the Audit Log page to view audit logs.

    Pro and Enterprise account tier admins can view audit logs for:

    • Account-level events in Prefect Cloud, such as:
    • Member invites
    • Changing a member\u2019s role
    • Member login and logout of Prefect Cloud
    • Creating or deleting a service account
    • Workspace-level events in Prefect Cloud, such as:
    • Adding a member to a workspace
    • Changing a member\u2019s workspace role
    • Creating or deleting a workspace

    Admins can filter audit logs on multiple dimensions to restrict the results they see by workspace, user, or event type. Available audit log events are displayed in the Events drop-down menu.

    Audit logs may also be filtered by date range. Audit log retention period varies by Prefect Cloud plan.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","audit logs","compliance"],"boost":2},{"location":"cloud/users/object-access-control-lists/","title":"Object Access Control Lists","text":"

    Prefect Cloud's Enterprise plan offers object-level access control lists to restrict access to specific users and service accounts within a workspace. ACLs are supported for blocks and deployments.

    Organization Admins and Workspace Owners can configure access control lists by navigating to an object and clicking manage access. When an ACL is added, all users and service accounts with access to an object via their workspace role will lose access if not explicitly added to the ACL.

    ACLs and visibility

    Objects not governed by access control lists such as flow runs, flows, and artifacts will be visible to a user within a workspace even if an associated block or deployment has been restricted for that user.

    See the Prefect Cloud plans to learn more about options for supporting object-level access control.

    ","tags":["UI","Permissions","Access","Prefect Cloud","enterprise","teams","workspaces","organizations","audit logs","compliance"],"boost":2},{"location":"cloud/users/roles/","title":"User and Service Account Roles","text":"

    Prefect Cloud's Pro and Enterprise tiers allow you to set team member access to the appropriate level within specific workspaces.

    Role-based access controls (RBAC) enable you to assign users granular permissions to perform certain activities.

    To give users access to functionality beyond the scope of Prefect\u2019s built-in workspace roles, Enterprise account Admins can create custom roles for users.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#built-in-roles","title":"Built-in roles","text":"

    Roles give users abilities at either the account level or at the individual workspace level.

    • An account-level role defines a user's default permissions within an account.
    • A workspace-level role defines a user's permissions within a specific workspace.

    The following sections outline the abilities of the built-in, Prefect-defined ac and workspace roles.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#account-level-roles","title":"Account-level roles","text":"

    The following built-in roles have permissions across an account in Prefect Cloud.

    Role Abilities Admin \u2022 Set/change all account profile settings allowed to be set/changed by a Prefect user. \u2022 Add and remove account members, and their account roles. \u2022 Create and delete service accounts in the account. \u2022 Create workspaces in the account. \u2022 Implicit workspace owner access on all workspaces in the account. Member \u2022 View account profile settings. \u2022 View workspaces I have access to in the account. \u2022 View account members and their roles. \u2022 View service accounts in the account.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#workspace-level-roles","title":"Workspace-level roles","text":"

    The following built-in roles have permissions within a given workspace in Prefect Cloud.

    Role Abilities Viewer \u2022 View flow runs within a workspace. \u2022 View deployments within a workspace. \u2022 View all work pools within a workspace. \u2022 View all blocks within a workspace. \u2022 View all automations within a workspace. \u2022 View workspace handle and description. Runner All Viewer abilities, plus: \u2022 Run deployments within a workspace. Developer All Runner abilities, plus: \u2022 Run flows within a workspace. \u2022 Delete flow runs within a workspace. \u2022 Create, edit, and delete deployments within a workspace. \u2022 Create, edit, and delete work pools within a workspace. \u2022 Create, edit, and delete all blocks and their secrets within a workspace. \u2022 Create, edit, and delete automations within a workspace. \u2022 View all workspace settings. Owner All Developer abilities, plus: \u2022 Add and remove account members, and set their role within a workspace. \u2022 Set the workspace\u2019s default workspace role for all users in the account. \u2022 Set, view, edit workspace settings. Worker The minimum scopes required for a worker to poll for and submit work.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#custom-workspace-roles","title":"Custom workspace roles","text":"

    The built-in roles will serve the needs of most users, but your team may need to configure custom roles, giving users access to specific permissions within a workspace.

    Custom roles can inherit permissions from a built-in role. This enables tweaks to the role to meet your team\u2019s needs, while ensuring users can still benefit from Prefect\u2019s default workspace role permission curation as new functionality becomes available.

    Custom workspace roles can also be created independent of Prefect\u2019s built-in roles. This option gives workspace admins full control of user access to workspace functionality. However, for non-inherited custom roles, the workspace admin takes on the responsibility for monitoring and setting permissions for new functionality as it is released.

    See Role permissions for details of permissions you may set for custom roles.

    After you create a new role, it become available in the account Members page and the Workspace Sharing page for you to apply to users.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#inherited-roles","title":"Inherited roles","text":"

    A custom role may be configured as an Inherited Role. Using an inherited role allows you to create a custom role using a set of initial permissions associated with a built-in Prefect role. Additional permissions can be added to the custom role. Permissions included in the inherited role cannot be removed.

    Custom roles created using an inherited role will follow Prefect's default workspace role permission curation as new functionality becomes available.

    To configure an inherited role when configuring a custom role, select the Inherit permission from a default role check box, then select the role from which the new role should inherit permissions.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#workspace-role-permissions","title":"Workspace role permissions","text":"

    The following permissions are available for custom roles.

    ","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#automations","title":"Automations","text":"Permission Description View automations User can see configured automations within a workspace. Create, edit, and delete automations User can create, edit, and delete automations within a workspace. Includes permissions of View automations.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#blocks","title":"Blocks","text":"Permission Description View blocks User can see configured blocks within a workspace. View secret block data User can see configured blocks and their secrets within a workspace. Includes permissions of\u00a0View blocks. Create, edit, and delete blocks User can create, edit, and delete blocks within a workspace. Includes permissions of View blocks and View secret block data.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#deployments","title":"Deployments","text":"Permission Description View deployments User can see configured deployments within a workspace. Run deployments User can run deployments within a workspace. This does not give a user permission to execute the flow associated with the deployment. This only gives a user (via their key) the ability to run a deployment \u2014 another user/key must actually execute that flow, such as a service account with an appropriate role. Includes permissions of View deployments. Create and edit deployments User can create and edit deployments within a workspace. Includes permissions of View deployments and Run deployments. Delete deployments User can delete deployments within a workspace. Includes permissions of View deployments, Run deployments, and Create and edit deployments.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#flows","title":"Flows","text":"Permission Description View flows and flow runs User can see flows and flow runs within a workspace. Create, update, and delete saved search filters User can create, update, and delete saved flow run search filters configured within a workspace. Includes permissions of View flows and flow runs. Create, update, and run flows User can create, update, and run flows within a workspace. Includes permissions of View flows and flow runs. Delete flows User can delete flows within a workspace. Includes permissions of View flows and flow runs and Create, update, and run flows.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#notifications","title":"Notifications","text":"Permission Description View notification policies User can see notification policies configured within a workspace. Create and edit notification policies User can create and edit notification policies configured within a workspace. Includes permissions of View notification policies. Delete notification policies User can delete notification policies configured within a workspace. Includes permissions of View notification policies and Create and edit notification policies.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#task-run-concurrency","title":"Task run concurrency","text":"Permission Description View concurrency limits User can see configured task run concurrency limits within a workspace. Create, edit, and delete concurrency limits User can create, edit, and delete task run concurrency limits within a workspace. Includes permissions of View concurrency limits.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#work-pools","title":"Work pools","text":"Permission Description View work pools User can see work pools configured within a workspace. Create, edit, and pause work pools User can create, edit, and pause work pools configured within a workspace. Includes permissions of View work pools. Delete work pools User can delete work pools configured within a workspace. Includes permissions of View work pools and Create, edit, and pause work pools.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/roles/#workspace-management","title":"Workspace management","text":"Permission Description View information about workspace service accounts User can see service accounts configured within a workspace. View information about workspace users User can see user accounts for users invited to the workspace. View workspace settings User can see settings configured within a workspace. Edit workspace settings User can edit settings for a workspace. Includes permissions of View workspace settings. Delete the workspace User can delete a workspace. Includes permissions of View workspace settings and Edit workspace settings.","tags":["UI","dashboard","Prefect Cloud","accounts","teams","workspaces","organizations","custom roles","RBAC"],"boost":2},{"location":"cloud/users/service-accounts/","title":"Service Accounts","text":"

    Service accounts enable you to create a Prefect Cloud API key that is not associated with a user account. Service accounts are typically used to configure API access for running workers or executing deployment flow runs on remote infrastructure.

    Service accounts are non-user accounts that have the following features:

    • Prefect Cloud API keys
    • Roles and permissions

    Using service account credentials, you can configure an execution environment to interact with your Prefect Cloud workspaces without a user having to manually log in from that environment. Service accounts may be created, added to workspaces, have their roles changed, or deleted without affecting other user accounts.

    Select Service Accounts to view, create, or edit service accounts.

    Service accounts are created at the account level, but individual workspaces may be shared with the service account. See workspace sharing for more information.

    Service account credentials

    When you create a service account, Prefect Cloud creates a new API key for the account and provides the API configuration command for the execution environment. Save these to a safe location for future use. If the access credentials are lost or compromised, you should regenerate the credentials from the service account page.

    Service account roles

    Service accounts are created at the account level, and can then be added to workspaces within the account.

    A service account may only be a Member of an account. It can never be an account Admin. You may apply any valid workspace-level role to a service account.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/users/service-accounts/#create-a-service-account","title":"Create a service account","text":"

    Within your account, on the Service Accounts page, select the + icon to create a new service account. You'll be prompted to configure:

    • The service account name. This name must be unique within your account.
    • An expiration date, or the Never Expire option.

    Service account roles

    A service account may only be a Member of an account. You may apply any valid workspace-level role to a service account when it is added to a workspace.

    Select Create to create the new service account.

    Note that API keys cannot be revealed again in the UI after you generate them, so copy the key to a secure location.

    You can change the API key and expiration for a service account by rotating the API key. Select Rotate API Key from the menu on the left side of the service account's information on this page.

    To delete a service account, select Remove from the menu on the left side of the service account's information.

    ","tags":["UI","Prefect Cloud","workspaces","deployments"],"boost":2},{"location":"cloud/users/sso/","title":"Single Sign-on (SSO)","text":"

    Prefect Cloud's Pro and Enterprise plans offer single sign-on (SSO) integration with your team\u2019s identity provider. SSO integration can bet set up with any identity provider that supports:

    • OIDC
    • SAML 2.0

    When using SSO, Prefect Cloud won't store passwords for any accounts managed by your identity provider. Members of your Prefect Cloud account will instead log in and authenticate using your identity provider.

    Once your SSO integration has been set up, non-admins will be required to authenticate through the SSO provider when accessing account resources.

    See the Prefect Cloud plans to learn more about options for supporting more users and workspaces, service accounts, and SSO.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","single sign-on","SSO","authentication"],"boost":2},{"location":"cloud/users/sso/#configuring-sso","title":"Configuring SSO","text":"

    Within your account, select the SSO page to enable SSO for users.

    If you haven't enabled SSO for a domain yet, enter the email domains for which you want to configure SSO in Prefect Cloud and save it.

    Under Enabled Domains, select the domains from the Domains list, then select Generate Link. This step creates a link you can use to configure SSO with your identity provider.

    Using the provided link navigate to the Identity Provider Configuration dashboard and select your identity provider to continue configuration. If your provider isn't listed, you can continue with the SAML or Open ID Connect choices instead.

    Once you complete SSO configuration your users will be required to authenticate via your identity provider when accessing account resources, giving you full control over application access.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","single sign-on","SSO","authentication"],"boost":2},{"location":"cloud/users/sso/#directory-sync","title":"Directory sync","text":"

    Directory sync automatically provisions and de-provisions users for your account.

    Provisioned users are given basic \u201cMember\u201d roles and will have access to any resources that role entails.

    When a user is unassigned from the Prefect Cloud application in your identity provider, they will automatically lose access to Prefect Cloud resources, allowing your IT team to control access to Prefect Cloud without ever signing into the app.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","single sign-on","SSO","authentication"],"boost":2},{"location":"cloud/users/sso/#scim-provisioning","title":"SCIM Provisioning","text":"

    Enterprise accounts have access to SCIM for user provisioning. The SSO tab provides access to enable SCIM provisioning.

    ","tags":["UI","dashboard","Prefect Cloud","enterprise","teams","workspaces","organizations","single sign-on","SSO","authentication"],"boost":2},{"location":"cloud/users/teams/","title":"Teams","text":"

    Prefect Cloud's Enterprise plan offers team management to simplify access control governance.

    Account Admins can configure teams and team membership from the account settings menu by clicking Teams. Teams are composed of users and service accounts. Teams can be added to workspaces or object access control lists just like users and service accounts.

    If SCIM is enabled on your account, the set of teams and the users within them is governed by your IDP. Prefect Cloud service accounts, which are not governed by your IDP, can be still be added to your existing set of teams.

    See the Prefect Cloud plans to learn more about options for supporting teams.

    ","tags":["UI","Permissions","Access","Prefect Cloud","enterprise","teams","workspaces","organizations","audit logs","compliance"],"boost":2},{"location":"community/","title":"Community","text":"

    There are many ways to get involved with the Prefect community

    • Join over 26,000 engineers in the Prefect Slack community
    • Get help in Prefect Discourse - the community-driven knowledge base
    • Give Prefect a \u2b50\ufe0f on GitHub
    • Contribute to Prefect's open source libraries
    • Become a Prefect Ambassador by joining Club 42
    ","tags":["community","Slack","Discourse"],"boost":2},{"location":"concepts/","title":"Explore Prefect concepts","text":"Concept Description Flows A Prefect workflow, defined as a Python function. Tasks Discrete units of work in a Prefect workflow. Deployments A server-side concept that encapsulates flow metadata, allowing it to be scheduled and triggered via API. Work Pools & Workers Use Prefect to dynamically provision and configure infrastructure in your execution environment. Schedules Tell the Prefect API how to create new flow runs for you automatically on a specified cadence. Results The data returned by a flow or a task. Artifacts Formatted outputs rendered in the Prefect UI, such as markdown, tables, or links. Incidents Identify, rectify and document issues in mission-critical workflows. States Rich objects that capture the status of a particular task run or flow run. Blocks Prefect primitives that enable the storage of configuration and provide a UI interface. Task Runners Configure how tasks are run - concurrently, in parallel, or in a distributed environment. Automations Configure actions that Prefect executes automatically based on trigger conditions. Block and Agent-Based Deployments Description Block-based Deployments Create deployments that rely on blocks. Infrastructure Blocks that specify infrastructure for flow runs created by a deployment. Storage Lets you configure how flow code for deployments is persisted and retrieved. Agents Like untyped workers.

    Many features specific to Prefect Cloud are in their own subheading.

    ","tags":["concepts","features","overview"],"boost":2},{"location":"concepts/agents/","title":"Agents","text":"

    Workers are recommended

    Agents are part of the block-based deployment model. Work Pools and Workers simplify the specification of a flow's infrastructure and runtime environment. If you have existing agents, you can upgrade from agents to workers to significantly enhance the experience of deploying flows.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#agent-overview","title":"Agent overview","text":"

    Agent processes are lightweight polling services that get scheduled work from a work pool and deploy the corresponding flow runs.

    Agents poll for work every 15 seconds by default. This interval is configurable in your profile settings with the PREFECT_AGENT_QUERY_INTERVAL setting.

    It is possible for multiple agent processes to be started for a single work pool. Each agent process sends a unique ID to the server to help disambiguate themselves and let users know how many agents are active.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#agent-options","title":"Agent options","text":"

    Agents are configured to pull work from one or more work pool queues. If the agent references a work queue that doesn't exist, it will be created automatically.

    Configuration parameters you can specify when starting an agent include:

    Option Description --api The API URL for the Prefect server. Default is the value of PREFECT_API_URL. --hide-welcome Do not display the startup ASCII art for the agent process. --limit Maximum number of flow runs to start simultaneously. [default: None] --match, -m Dynamically matches work queue names with the specified prefix for the agent to pull from,for example dev- will match all work queues with a name that starts with dev-. [default: None] --pool, -p A work pool name for the agent to pull from. [default: None] --prefetch-seconds The amount of time before a flow run's scheduled start time to begin submission. Default is the value of PREFECT_AGENT_PREFETCH_SECONDS. --run-once Only run agent polling once. By default, the agent runs forever. [default: no-run-once] --work-queue, -q One or more work queue names for the agent to pull from. [default: None]

    You must start an agent within an environment that can access or create the infrastructure needed to execute flow runs. Your agent will deploy flow runs to the infrastructure specified by the deployment.

    Prefect must be installed in execution environments

    Prefect must be installed in any environment in which you intend to run the agent or execute a flow run.

    PREFECT_API_URL and PREFECT_API_KEY settings for agents

    PREFECT_API_URL must be set for the environment in which your agent is running or specified when starting the agent with the --api flag. You must also have a user or service account with the Worker role, which can be configured by setting the PREFECT_API_KEY.

    If you want an agent to communicate with Prefect Cloud or a Prefect server from a remote execution environment such as a VM or Docker container, you must configure PREFECT_API_URL in that environment.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#starting-an-agent","title":"Starting an agent","text":"

    Use the prefect agent start CLI command to start an agent. You must pass at least one work pool name or match string that the agent will poll for work. If the work pool does not exist, it will be created.

    prefect agent start -p [work pool name]\n

    For example:

    Starting agent with ephemeral API...\n\u00a0 ___ ___ ___ ___ ___ ___ _____ \u00a0 \u00a0 _ \u00a0 ___ ___ _\u00a0 _ _____\n\u00a0| _ \\ _ \\ __| __| __/ __|_ \u00a0 _| \u00a0 /_\\ / __| __| \\| |_ \u00a0 _|\n\u00a0|\u00a0 _/ \u00a0 / _|| _|| _| (__\u00a0 | |\u00a0 \u00a0 / _ \\ (_ | _|| .` | | |\n\u00a0|_| |_|_\\___|_| |___\\___| |_| \u00a0 /_/ \\_\\___|___|_|\\_| |_|\n\nAgent started! Looking for work from work pool 'my-pool'...\n

    By default, the agent polls the API specified by the PREFECT_API_URL environment variable. To configure the agent to poll from a different server location, use the --api flag, specifying the URL of the server.

    In addition, agents can match multiple queues in a work pool by providing a --match string instead of specifying all of the queues. The agent will poll every queue with a name that starts with the given string. New queues matching this prefix will be found by the agent without needing to restart it.

    For example:

    prefect agent start --match \"foo-\"\n

    This example will poll every work queue that starts with \"foo-\".

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#configuring-prefetch","title":"Configuring prefetch","text":"

    By default, the agent begins submission of flow runs a short time (10 seconds) before they are scheduled to run. This allows time for the infrastructure to be created, so the flow run can start on time. In some cases, infrastructure will take longer than this to actually start the flow run. In these cases, the prefetch can be increased using the --prefetch-seconds option or the PREFECT_AGENT_PREFETCH_SECONDS setting.

    Submission can begin an arbitrary amount of time before the flow run is scheduled to start. If this value is larger than the amount of time it takes for the infrastructure to start, the flow run will wait until its scheduled start time. This allows flow runs to start exactly on time.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#troubleshooting","title":"Troubleshooting","text":"","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/agents/#agent-crash-or-keyboard-interrupt","title":"Agent crash or keyboard interrupt","text":"

    If the agent process is ended abruptly, you can sometimes have left over flows that were destined for the agent whose process was ended. In the UI, these will show up as pending. You will need to delete these flows in order for the restarted agent to begin processing the work queue again. Take note of the flows you deleted, you might need to set them to run manually.

    ","tags":["agents","deployments"],"boost":0.5},{"location":"concepts/artifacts/","title":"Artifacts","text":"

    Artifacts are persisted outputs such as tables, Markdown, or links. They are stored on Prefect Cloud or a Prefect server instance and rendered in the Prefect UI. Artifacts make it easy to track and monitor the objects that your flows produce and update over time.

    Published artifacts may be associated with a particular task run or flow run. Artifacts can also be created outside of any flow run context.

    Whether you're publishing links, Markdown, or tables, artifacts provide a powerful and flexible way to showcase data within your workflows.

    With artifacts, you can easily manage and share information with your team, providing valuable insights and context.

    Common use cases for artifacts include:

    • Debugging: By publishing data that you care about in the UI, you can easily see when and where your results were written. If an artifact doesn't look the way you expect, you can find out which flow run last updated it, and you can click through a link in the artifact to a storage location (such as an S3 bucket).
    • Data quality checks: Artifacts can be used to publish data quality checks from in-progress tasks. This can help ensure that data quality is maintained throughout the pipeline. During long-running tasks such as ML model training, you might use artifacts to publish performance graphs. This can help you visualize how well your models are performing and make adjustments as needed. You can also track the versions of these artifacts over time, making it easier to identify changes in your data.
    • Documentation: Artifacts can be used to publish documentation and sample data to help you keep track of your work and share information with your colleagues. For instance, artifacts allow you to add a description to let your colleagues know why this piece of data is important.
    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#creating-artifacts","title":"Creating artifacts","text":"

    Creating artifacts allows you to publish data from task and flow runs or outside of a flow run context. Currently, you can render three artifact types: links, Markdown, and tables.

    Artifacts render individually

    Please note that every artifact created within a task will be displayed as an individual artifact in the Prefect UI. This means that each call to create_link_artifact() or create_markdown_artifact() generates a distinct artifact.

    Unlike the print() command, where you can concatenate multiple calls to include additional items in a report, within a task, these commands must be used multiple times if necessary.

    To create artifacts like reports or summaries using create_markdown_artifact(), compile your message string separately and then pass it to create_markdown_artifact() to create the complete artifact.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#creating-link-artifacts","title":"Creating link artifacts","text":"

    To create a link artifact, use the create_link_artifact() function. To create multiple versions of the same artifact and/or view them on the Artifacts page of the Prefect UI, provide a key argument to the create_link_artifact() function to track an artifact's history over time. Without a key, the artifact will only be visible in the Artifacts tab of the associated flow run or task run.\"

    from prefect import flow, task\nfrom prefect.artifacts import create_link_artifact\n\n@task\ndef my_first_task():\n        create_link_artifact(\n            key=\"irregular-data\",\n            link=\"https://nyc3.digitaloceanspaces.com/my-bucket-name/highly_variable_data.csv\",\n            description=\"## Highly variable data\",\n        )\n\n@task\ndef my_second_task():\n        create_link_artifact(\n            key=\"irregular-data\",\n            link=\"https://nyc3.digitaloceanspaces.com/my-bucket-name/low_pred_data.csv\",\n            description=\"# Low prediction accuracy\",\n        )\n\n@flow\ndef my_flow():\n    my_first_task()\n    my_second_task()\n\nif __name__ == \"__main__\":\n    my_flow()\n

    Tip

    You can specify multiple artifacts with the same key to more easily track something very specific that you care about, such as irregularities in your data pipeline.

    After running the above flows, you can find your new artifacts in the Artifacts page of the UI. Click into the \"irregular-data\" artifact and see all versions of it, along with custom descriptions and links to the relevant data.

    Here, you'll also be able to view information about your artifact such as its associated flow run or task run id, previous and future versions of the artifact (multiple artifacts can have the same key in order to show lineage), the data you've stored (in this case a Markdown-rendered link), an optional Markdown description, and when the artifact was created or updated.

    To make the links more readable for you and your collaborators, you can pass in a link_text argument for your link artifacts:

    from prefect import flow\nfrom prefect.artifacts import create_link_artifact\n\n@flow\ndef my_flow():\n    create_link_artifact(\n        key=\"my-important-link\",\n        link=\"https://www.prefect.io/\",\n        link_text=\"Prefect\",\n    )\n\nif __name__ == \"__main__\":\n    my_flow()\n

    In the above example, the create_link_artifact method is used within a flow to create a link artifact with a key of my-important-link. The link parameter is used to specify the external resource to be linked to, and link_text is used to specify the text to be displayed for the link. An optional description could also be added for context.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#creating-markdown-artifacts","title":"Creating Markdown artifacts","text":"

    To create a Markdown artifact, you can use the create_markdown_artifact() function. To create multiple versions of the same artifact and/or view them on the Artifacts page of the Prefect UI, provide a key argument to the create_markdown_artifact() function to track an artifact's history over time. Without a key, the artifact will only be visible in the Artifacts tab of the associated flow run or task run.\"

    Don't indent Markdown

    Markdown in mult-line strings must be unindented to be interpreted correctly.

    from prefect import flow, task\nfrom prefect.artifacts import create_markdown_artifact\n\n@task\ndef markdown_task():\n    na_revenue = 500000\n    markdown_report = f\"\"\"# Sales Report\n\n## Summary\n\nIn the past quarter, our company saw a significant increase in sales, with a total revenue of $1,000,000. \nThis represents a 20% increase over the same period last year.\n\n## Sales by Region\n\n| Region        | Revenue |\n|:--------------|-------:|\n| North America | ${na_revenue:,} |\n| Europe        | $250,000 |\n| Asia          | $150,000 |\n| South America | $75,000 |\n| Africa        | $25,000 |\n\n## Top Products\n\n1. Product A - $300,000 in revenue\n2. Product B - $200,000 in revenue\n3. Product C - $150,000 in revenue\n\n## Conclusion\n\nOverall, these results are very encouraging and demonstrate the success of our sales team in increasing revenue \nacross all regions. However, we still have room for improvement and should focus on further increasing sales in \nthe coming quarter.\n\"\"\"\n    create_markdown_artifact(\n        key=\"gtm-report\",\n        markdown=markdown_report,\n        description=\"Quarterly Sales Report\",\n    )\n\n@flow()\ndef my_flow():\n    markdown_task()\n\n\nif __name__ == \"__main__\":\n    my_flow()\n

    After running the above flow, you should see your \"gtm-report\" artifact in the Artifacts page of the UI.

    As with all artifacts, you'll be able to view the associated flow run or task run id, previous and future versions of the artifact, your rendered Markdown data, and your optional Markdown description.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#create-table-artifacts","title":"Create table artifacts","text":"

    You can create a table artifact by calling create_table_artifact(). To create multiple versions of the same artifact and/or view them on the Artifacts page of the Prefect UI, provide a key argument to the create_table_artifact() function to track an artifact's history over time. Without a key, the artifact will only be visible in the artifacts tab of the associated flow run or task run.\"

    Note

    The create_table_artifact() function accepts a table argument, which can be provided as either a list of lists, a list of dictionaries, or a dictionary of lists.

    from prefect.artifacts import create_table_artifact\n\ndef my_fn():\n    highest_churn_possibility = [\n       {'customer_id':'12345', 'name': 'John Smith', 'churn_probability': 0.85 }, \n       {'customer_id':'56789', 'name': 'Jane Jones', 'churn_probability': 0.65 } \n    ]\n\n    create_table_artifact(\n        key=\"personalized-reachout\",\n        table=highest_churn_possibility,\n        description= \"# Marvin, please reach out to these customers today!\"\n    )\n\nif __name__ == \"__main__\":\n    my_fn()\n

    As you can see, you don't need to create an artifact in a flow run context. You can create one anywhere in a Python script and see it in the Prefect UI.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#managing-artifacts","title":"Managing artifacts","text":"","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#reading-artifacts","title":"Reading artifacts","text":"

    In the Prefect UI, you can view all of the latest versions of your artifacts and click into a specific artifact to see its lineage over time. Additionally, you can inspect all versions of an artifact with a given key by running:

    prefect artifact inspect <my_key>\n

    or view all artifacts by running:

    prefect artifact ls\n

    You can also use the Prefect REST API to programmatically filter your results.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#deleting-artifacts","title":"Deleting artifacts","text":"

    You can delete an artifact directly using the CLI to delete specific artifacts with a given key or id:

    prefect artifact delete <my_key>\n
    prefect artifact delete --id <my_id>\n

    Alternatively, you can delete artifacts using the Prefect REST API.

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/artifacts/#artifacts-api","title":"Artifacts API","text":"

    Prefect provides the Prefect REST API to allow you to create, read, and delete artifacts programmatically. With the Artifacts API, you can automate the creation and management of artifacts as part of your workflow.

    For example, to read the five most recently created Markdown, table, and link artifacts, you can run the following:

    import requests\n\nPREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/abc/workspaces/xyz\"\nPREFECT_API_KEY=\"pnu_ghijk\"\ndata = {\n    \"sort\": \"CREATED_DESC\",\n    \"limit\": 5,\n    \"artifacts\": {\n        \"key\": {\n            \"exists_\": True\n        }\n    }\n}\n\nheaders = {\"Authorization\": f\"Bearer {PREFECT_API_KEY}\"}\nendpoint = f\"{PREFECT_API_URL}/artifacts/filter\"\n\nresponse = requests.post(endpoint, headers=headers, json=data)\nassert response.status_code == 200\nfor artifact in response.json():\n    print(artifact)\n

    If you don't specify a key or that a key must exist, you will also return results (which are a type of key-less artifact).

    See the rest of the Prefect REST API documentation on artifacts for more information!

    ","tags":["artifacts","UI","Markdown"],"boost":2},{"location":"concepts/automations/","title":"Automations","text":"

    Automations in Prefect Cloud enable you to configure actions that Prefect executes automatically based on trigger conditions related to your flows and work pools.

    Using triggers and actions you can automatically kick off flow runs, pause deployments, or send custom notifications in response to real-time monitoring events.

    Automations are only available in Prefect Cloud

    Notifications in an open-source Prefect server provide a subset of the notification message-sending features available in Automations.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#automations-overview","title":"Automations overview","text":"

    The Automations page provides an overview of all configured automations for your workspace.

    Selecting the toggle next to an automation pauses execution of the automation.

    The button next to the toggle provides commands to copy the automation ID, edit the automation, or delete the automation.

    Select the name of an automation to view Details about it and relevant Events.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#create-an-automation","title":"Create an automation","text":"

    On the Automations page, select the + icon to create a new automation. You'll be prompted to configure:

    • A trigger condition that causes the automation to execute.
    • One or more actions carried out by the automation.
    • Details about the automation, such as a name and description.
    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#triggers","title":"Triggers","text":"

    Triggers specify the conditions under which your action should be performed. Triggers can be of several types, including triggers based on:

    • Flow run state change
    • Note - Flow Run Tags currently are only evaluated with OR criteria
    • Work pool status
    • Custom event triggers

    Automations API

    The automations API enables further programmatic customization of trigger and action policies based on arbitrary events.

    Importantly, triggers can be configured not only in reaction to events, but also proactively: to trigger in the absence of an event you expect to see.

    For example, in the case of flow run state change triggers, you might expect production flows to finish in no longer than thirty minutes. But transient infrastructure or network issues could cause your flow to get \u201cstuck\u201d in a running state. A trigger could kick off an action if the flow stays in a running state for more than 30 minutes. This action could be on the flow itself, such as canceling or restarting it, or it could take the form of a notification so someone can take manual remediation steps.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#actions","title":"Actions","text":"

    Actions specify what your automation does when its trigger criteria are met. Current action types include:

    • Cancel a flow run
    • Pause a flow run
    • Run a deployment
    • Pause or resume a deployment schedule
    • Pause or resume a work queue
    • Pause or resume an automation
    • Send a notification
    • Call a webhook

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#selected-and-inferred-action-targets","title":"Selected and inferred action targets","text":"

    Some actions require you to either select the target of the action, or specify that the target of the action should be inferred.

    Selected targets are simple, and useful for when you know exactly what object your action should act on \u2014 for example, the case of a cleanup flow you want to run or a specific notification you\u2019d like to send.

    Inferred targets are deduced from the trigger itself.

    For example, if a trigger fires on a flow run that is stuck in a running state, and the action is to cancel an inferred flow run, the flow run to cancel is inferred as the stuck run that caused the trigger to fire.

    Similarly, if a trigger fires on a work queue event and the corresponding action is to pause an inferred work queue, the inferred work queue is the one that emitted the event.

    Prefect tries to infer the relevant event whenever possible, but sometimes one does not exist.

    Specify a name and, optionally, a description for the automation.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#custom-triggers","title":"Custom triggers","text":"

    Custom triggers allow advanced configuration of the conditions on which an automation executes its actions. Several custom trigger fields accept values that end with trailing wildcards, like \"prefect.flow-run.*\".

    The schema that defines a trigger is as follows:

    Name Type Supports trailing wildcards Description match object Labels for resources which this Automation will match. match_related object Labels for related resources which this Automation will match. after array of strings Event(s), one of which must have first been seen to start this automation. expect array of strings The event(s) this automation is expecting to see. If empty, this automation will evaluate any matched event. for_each array of strings Evaluate the Automation separately for each distinct value of these labels on the resource. By default, labels refer to the primary resource of the triggering event. You may also refer to labels from related resources by specifying related:<role>:<label>. This will use the value of that label for the first related resource in that role. posture string enum N/A The posture of this Automation, either Reactive or Proactive. Reactive automations respond to the presence of the expected events, while Proactive automations respond to the absence of those expected events. threshold integer N/A The number of events required for this Automation to trigger (for Reactive automations), or the number of events expected (for Proactive automations) within number N/A The time period over which the events must occur. For Reactive triggers, this may be as low as 0 seconds, but must be at least 10 seconds for Proactive triggers","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#resource-matching","title":"Resource matching","text":"

    match and match_related control which events a trigger considers for evaluation by filtering on the contents of their resource and related fields, respectively. Each label added to a match filter is ANDed with the other labels, and can accept a single value or a list of multiple values that are ORed together.

    Consider the resource and related fields on the following prefect.flow-run.Completed event, truncated for the sake of example. Its primary resource is a flow run, and since that flow run was started via a deployment, it is related to both its flow and its deployment:

    \"resource\": {\n  \"prefect.resource.id\": \"prefect.flow-run.925eacce-7fe5-4753-8f02-77f1511543db\",\n  \"prefect.resource.name\": \"cute-kittiwake\"\n}\n\"related\": [\n  {\n    \"prefect.resource.id\": \"prefect.flow.cb6126db-d528-402f-b439-96637187a8ca\",\n    \"prefect.resource.role\": \"flow\",\n    \"prefect.resource.name\": \"hello\"\n  },\n  {\n    \"prefect.resource.id\": \"prefect.deployment.37ca4a08-e2d9-4628-a310-cc15a323378e\",\n    \"prefect.resource.role\": \"deployment\",\n    \"prefect.resource.name\": \"example\"\n  }\n]\n

    There are a number of valid ways to select the above event for evaluation, and the approach depends on the purpose of the automation.

    The following configuration will filter for any events whose primary resource is a flow run, and that flow run has a name starting with cute- or radical-.

    \"match\": {\n  \"prefect.resource.id\": \"prefect.flow-run.*\",\n  \"prefect.resource.name\": [\"cute-*\", \"radical-*\"]\n},\n\"match_related\": {},\n...\n

    This configuration, on the other hand, will filter for any events for which this specific deployment is a related resource.

    \"match\": {},\n\"match_related\": {\n  \"prefect.resource.id\": \"prefect.deployment.37ca4a08-e2d9-4628-a310-cc15a323378e\"\n},\n...\n

    Both of the above approaches will select the example prefect.flow-run.Completed event, but will permit additional, possibly undesired events through the filter as well. match and match_related can be combined for more restrictive filtering:

    \"match\": {\n  \"prefect.resource.id\": \"prefect.flow-run.*\",\n  \"prefect.resource.name\": [\"cute-*\", \"radical-*\"]\n},\n\"match_related\": {\n  \"prefect.resource.id\": \"prefect.deployment.37ca4a08-e2d9-4628-a310-cc15a323378e\"\n},\n...\n

    Now this trigger will filter only for events whose primary resource is a flow run started by a specific deployment, and that flow run has a name starting with cute- or radical-.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#expected-events","title":"Expected events","text":"

    Once an event has passed through the match filters, it must be decided if this event should be counted toward the trigger's threshold. Whether that is the case is determined by the event names present in expect.

    This configuration informs the trigger to evaluate only prefect.flow-run.Completed events that have passed the match filters.

    \"expect\": [\n  \"prefect.flow-run.Completed\"\n],\n...\n

    threshold decides the quantity of expected events needed to satisfy the trigger. Increasing the threshold above 1 will also require use of within to define a range of time in which multiple events are seen. The following configuration will expect two occurrences of prefect.flow-run.Completed within 60 seconds.

    \"expect\": [\n  \"prefect.flow-run.Completed\"\n],\n\"threshold\": 2,\n\"within\": 60,\n...\n

    after can be used to handle scenarios that require more complex event reactivity.

    Take, for example, this flow which emits an event indicating the table it operates on is missing or empty:

    from prefect import flow\nfrom prefect.events import emit_event\nfrom db import Table\n\n\n@flow\ndef transform(table_name: str):\n  table = Table(table_name)\n\n  if not table.exists():\n    emit_event(\n        event=\"table-missing\",\n        resource={\"prefect.resource.id\": \"etl-events.transform\"}\n    )\n  elif table.is_empty():\n    emit_event(\n        event=\"table-empty\",\n        resource={\"prefect.resource.id\": \"etl-events.transform\"}\n    )\n  else:\n    # transform data\n

    The following configuration uses after to prevent this automation from firing unless either a table-missing or a table-empty event has occurred before a flow run of this deployment completes.

    Tip

    Note how match and match_related are used to ensure the trigger only evaluates events that are relevant to its purpose.

    \"match\": {\n  \"prefect.resource.id\": [\n    \"prefect.flow-run.*\",\n    \"etl-events.transform\"\n  ]\n},\n\"match_related\": {\n  \"prefect.resource.id\": \"prefect.deployment.37ca4a08-e2d9-4628-a310-cc15a323378e\"\n}\n\"after\": [\n  \"table-missing\",\n  \"table-empty\"\n]\n\"expect\": [\n  \"prefect.flow-run.Completed\"\n],\n...\n
    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#evaluation-strategy","title":"Evaluation strategy","text":"

    All of the previous examples were designed around a reactive posture - that is, count up events toward the threshold until it is met, then execute actions. To respond to the absence of events, use a proactive posture. A proactive trigger will fire when its threshold has not been met by the end of the window of time defined by within. Proactive triggers must have a within of at least 10 seconds.

    The following trigger will fire if a prefect.flow-run.Completed event is not seen within 60 seconds after a prefect.flow-run.Running event is seen.

    {\n  \"match\": {\n    \"prefect.resource.id\": \"prefect.flow-run.*\"\n  },\n  \"match_related\": {},\n  \"after\": [\n    \"prefect.flow-run.Running\"\n  ],\n  \"expect\": [\n    \"prefect.flow-run.Completed\"\n  ],\n  \"for_each\": [],\n  \"posture\": \"Proactive\",\n  \"threshold\": 1,\n  \"within\": 60\n}\n
    However, without for_each, a prefect.flow-run.Completed event from a different flow run than the one that started this trigger with its prefect.flow-run.Running event could satisfy the condition. Adding a for_each of prefect.resource.id will cause this trigger to be evaluated separately for each flow run id associated with these events.

    {\n  \"match\": {\n    \"prefect.resource.id\": \"prefect.flow-run.*\"\n  },\n  \"match_related\": {},\n  \"after\": [\n    \"prefect.flow-run.Running\"\n  ],\n  \"expect\": [\n    \"prefect.flow-run.Completed\"\n  ],\n  \"for_each\": [\n    \"prefect.resource.id\"\n  ],\n  \"posture\": \"Proactive\",\n  \"threshold\": 1,\n  \"within\": 60\n}\n
    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#create-an-automation-via-deployment-triggers","title":"Create an automation via deployment triggers","text":"

    To enable the simple configuration of event-driven deployments, Prefect provides deployment triggers - a shorthand for creating automations that are linked to specific deployments to run them based on the presence or absence of events.

    # prefect.yaml\ndeployments:\n  - name: my-deployment\n    entrypoint: path/to/flow.py:decorated_fn\n    work_pool:\n      name: my-process-pool\n    triggers:\n      - enabled: true\n        match:\n          prefect.resource.id: my.external.resource\n        expect:\n          - external.resource.pinged\n        parameters:\n          param_1: \"{{ event }}\"\n

    At deployment time, this will create a linked automation that is triggered by events matching your chosen grammar, which will pass the templatable event as a parameter to the deployment's flow run.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#pass-triggers-to-prefect-deploy","title":"Pass triggers to prefect deploy","text":"

    You can pass one or more --trigger arguments to prefect deploy, which can be either a JSON string or a path to a .yaml or .json file.

    # Pass a trigger as a JSON string\nprefect deploy -n test-deployment \\\n  --trigger '{\n    \"enabled\": true, \n    \"match\": {\n      \"prefect.resource.id\": \"prefect.flow-run.*\"\n    }, \n    \"expect\": [\"prefect.flow-run.Completed\"]\n  }'\n\n# Pass a trigger using a JSON/YAML file\nprefect deploy -n test-deployment --trigger triggers.yaml\nprefect deploy -n test-deployment --trigger my_stuff/triggers.json\n

    For example, a triggers.yaml file could have many triggers defined:

    triggers:\n  - enabled: true\n    match:\n      prefect.resource.id: my.external.resource\n    expect:\n      - external.resource.pinged\n    parameters:\n      param_1: \"{{ event }}\"\n  - enabled: true\n    match:\n      prefect.resource.id: my.other.external.resource\n    expect:\n      - some.other.event\n    parameters:\n      param_1: \"{{ event }}\"\n
    Both of the above triggers would be attached to test-deployment after running prefect deploy.

    Triggers passed to prefect deploy will override any triggers defined in prefect.yaml

    While you can define triggers in prefect.yaml for a given deployment, triggers passed to prefect deploy will take precedence over those defined in prefect.yaml.

    Note that deployment triggers contribute to the total number of automations in your workspace.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#automation-notifications","title":"Automation notifications","text":"

    Notifications enable you to set up automation actions that send a message.

    Automation notifications support sending notifications via any predefined block that is capable of and configured to send a message. That includes, for example:

    • Slack message to a channel
    • Microsoft Teams message to a channel
    • Email to a configured email address

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/automations/#templating-notifications-with-jinja","title":"Templating notifications with Jinja","text":"

    The notification body can include templated variables using Jinja syntax. Templated variable enable you to include details relevant to automation trigger, such as a flow or pool name.

    Jinja templated variable syntax wraps the variable name in double curly brackets, like {{ variable }}.

    You can access properties of the underlying flow run objects including:

    • flow_run
    • flow
    • deployment
    • work_queue
    • work_pool

    In addition to its native properties, each object includes an id along with created and updated timestamps.

    The flow_run|ui_url token returns the URL for viewing the flow run in Prefect Cloud.

    Here\u2019s an example for something that would be relevant to a flow run state-based notification:

    Flow run {{ flow_run.name }} entered state {{ flow_run.state.name }}. \n\n    Timestamp: {{ flow_run.state.timestamp }}\n    Flow ID: {{ flow_run.flow_id }}\n    Flow Run ID: {{ flow_run.id }}\n    State message: {{ flow_run.state.message }}\n

    The resulting Slack webhook notification would look something like this:

    You could include flow and deployment properties.

    Flow run {{ flow_run.name }} for flow {{ flow.name }}\nentered state {{ flow_run.state.name }}\nwith message {{ flow_run.state.message }}\n\nFlow tags: {{ flow_run.tags }}\nDeployment name: {{ deployment.name }}\nDeployment version: {{ deployment.version }}\nDeployment parameters: {{ deployment.parameters }}\n

    An automation that reports on work pool status might include notifications using work_pool properties.

    Work pool status alert!\n\nName: {{ work_pool.name }}\nLast polled: {{ work_pool.last_polled }}\n

    In addition to those shortcuts for flows, deployments, and work pools, you have access to the automation and the event that triggered the automation. See the Automations API for additional details.

    Automation: {{ automation.name }}\nDescription: {{ automation.description }}\n\nEvent: {{ event.id }}\nResource:\n{% for label, value in event.resource %}\n{{ label }}: {{ value }}\n{% endfor %}\nRelated Resources:\n{% for related in event.related %}\n    Role: {{ related.role }}\n    {% for label, value in event.resource %}\n    {{ label }}: {{ value }}\n    {% endfor %}\n{% endfor %}\n

    Note that this example also illustrates the ability to use Jinja features such as iterator and for loop control structures when templating notifications.

    ","tags":["UI","states","flow runs","events","triggers","Prefect Cloud"],"boost":2},{"location":"concepts/blocks/","title":"Blocks","text":"

    Blocks are a primitive within Prefect that enable the storage of configuration and provide an interface for interacting with external systems.

    With blocks, you can securely store credentials for authenticating with services like AWS, GitHub, Slack, and any other system you'd like to orchestrate with Prefect.

    Blocks expose methods that provide pre-built functionality for performing actions against an external system. They can be used to download data from or upload data to an S3 bucket, query data from or write data to a database, or send a message to a Slack channel.

    You may configure blocks through code or via the Prefect Cloud and the Prefect server UI.

    You can access blocks for both configuring flow deployments and directly from within your flow code.

    Prefect provides some built-in block types that you can use right out of the box. Additional blocks are available through Prefect Integrations. To use these blocks you can pip install the package, then register the blocks you want to use with Prefect Cloud or a Prefect server.

    Prefect Cloud and the Prefect server UI display a library of block types available for you to configure blocks that may be used by your flows.

    Blocks and parameters

    Blocks are useful for configuration that needs to be shared across flow runs and between flows.

    For configuration that will change between flow runs, we recommend using parameters.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#prefect-built-in-blocks","title":"Prefect built-in blocks","text":"

    Prefect provides a broad range of commonly used, built-in block types. These block types are available in Prefect Cloud and the Prefect server UI.

    Block Slug Description Azure azure Store data as a file on Azure Datalake and Azure Blob Storage. Date Time date-time A block that represents a datetime. Docker Container docker-container Runs a command in a container. Docker Registry docker-registry Connects to a Docker registry. Requires a Docker Engine to be connectable. GCS gcs Store data as a file on Google Cloud Storage. GitHub github Interact with files stored on public GitHub repositories. JSON json A block that represents JSON. Kubernetes Cluster Config kubernetes-cluster-config Stores configuration for interaction with Kubernetes clusters. Kubernetes Job kubernetes-job Runs a command as a Kubernetes Job. Local File System local-file-system Store data as a file on a local file system. Microsoft Teams Webhook ms-teams-webhook Enables sending notifications via a provided Microsoft Teams webhook. Opsgenie Webhook opsgenie-webhook Enables sending notifications via a provided Opsgenie webhook. Pager Duty Webhook pager-duty-webhook Enables sending notifications via a provided PagerDuty webhook. Process process Run a command in a new process. Remote File System remote-file-system Store data as a file on a remote file system. Supports any remote file system supported by fsspec. S3 s3 Store data as a file on AWS S3. Secret secret A block that represents a secret value. The value stored in this block will be obfuscated when this block is logged or shown in the UI. Slack Webhook slack-webhook Enables sending notifications via a provided Slack webhook. SMB smb Store data as a file on a SMB share. String string A block that represents a string. Twilio SMS twilio-sms Enables sending notifications via Twilio SMS. Webhook webhook Block that enables calling webhooks.","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#blocks-in-prefect-integrations","title":"Blocks in Prefect Integrations","text":"

    Blocks can also be created by anyone and shared with the community. You'll find blocks that are available for consumption in many of the published Prefect Integrations. The following table provides an overview of the blocks available from our most popular Prefect Integrations.

    Integration Block Slug prefect-airbyte Airbyte Connection airbyte-connection prefect-airbyte Airbyte Server airbyte-server prefect-aws AWS Credentials aws-credentials prefect-aws ECS Task ecs-task prefect-aws MinIO Credentials minio-credentials prefect-aws S3 Bucket s3-bucket prefect-azure Azure Blob Storage Credentials azure-blob-storage-credentials prefect-azure Azure Container Instance Credentials azure-container-instance-credentials prefect-azure Azure Container Instance Job azure-container-instance-job prefect-azure Azure Cosmos DB Credentials azure-cosmos-db-credentials prefect-azure AzureML Credentials azureml-credentials prefect-bitbucket BitBucket Credentials bitbucket-credentials prefect-bitbucket BitBucket Repository bitbucket-repository prefect-census Census Credentials census-credentials prefect-census Census Sync census-sync prefect-databricks Databricks Credentials databricks-credentials prefect-dbt dbt CLI BigQuery Target Configs dbt-cli-bigquery-target-configs prefect-dbt dbt CLI Profile dbt-cli-profile prefect-dbt dbt Cloud Credentials dbt-cloud-credentials prefect-dbt dbt CLI Global Configs dbt-cli-global-configs prefect-dbt dbt CLI Postgres Target Configs dbt-cli-postgres-target-configs prefect-dbt dbt CLI Snowflake Target Configs dbt-cli-snowflake-target-configs prefect-dbt dbt CLI Target Configs dbt-cli-target-configs prefect-docker Docker Host docker-host prefect-docker Docker Registry Credentials docker-registry-credentials prefect-email Email Server Credentials email-server-credentials prefect-firebolt Firebolt Credentials firebolt-credentials prefect-firebolt Firebolt Database firebolt-database prefect-gcp BigQuery Warehouse bigquery-warehouse prefect-gcp GCP Cloud Run Job cloud-run-job prefect-gcp GCP Credentials gcp-credentials prefect-gcp GcpSecret gcpsecret prefect-gcp GCS Bucket gcs-bucket prefect-gcp Vertex AI Custom Training Job vertex-ai-custom-training-job prefect-github GitHub Credentials github-credentials prefect-github GitHub Repository github-repository prefect-gitlab GitLab Credentials gitlab-credentials prefect-gitlab GitLab Repository gitlab-repository prefect-hex Hex Credentials hex-credentials prefect-hightouch Hightouch Credentials hightouch-credentials prefect-kubernetes Kubernetes Credentials kubernetes-credentials prefect-monday Monday Credentials monday-credentials prefect-monte-carlo Monte Carlo Credentials monte-carlo-credentials prefect-openai OpenAI Completion Model openai-completion-model prefect-openai OpenAI Image Model openai-image-model prefect-openai OpenAI Credentials openai-credentials prefect-slack Slack Credentials slack-credentials prefect-slack Slack Incoming Webhook slack-incoming-webhook prefect-snowflake Snowflake Connector snowflake-connector prefect-snowflake Snowflake Credentials snowflake-credentials prefect-sqlalchemy Database Credentials database-credentials prefect-sqlalchemy SQLAlchemy Connector sqlalchemy-connector prefect-twitter Twitter Credentials twitter-credentials","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#using-existing-block-types","title":"Using existing block types","text":"

    Blocks are classes that subclass the Block base class. They can be instantiated and used like normal classes.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#instantiating-blocks","title":"Instantiating blocks","text":"

    For example, to instantiate a block that stores a JSON value, use the JSON block:

    from prefect.blocks.system import JSON\n\njson_block = JSON(value={\"the_answer\": 42})\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#saving-blocks","title":"Saving blocks","text":"

    If this JSON value needs to be retrieved later to be used within a flow or task, we can use the .save() method on the block to store the value in a block document on the Prefect database for retrieval later:

    json_block.save(name=\"life-the-universe-everything\")\n

    Utilizing the UI

    Blocks documents can also be created and updated via the Prefect UI.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#loading-blocks","title":"Loading blocks","text":"

    The name given when saving the value stored in the JSON block can be used when retrieving the value during a flow or task run:

    from prefect import flow\nfrom prefect.blocks.system import JSON\n\n@flow\ndef what_is_the_answer():\n    json_block = JSON.load(\"life-the-universe-everything\")\n    print(json_block.value[\"the_answer\"])\n\nwhat_is_the_answer() # 42\n

    Blocks can also be loaded with a unique slug that is a combination of a block type slug and a block document name.

    To load our JSON block document from before, we can run the following:

    from prefect.blocks.core import Block\n\njson_block = Block.load(\"json/life-the-universe-everything\")\nprint(json_block.value[\"the-answer\"]) #42\n

    Sharing Blocks

    Blocks can also be loaded by fellow Workspace Collaborators, available on Prefect Cloud.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#deleting-blocks","title":"Deleting blocks","text":"

    You can delete a block by using the .delete() method on the block:

    from prefect.blocks.core import Block\nBlock.delete(\"json/life-the-universe-everything\")\n

    You can also use the CLI to delete specific blocks with a given slug or id:

    prefect block delete json/life-the-universe-everything\n
    prefect block delete --id <my-id>\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#creating-new-block-types","title":"Creating new block types","text":"

    To create a custom block type, define a class that subclasses Block. The Block base class builds off of Pydantic's BaseModel, so custom blocks can be declared in same manner as a Pydantic model.

    Here's a block that represents a cube and holds information about the length of each edge in inches:

    from prefect.blocks.core import Block\n\nclass Cube(Block):\n    edge_length_inches: float\n

    You can also include methods on a block include useful functionality. Here's the same cube block with methods to calculate the volume and surface area of the cube:

    from prefect.blocks.core import Block\n\nclass Cube(Block):\n    edge_length_inches: float\n\n    def get_volume(self):\n        return self.edge_length_inches**3\n\n    def get_surface_area(self):\n        return 6 * self.edge_length_inches**2\n

    Now the Cube block can be used to store different cube configuration that can later be used in a flow:

    from prefect import flow\n\nrubiks_cube = Cube(edge_length_inches=2.25)\nrubiks_cube.save(\"rubiks-cube\")\n\n@flow\ndef calculate_cube_surface_area(cube_name):\n    cube = Cube.load(cube_name)\n    print(cube.get_surface_area())\n\ncalculate_cube_surface_area(\"rubiks-cube\") # 30.375\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#secret-fields","title":"Secret fields","text":"

    All block values are encrypted before being stored, but if you have values that you would not like visible in the UI or in logs, then you can use the SecretStr field type provided by Pydantic to automatically obfuscate those values. This can be useful for fields that are used to store credentials like passwords and API tokens.

    Here's an example of an AWSCredentials block that uses SecretStr:

    from typing import Optional\n\nfrom prefect.blocks.core import Block\nfrom pydantic import SecretStr\n\nclass AWSCredentials(Block):\n    aws_access_key_id: Optional[str] = None\n    aws_secret_access_key: Optional[SecretStr] = None\n    aws_session_token: Optional[str] = None\n    profile_name: Optional[str] = None\n    region_name: Optional[str] = None\n

    Because aws_secret_access_key has the SecretStr type hint assigned to it, the value of that field will not be exposed if the object is logged:

    aws_credentials_block = AWSCredentials(\n    aws_access_key_id=\"AKIAJKLJKLJKLJKLJKLJK\",\n    aws_secret_access_key=\"secret_access_key\"\n)\n\nprint(aws_credentials_block)\n# aws_access_key_id='AKIAJKLJKLJKLJKLJKLJK' aws_secret_access_key=SecretStr('**********') aws_session_token=None profile_name=None region_name=None\n

    There's also use the SecretDict field type provided by Prefect. This type will allow you to add a dictionary field to your block that will have values at all levels automatically obfuscated in the UI or in logs. This is useful for blocks where typing or structure of secret fields is not known until configuration time.

    Here's an example of a block that uses SecretDict:

    from typing import Dict\n\nfrom prefect.blocks.core import Block\nfrom prefect.blocks.fields import SecretDict\n\n\nclass SystemConfiguration(Block):\n    system_secrets: SecretDict\n    system_variables: Dict\n\n\nsystem_configuration_block = SystemConfiguration(\n    system_secrets={\n        \"password\": \"p@ssw0rd\",\n        \"api_token\": \"token_123456789\",\n        \"private_key\": \"<private key here>\",\n    },\n    system_variables={\n        \"self_destruct_countdown_seconds\": 60,\n        \"self_destruct_countdown_stop_time\": 7,\n    },\n)\n
    system_secrets will be obfuscated when system_configuration_block is displayed, but system_variables will be shown in plain-text:

    print(system_configuration_block)\n# SystemConfiguration(\n#   system_secrets=SecretDict('{'password': '**********', 'api_token': '**********', 'private_key': '**********'}'), \n#   system_variables={'self_destruct_countdown_seconds': 60, 'self_destruct_countdown_stop_time': 7}\n# )\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#blocks-metadata","title":"Blocks metadata","text":"

    The way that a block is displayed can be controlled by metadata fields that can be set on a block subclass.

    Available metadata fields include:

    Property Description _block_type_name Display name of the block in the UI. Defaults to the class name. _block_type_slug Unique slug used to reference the block type in the API. Defaults to a lowercase, dash-delimited version of the block type name. _logo_url URL pointing to an image that should be displayed for the block type in the UI. Default to None. _description Short description of block type. Defaults to docstring, if provided. _code_example Short code snippet shown in UI for how to load/use block type. Default to first example provided in the docstring of the class, if provided.","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#nested-blocks","title":"Nested blocks","text":"

    Block are composable. This means that you can create a block that uses functionality from another block by declaring it as an attribute on the block that you're creating. It also means that configuration can be changed for each block independently, which allows configuration that may change on different time frames to be easily managed and configuration can be shared across multiple use cases.

    To illustrate, here's a an expanded AWSCredentials block that includes the ability to get an authenticated session via the boto3 library:

    from typing import Optional\n\nimport boto3\nfrom prefect.blocks.core import Block\nfrom pydantic import SecretStr\n\nclass AWSCredentials(Block):\n    aws_access_key_id: Optional[str] = None\n    aws_secret_access_key: Optional[SecretStr] = None\n    aws_session_token: Optional[str] = None\n    profile_name: Optional[str] = None\n    region_name: Optional[str] = None\n\n    def get_boto3_session(self):\n        return boto3.Session(\n            aws_access_key_id = self.aws_access_key_id\n            aws_secret_access_key = self.aws_secret_access_key\n            aws_session_token = self.aws_session_token\n            profile_name = self.profile_name\n            region_name = self.region\n        )\n

    The AWSCredentials block can be used within an S3Bucket block to provide authentication when interacting with an S3 bucket:

    import io\n\nclass S3Bucket(Block):\n    bucket_name: str\n    credentials: AWSCredentials\n\n    def read(self, key: str) -> bytes:\n        s3_client = self.credentials.get_boto3_session().client(\"s3\")\n\n        stream = io.BytesIO()\n        s3_client.download_fileobj(Bucket=self.bucket_name, key=key, Fileobj=stream)\n\n        stream.seek(0)\n        output = stream.read()\n\n        return output\n\n    def write(self, key: str, data: bytes) -> None:\n        s3_client = self.credentials.get_boto3_session().client(\"s3\")\n        stream = io.BytesIO(data)\n        s3_client.upload_fileobj(stream, Bucket=self.bucket_name, Key=key)\n

    You can use this S3Bucket block with previously saved AWSCredentials block values in order to interact with the configured S3 bucket:

    my_s3_bucket = S3Bucket(\n    bucket_name=\"my_s3_bucket\",\n    credentials=AWSCredentials.load(\"my_aws_credentials\")\n)\n\nmy_s3_bucket.save(\"my_s3_bucket\")\n

    Saving block values like this links the values of the two blocks so that any changes to the values stored for the AWSCredentials block with the name my_aws_credentials will be seen the next time that block values for the S3Bucket block named my_s3_bucket is loaded.

    Values for nested blocks can also be hard coded by not first saving child blocks:

    my_s3_bucket = S3Bucket(\n    bucket_name=\"my_s3_bucket\",\n    credentials=AWSCredentials(\n        aws_access_key_id=\"AKIAJKLJKLJKLJKLJKLJK\",\n        aws_secret_access_key=\"secret_access_key\"\n    )\n)\n\nmy_s3_bucket.save(\"my_s3_bucket\")\n

    In the above example, the values for AWSCredentials are saved with my_s3_bucket and will not be usable with any other blocks.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#handling-updates-to-custom-block-types","title":"Handling updates to custom Block types","text":"

    Let's say that you now want to add a bucket_folder field to your custom S3Bucket block that represents the default path to read and write objects from (this field exists on our implementation).

    We can add the new field to the class definition:

    class S3Bucket(Block):\n    bucket_name: str\n    credentials: AWSCredentials\n    bucket_folder: str = None\n    ...\n

    Then register the updated block type with either Prefect Cloud or your self-hosted Prefect server.

    If you have any existing blocks of this type that were created before the update and you'd prefer to not re-create them, you can migrate them to the new version of your block type by adding the missing values:

    # Bypass Pydantic validation to allow your local Block class to load the old block version\nmy_s3_bucket_block = S3Bucket.load(\"my-s3-bucket\", validate=False)\n\n# Set the new field to an appropriate value\nmy_s3_bucket_block.bucket_path = \"my-default-bucket-path\"\n\n# Overwrite the old block values and update the expected fields on the block\nmy_s3_bucket_block.save(\"my-s3-bucket\", overwrite=True)\n
    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/blocks/#registering-blocks-for-use-in-the-prefect-ui","title":"Registering blocks for use in the Prefect UI","text":"

    Blocks can be registered from a Python module available in the current virtual environment with a CLI command like this:

    $ prefect block register --module prefect_aws.credentials\n

    This command is useful for registering all blocks found in the credentials module within Prefect Integrations.

    Or, if a block has been created in a .py file, the block can also be registered with the CLI command:

    $ prefect block register --file my_block.py\n

    The registered block will then be available in the Prefect UI for configuration.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","deployments"],"boost":2},{"location":"concepts/deployments-block-based/","title":"Block Based Deployments","text":"

    Workers are recommended

    This page is about the block-based deployment model. The Work Pools and Workers based deployment model simplifies the specification of a flow's infrastructure and runtime environment. If you have existing agents, you can upgrade from agents to workers to significantly enhance the experience of deploying flows.

    We encourage you to check out the new deployment experience with guided command line prompts and convenient CI/CD with prefect.yaml files.

    With remote storage blocks, you can package not only your flow code script but also any supporting files, including your custom modules, SQL scripts and any configuration files needed in your project.

    To define how your flow execution environment should be configured, you may either reference pre-configured infrastructure blocks or let Prefect create those automatically for you as anonymous blocks (this happens when you specify the infrastructure type using --infra flag during the build process).

    Work queue affinity improved starting from Prefect 2.0.5

    Until Prefect 2.0.4, tags were used to associate flow runs with work queues. Starting in Prefect 2.0.5, tag-based work queues are deprecated. Instead, work queue names are used to explicitly direct flow runs from deployments into queues.

    Note that backward compatibility is maintained and work queues that use tag-based matching can still be created and will continue to work. However, those work queues are now considered legacy and we encourage you to use the new behavior by specifying work queues explicitly on agents and deployments.

    See Agents & Work Pools for details.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deployments-and-flows","title":"Deployments and flows","text":"

    Each deployment is associated with a single flow, but any given flow can be referenced by multiple deployments.

    Deployments are uniquely identified by the combination of: flow_name/deployment_name.

    graph LR\n    F(\"my_flow\"):::yellow -.-> A(\"Deployment 'daily'\"):::tan --> W(\"my_flow/daily\"):::fgreen\n    F -.-> B(\"Deployment 'weekly'\"):::gold  --> X(\"my_flow/weekly\"):::green\n    F -.-> C(\"Deployment 'ad-hoc'\"):::dgold --> Y(\"my_flow/ad-hoc\"):::dgreen\n    F -.-> D(\"Deployment 'trigger-based'\"):::dgold --> Z(\"my_flow/trigger-based\"):::dgreen\n\n    classDef gold fill:goldenrod,stroke:goldenrod,stroke-width:4px,color:white\n    classDef yellow fill:gold,stroke:gold,stroke-width:4px\n    classDef dgold fill:darkgoldenrod,stroke:darkgoldenrod,stroke-width:4px,color:white\n    classDef tan fill:tan,stroke:tan,stroke-width:4px,color:white\n    classDef fgreen fill:forestgreen,stroke:forestgreen,stroke-width:4px,color:white\n    classDef green fill:green,stroke:green,stroke-width:4px,color:white\n    classDef dgreen fill:darkgreen,stroke:darkgreen,stroke-width:4px,color:white

    This enables you to run a single flow with different parameters, based on multiple schedules and triggers, and in different environments. This also enables you to run different versions of the same flow for testing and production purposes.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deployment-definition","title":"Deployment definition","text":"

    A deployment definition captures the settings for creating a deployment object on the Prefect API. You can create the deployment definition by:

    • Run the prefect deployment build CLI command with deployment options to create a deployment.yaml deployment definition file, then run prefect deployment apply to create a deployment on the API using the settings in deployment.yaml.
    • Define a Deployment Python object, specifying the deployment options as properties of the object, then building and applying the object using methods of Deployment.

    The minimum required information to create a deployment includes:

    • The path and filename of the file containing the flow script.
    • The name of the entrypoint flow function \u2014 this is the flow function that starts the flow and calls and additional tasks or subflows.
    • The name of the deployment.

    You may provide additional settings for the deployment. Any settings you do not explicitly specify are inferred from defaults.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-deployment-on-the-cli","title":"Create a deployment on the CLI","text":"

    To create a deployment on the CLI, there are two steps:

    1. Build the deployment definition file deployment.yaml. This step includes uploading your flow to its configured remote storage location, if one is specified.
    2. Create the deployment on the API.
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#build-the-deployment","title":"Build the deployment","text":"

    To build the deployment definition file deployment.yaml, run the prefect deployment build Prefect CLI command from the folder containing your flow script and any dependencies of the script.

    $ prefect deployment build [OPTIONS] PATH\n

    Path to the flow is specified in the format path-to-script:flow-function-name \u2014 The path and filename of the flow script file, a colon, then the name of the entrypoint flow function.

    For example:

    $ prefect deployment build -n marvin -p default-agent-pool -q test flows/marvin.py:say_hi\n

    When you run this command, Prefect:

    • Creates a marvin_flow-deployment.yaml file for your deployment based on your flow code and options.
    • Uploads your flow files to the configured storage location (local by default).
    • Submit your deployment to the work queue test. The work queue test will be created if it doesn't exist.

    Uploading files may require storage filesystem libraries

    Note that the appropriate filesystem library supporting the storage location must be installed prior to building a deployment with a storage block. For example, the AWS S3 Storage block requires the s3fs library.

    Ignore files or directories from a deployment

    By default, Prefect uploads all files in the current folder to the configured storage location (local by default) when you build a deployment.

    If you want to omit certain files or directories from your deployments, add a .prefectignore file to the root directory. .prefectignore enables users to omit certain files or directories from their deployments.

    Similar to other .ignore files, the syntax supports pattern matching, so an entry of *.pyc will ensure all .pyc files are ignored by the deployment call when uploading to remote storage.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deployment-build-options","title":"Deployment build options","text":"

    You may specify additional options to further customize your deployment.

    Options Description PATH Path, filename, and flow name of the flow definition. (Required) --apply, -a When provided, automatically registers the resulting deployment with the API. --cron TEXT A cron string that will be used to set a CronSchedule on the deployment. For example, --cron \"*/1 * * * *\" to create flow runs from that deployment every minute. --help Display help for available commands and options. --infra-block TEXT, -ib The infrastructure block to use, in block-type/block-name format. --infra, -i The infrastructure type to use. (Default is Process) --interval INTEGER An integer specifying an interval (in seconds) that will be used to set an IntervalSchedule on the deployment. For example, --interval 60 to create flow runs from that deployment every minute. --name TEXT, -n The name of the deployment. --output TEXT, -o Optional location for the YAML manifest generated as a result of the build step. You can version-control that file, but it's not required since the CLI can generate everything you need to define a deployment. --override TEXT One or more optional infrastructure overrides provided as a dot delimited path. For example, specify an environment variable: env.env_key=env_value. For Kubernetes, specify customizations: customizations='[{\"op\": \"add\",\"path\": \"/spec/template/spec/containers/0/resources/limits\", \"value\": {\"memory\": \"8Gi\",\"cpu\": \"4000m\"}}]' (note the string format). --param An optional parameter override, values are parsed as JSON strings. For example, --param question=ultimate --param answer=42. --params An optional parameter override in a JSON string format. For example, --params=\\'{\"question\": \"ultimate\", \"answer\": 42}\\'. --path An optional path to specify a subdirectory of remote storage to upload to, or to point to a subdirectory of a locally stored flow. --pool TEXT, -p The work pool that will handle this deployment's runs. \u2502 --rrule TEXT An RRule that will be used to set an RRuleSchedule on the deployment. For example, --rrule 'FREQ=HOURLY;BYDAY=MO,TU,WE,TH,FR;BYHOUR=9,10,11,12,13,14,15,16,17' to create flow runs from that deployment every hour but only during business hours. --skip-upload When provided, skips uploading this deployment's files to remote storage. --storage-block TEXT, -sb The storage block to use, in block-type/block-name or block-type/block-name/path format. Note that the appropriate library supporting the storage filesystem must be installed. --tag TEXT, -t One or more optional tags to apply to the deployment. --version TEXT, -v An optional version for the deployment. This could be a git commit hash if you use this command from a CI/CD pipeline. --work-queue TEXT, -q The work queue that will handle this deployment's runs. It will be created if it doesn't already exist. Defaults to None. Note that if a work queue is not set, work will not be scheduled.","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#block-identifiers","title":"Block identifiers","text":"

    When specifying a storage block with the -sb or --storage-block flag, you may specify the block by passing its slug. The storage block slug is formatted as block-type/block-name.

    For example, s3/example-block is the slug for an S3 block named example-block.

    In addition, when passing the storage block slug, you may pass just the block slug or the block slug and a path.

    • block-type/block-name indicates just the block, including any path included in the block configuration.
    • block-type/block-name/path indicates a storage path in addition to any path included in the block configuration.

    When specifying an infrastructure block with the -ib or --infra-block flag, you specify the block by passing its slug. The infrastructure block slug is formatted as block-type/block-name.

    Block name Block class name Block type for a slug Azure Azure azure Docker Container DockerContainer docker-container GitHub GitHub github GCS GCS gcs Kubernetes Job KubernetesJob kubernetes-job Process Process process Remote File System RemoteFileSystem remote-file-system S3 S3 s3 SMB SMB smb GitLab Repository GitLabRepository gitlab-repository

    Note that the appropriate library supporting the storage filesystem must be installed prior to building a deployment with a storage block. For example, the AWS S3 Storage block requires the s3fs library. See Storage for more information.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deploymentyaml","title":"deployment.yaml","text":"

    A deployment's YAML file configures additional settings needed to create a deployment on the server.

    A single flow may have multiple deployments created for it, with different schedules, tags, and so on. A single flow definition may have multiple deployment YAML files referencing it, each specifying different settings. The only requirement is that each deployment must have a unique name.

    The default {flow-name}-deployment.yaml filename may be edited as needed with the --output flag to prefect deployment build.

    ###\n### A complete description of a Prefect Deployment for flow 'Cat Facts'\n###\nname: catfact\ndescription: null\nversion: c0fc95308d8137c50d2da51af138aa23\n# The work queue that will handle this deployment's runs\nwork_queue_name: test\nwork_pool_name: null\ntags: []\nparameters: {}\nschedule: null\ninfra_overrides: {}\ninfrastructure:\n  type: process\n  env: {}\n  labels: {}\n  name: null\n  command:\n  - python\n  - -m\n  - prefect.engine\n  stream_output: true\n###\n### DO NOT EDIT BELOW THIS LINE\n###\nflow_name: Cat Facts\nmanifest_path: null\nstorage: null\npath: /Users/terry/test/testflows/catfact\nentrypoint: catfact.py:catfacts_flow\nparameter_openapi_schema:\n  title: Parameters\n  type: object\n  properties:\n    url:\n      title: url\n  required:\n  - url\n  definitions: null\n

    Editing deployment.yaml

    Note the big DO NOT EDIT comment in your deployment's YAML: In practice, anything above this block can be freely edited before running prefect deployment apply to create the deployment on the API.

    We recommend editing most of these fields from the CLI or Prefect UI for convenience.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#parameters-in-deployments","title":"Parameters in deployments","text":"

    You may provide default parameter values in the deployment.yaml configuration, and these parameter values will be used for flow runs based on the deployment.

    To configure default parameter values, add them to the parameters: {} line of deployment.yaml as JSON key-value pairs. The parameter list configured in deployment.yaml must match the parameters expected by the entrypoint flow function.

    parameters: {\"name\": \"Marvin\", \"num\": 42, \"url\": \"https://catfact.ninja/fact\"}\n

    Passing **kwargs as flow parameters

    You may pass **kwargs as a deployment parameter as a \"kwargs\":{} JSON object containing the key-value pairs of any passed keyword arguments.

    parameters: {\"name\": \"Marvin\", \"kwargs\":{\"cattype\":\"tabby\",\"num\": 42}\n

    You can edit default parameters for deployments in the Prefect UI, and you can override default parameter values when creating ad-hoc flow runs via the Prefect UI.

    To edit parameters in the Prefect UI, go the the details page for a deployment, then select Edit from the commands menu. If you change parameter values, the new values are used for all future flow runs based on the deployment.

    To create an ad-hoc flow run with different parameter values, go the the details page for a deployment, select Run, then select Custom. You will be able to provide custom values for any editable deployment fields. Under Parameters, select Custom. Provide the new values, then select Save. Select Run to begin the flow run with custom values.

    If you want the Prefect API to verify the parameter values passed to a flow run against the schema defined by parameter_openapi_schema, set enforce_parameter_schema to true.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-deployment","title":"Create a deployment","text":"

    When you've configured deployment.yaml for a deployment, you can create the deployment on the API by running the prefect deployment apply Prefect CLI command.

    $ prefect deployment apply catfacts_flow-deployment.yaml\n

    For example:

    $ prefect deployment apply ./catfacts_flow-deployment.yaml\nSuccessfully loaded 'catfact'\nDeployment '76a9f1ac-4d8c-4a92-8869-615bec502685' successfully created.\n

    prefect deployment apply accepts an optional --upload flag that, when provided, uploads this deployment's files to remote storage.

    Once the deployment has been created, you'll see it in the Prefect UI and can inspect it using the CLI.

    $ prefect deployment ls\n                               Deployments\n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                           \u2503 ID                                   \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 Cat Facts/catfact              \u2502 76a9f1ac-4d8c-4a92-8869-615bec502685 \u2502\n\u2502 leonardo_dicapriflow/hello_leo \u2502 fb4681d7-aa5a-4617-bf6f-f67e6f964984 \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n

    When you run a deployed flow with Prefect, the following happens:

    • The user runs the deployment, which creates a flow run. (The API creates flow runs automatically for deployments with schedules.)
    • An agent picks up the flow run from a work queue and uses an infrastructure block to create infrastructure for the run.
    • The flow run executes within the infrastructure.

    Agents and work pools enable the Prefect orchestration engine and API to run deployments in your local execution environments. To execute deployed flow runs you need to configure at least one agent.

    Scheduled flow runs

    Scheduled flow runs will not be created unless the scheduler is running with either Prefect Cloud or a local Prefect server started with prefect server start.

    Scheduled flow runs will not run unless an appropriate agent and work pool are configured.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-deployment-from-a-python-object","title":"Create a deployment from a Python object","text":"

    You can also create deployments from Python scripts by using the prefect.deployments.Deployment class.

    Create a new deployment using configuration defaults for an imported flow:

    from my_project.flows import my_flow\nfrom prefect.deployments import Deployment\n\ndeployment = Deployment.build_from_flow(\n    flow=my_flow,\n    name=\"example-deployment\", \n    version=1, \n    work_queue_name=\"demo\",\n    work_pool_name=\"default-agent-pool\",\n)\ndeployment.apply()\n

    Create a new deployment with a pre-defined storage block and an infrastructure override:

    from my_project.flows import my_flow\nfrom prefect.deployments import Deployment\nfrom prefect.filesystems import S3\n\nstorage = S3.load(\"dev-bucket\") # load a pre-defined block\n\ndeployment = Deployment.build_from_flow(\n    flow=my_flow,\n    name=\"s3-example\",\n    version=2,\n    work_queue_name=\"aws\",\n    work_pool_name=\"default-agent-pool\",\n    storage=storage,\n    infra_overrides={\n        \"env\": {\n            \"ENV_VAR\": \"value\"\n        }\n    },\n)\n\ndeployment.apply()\n

    If you have settings that you want to share from an existing deployment you can load those settings:

    deployment = Deployment(\n    name=\"a-name-you-used\", \n    flow_name=\"name-of-flow\"\n)\ndeployment.load() # loads server-side settings\n

    Once the existing deployment settings are loaded, you may update them as needed by changing deployment properties.

    View all of the parameters for the Deployment object in the Python API documentation.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#deployment-api-representation","title":"Deployment API representation","text":"

    When you create a deployment, it is constructed from deployment definition data you provide and additional properties set by client-side utilities.

    Deployment properties include:

    Property Description id An auto-generated UUID ID value identifying the deployment. created A datetime timestamp indicating when the deployment was created. updated A datetime timestamp indicating when the deployment was last changed. name The name of the deployment. version The version of the deployment description A description of the deployment. flow_id The id of the flow associated with the deployment. schedule An optional schedule for the deployment. is_schedule_active Boolean indicating whether the deployment schedule is active. Default is True. infra_overrides One or more optional infrastructure overrides parameters An optional dictionary of parameters for flow runs scheduled by the deployment. tags An optional list of tags for the deployment. work_queue_name The optional work queue that will handle the deployment's run parameter_openapi_schema JSON schema for flow parameters. enforce_parameter_schema Whether the API should validate the parameters passed to a flow run against the schema defined by parameter_openapi_schema path The path to the deployment.yaml file entrypoint The path to a flow entry point storage_document_id Storage block configured for the deployment. infrastructure_document_id Infrastructure block configured for the deployment.

    You can inspect a deployment using the CLI with the prefect deployment inspect command, referencing the deployment with <flow_name>/<deployment_name>.

    $ prefect deployment inspect 'Cat Facts/catfact'\n{\n    'id': '76a9f1ac-4d8c-4a92-8869-615bec502685',\n    'created': '2022-07-26T03:48:14.723328+00:00',\n    'updated': '2022-07-26T03:50:02.043238+00:00',\n    'name': 'catfact',\n    'version': '899b136ebc356d58562f48d8ddce7c19',\n    'description': None,\n    'flow_id': '2c7b36d1-0bdb-462e-bb97-f6eb9fef6fd5',\n    'schedule': None,\n    'is_schedule_active': True,\n    'infra_overrides': {},\n    'parameters': {},\n    'tags': [],\n    'work_queue_name': 'test',\n    'parameter_openapi_schema': {\n        'title': 'Parameters',\n        'type': 'object',\n        'properties': {'url': {'title': 'url'}},\n        'required': ['url']\n    },\n    'path': '/Users/terry/test/testflows/catfact',\n    'entrypoint': 'catfact.py:catfacts_flow',\n    'manifest_path': None,\n    'storage_document_id': None,\n    'infrastructure_document_id': 'f958db1c-b143-4709-846c-321125247e07',\n    'infrastructure': {\n        'type': 'process',\n        'env': {},\n        'labels': {},\n        'name': None,\n        'command': ['python', '-m', 'prefect.engine'],\n        'stream_output': True\n    }\n}\n
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-from-a-deployment","title":"Create a flow run from a deployment","text":"","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-with-a-schedule","title":"Create a flow run with a schedule","text":"

    If you specify a schedule for a deployment, the deployment will execute its flow automatically on that schedule as long as a Prefect server and agent are running. Prefect Cloud creates schedules flow runs automatically, and they will run on schedule if an agent is configured to pick up flow runs for the deployment.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-with-an-event-trigger","title":"Create a flow run with an event trigger","text":"

    deployment triggers are only available in Prefect Cloud

    Deployments can optionally take a trigger specification, which will configure an automation to run the deployment based on the presence or absence of events, and optionally pass event data into the deployment run as parameters via jinja templating.

    triggers:\n  - enabled: true\n    match:\n      prefect.resource.id: prefect.flow-run.*\n    expect:\n      - prefect.flow-run.Completed\n    match_related:\n      prefect.resource.name: prefect.flow.etl-flow\n      prefect.resource.role: flow\n    parameters:\n      param_1: \"{{ event }}\"\n

    When applied, this deployment will start a flow run upon the completion of the upstream flow specified in the match_related key, with the flow run passed in as a parameter. Triggers can be configured to respond to the presence or absence of arbitrary internal or external events. The trigger system and API are detailed in Automations.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-with-prefect-ui","title":"Create a flow run with Prefect UI","text":"

    In the Prefect UI, you can click the Run button next to any deployment to execute an ad hoc flow run for that deployment.

    The prefect deployment CLI command provides commands for managing and running deployments locally.

    Command Description apply Create or update a deployment from a YAML file. build Generate a deployment YAML from /path/to/file.py:flow_function. delete Delete a deployment. inspect View details about a deployment. ls View all deployments or deployments for specific flows. pause-schedule Pause schedule of a given deployment. resume-schedule Resume schedule of a given deployment. run Create a flow run for the given flow and deployment. set-schedule Set schedule for a given deployment.","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#create-a-flow-run-in-a-python-script","title":"Create a flow run in a Python script","text":"

    You can create a flow run from a deployment in a Python script with the run_deployment function.

    from prefect.deployments import run_deployment\n\n\ndef main():\n    response = run_deployment(name=\"flow-name/deployment-name\")\n    print(response)\n\n\nif __name__ == \"__main__\":\n   main()\n

    PREFECT_API_URL setting for agents

    You'll need to configure agents and work pools that can create flow runs for deployments in remote environments. PREFECT_API_URL must be set for the environment in which your agent is running.

    If you want the agent to communicate with Prefect Cloud from a remote execution environment such as a VM or Docker container, you must configure PREFECT_API_URL in that environment.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments-block-based/#examples","title":"Examples","text":"
    • How to deploy Prefect flows to AWS
    • How to deploy Prefect flows to GCP
    • How to deploy Prefect flows to Azure
    • How to deploy Prefect flows using files stored locally
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"]},{"location":"concepts/deployments/","title":"Deployments","text":"

    Deployments are server-side representations of flows. They store the crucial metadata needed for remote orchestration including when, where, and how a workflow should run. Deployments elevate workflows from functions that you must call manually to API-managed entities that can be triggered remotely.

    Here we will focus largely on the metadata that defines a deployment and how it is used. Different ways of creating a deployment populate these fields differently.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#overview","title":"Overview","text":"

    Every Prefect deployment references one and only one \"entrypoint\" flow (though that flow may itself call any number of subflows). Different deployments may reference the same underlying flow, a useful pattern when developing or promoting workflow changes through staged environments.

    The complete schema that defines a deployment is as follows:

    class Deployment:\n    \"\"\"\n    Structure of the schema defining a deployment\n    \"\"\"\n\n    # required defining data\n    name: str \n    flow_id: UUID\n    entrypoint: str\n    path: str = None\n\n    # workflow scheduling and parametrization\n    parameters: dict = None\n    parameter_openapi_schema: dict = None\n    schedule: Schedule = None\n    is_schedule_active: bool = True\n    trigger: Trigger = None\n\n    # metadata for bookkeeping\n    version: str = None\n    description: str = None\n    tags: list = None\n\n    # worker-specific fields\n    work_pool_name: str = None\n    work_queue_name: str = None\n    infra_overrides: dict = None\n    pull_steps: dict = None\n

    All methods for creating Prefect deployments are interfaces for populating this schema. Let's look at each section in turn.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#required-data","title":"Required data","text":"

    Deployments universally require both a name and a reference to an underlying Flow. In almost all instances of deployment creation, users do not need to concern themselves with the flow_id as most interfaces will only need the flow's name. Note that the deployment name is not required to be unique across all deployments but is required to be unique for a given flow ID. As a consequence, you will often see references to the deployment's unique identifying name {FLOW_NAME}/{DEPLOYMENT_NAME}. For example, triggering a run of a deployment from the Prefect CLI can be done via:

    prefect deployment run my-first-flow/my-first-deployment\n

    The other two fields are less obvious:

    • path: the path can generally be interpreted as the runtime working directory for the flow. For example, if a deployment references a workflow defined within a Docker image, the path will be the absolute path to the parent directory where that workflow will run anytime the deployment is triggered. This interpretation is more subtle in the case of flows defined in remote filesystems.
    • entrypoint: the entrypoint of a deployment is a relative reference to a function decorated as a flow that exists on some filesystem. It is always specified relative to the path. Entrypoints use Python's standard path-to-object syntax (e.g., path/to/file.py:function_name or simply path:object).

    The entrypoint must reference the same flow as the flow ID.

    Note that Prefect requires that deployments reference flows defined within Python files. Flows defined within interactive REPLs or notebooks cannot currently be deployed as such. They are still valid flows that will be monitored by the API and observable in the UI whenever they are run, but Prefect cannot trigger them.

    Deployments do not contain code definitions

    Deployment metadata references code that exists in potentially diverse locations within your environment. This separation of concerns means that your flow code stays within your storage and execution infrastructure and never lives on the Prefect server or database.

    This is the heart of the Prefect hybrid model: there's a boundary between your proprietary assets, such as your flow code, and the Prefect backend (including Prefect Cloud).

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#scheduling-and-parametrization","title":"Scheduling and parametrization","text":"

    One of the primary motivations for creating deployments of flows is to remotely schedule and trigger them. Just as flows can be called as functions with different input values, so can deployments be triggered or scheduled with different values through the use of parameters.

    The six fields here capture the necessary metadata to perform such actions:

    • schedule: a schedule object. Most of the convenient interfaces for creating deployments allow users to avoid creating this object themselves. For example, when updating a deployment schedule in the UI basic information such as a cron string or interval is all that's required.
    • trigger (Cloud-only): triggers allow you to define event-based rules for running a deployment. For more information see Automations.
    • parameter_openapi_schema: an OpenAPI compatible schema that defines the types and defaults for the flow's parameters. This is used by both the UI and the backend to expose options for creating manual runs as well as type validation.
    • parameters: default values of flow parameters that this deployment will pass on each run. These can be overwritten through a trigger or when manually creating a custom run.
    • enforce_parameter_schema: a boolean flag that determines whether the API should validate the parameters passed to a flow run against the schema defined by parameter_openapi_schema.

    Scheduling is asynchronous and decoupled

    Because deployments are nothing more than metadata, runs can be created at anytime. Note that pausing a schedule, updating your deployment, and other actions reset your auto-scheduled runs.

    Scheduling deployments from within Python flow code

    Note that Prefect provides a run_deployment function that can be used to schedule the run of an existing deployment when your Python code executes.

    from prefect.deployments import run_deployment\n\ndef main():\n    run_deployment(name=\"my_flow_name/my_deployment_name\")\n

    Pass timeout=0 to return immediately and not block.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#versioning-and-bookkeeping","title":"Versioning and bookkeeping","text":"

    Versions, descriptions and tags are omnipresent fields throughout Prefect that can be easy to overlook. However, putting some extra thought into how you use these fields can pay dividends down the road.

    • version: versions are always set by the client and can be any arbitrary string. We recommend tightly coupling this field on your deployments to your software development lifecycle. For example if you leverage git to manage code changes, use either a tag or commit hash in this field. If you don't set a value for the version, Prefect will compute a hash
    • description: the description field of a deployment is a place to provide rich reference material for downstream stakeholders such as intended use and parameter documentation. Markdown formatting will be rendered in the Prefect UI, allowing for section headers, links, tables, and other formatting. If not provided explicitly, Prefect will use the docstring of your flow function as a default value.
    • tags: tags are a mechanism for grouping related work together across a diverse set of objects. Tags set on a deployment will be inherited by that deployment's flow runs. These tags can then be used to filter what runs are displayed on the primary UI dashboard, allowing you to customize different views into your work. In addition, in Prefect Cloud you can easily find objects through searching by tag.

    All of these bits of metadata can be leveraged to great effect by injecting them into the processes that Prefect is orchestrating. For example you can use both run ID and versions to organize files that you produce from your workflows, or by associating your flow run's tags with the metadata of a job it orchestrates. This metadata is available during execution through Prefect runtime.

    Everything has a version

    Deployments aren't the only entity in Prefect with a version attached; both flows and tasks also have versions that can be set through their respective decorators. These versions will be sent to the API anytime the flow or task is run and thereby allow you to audit your changes across all levels.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#workers-and-work-pools","title":"Workers and Work Pools","text":"

    Workers and work pools are an advanced deployment pattern that allow you to dynamically provision infrastructure for each flow run. In addition, the work pool job template interface allows users to create and govern opinionated interfaces to their workflow infrastructure. To do this, a deployment using workers needs to evaluate the following fields:

    • work_pool_name: the name of the work pool this deployment will be associated with. Work pool types mirror infrastructure types and therefore the decision here affects the options available for the other fields.
    • work_queue_name: if you are using work queues to either manage priority or concurrency, you can associate a deployment with a specific queue within a work pool using this field.
    • infra_overrides: often called job_variables within various interfaces, this field allows deployment authors to customize whatever infrastructure options have been exposed on this work pool. This field is often used for things such as Docker image names, Kubernetes annotations and limits, and environment variables.
    • pull_steps: a JSON description of steps that should be performed to retrieve flow code or configuration and prepare the runtime environment for workflow execution.

    Pull steps allow users to highly decouple their workflow architecture. For example, a common use of pull steps is to dynamically pull code from remote filesystems such as GitHub with each run of their deployment.

    For more information see the guide to deploying with a worker.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#two-approaches-to-deployments","title":"Two approaches to deployments","text":"

    There are two primary ways to deploy flows with Prefect, differentiated by how much control Prefect has over the infrastructure in which the flows run.

    In one setup, deploying Prefect flows is analogous to deploying a webserver - users author their workflows and then start a long-running process (often within a Docker container) that is responsible for managing all of the runs for the associated deployment(s).

    In the other setup, you do a little extra up-front work to set up a work pool and a base job template that defines how individual flow runs will be submitted to infrastructure.

    Prefect provides several types of work pools corresponding to different types of infrastructure. Prefect Cloud provides a Prefect Managed work pool option that is the simplest way to run workflows remotely. A cloud-provider account, such as AWS, is not required with a Prefect Managed work pool.

    Some work pool types require a client-side worker to submit job definitions to the appropriate infrastructure with each run.

    Each of these setups can support production workloads. The choice ultimately boils down to your use case and preferences. Read further to decide which setup is best for your situation.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#serving-flows-on-long-lived-infrastructure","title":"Serving flows on long-lived infrastructure","text":"

    When you have several flows running regularly, the serve method of the Flow object or the serve utility is a great option for managing multiple flows simultaneously.

    Once you have authored your flow and decided on its deployment settings as described above, all that's left is to run this long-running process in a location of your choosing. The process will stay in communication with the Prefect API, monitoring for work and submitting each run within an individual subprocess. Note that because runs are submitted to subprocesses, any external infrastructure configuration will need to be setup beforehand and kept associated with this process.

    This approach has many benefits:

    • Users are in complete control of their infrastructure, and anywhere the \"serve\" Python process can run is a suitable deployment environment.
    • It is simple to reason about.
    • Creating deployments requires a minimal set of decisions.
    • Iteration speed is fast.

    However, there are a few reasons you might consider running flows on dynamically provisioned infrastructure with work pools instead:

    • Flows that have expensive infrastructure needs may be more costly in this setup due to the long-running process.
    • Flows with heterogeneous infrastructure needs across runs will be more difficult to configure and schedule.
    • Large volumes of deployments can be harder to track.
    • If your internal team structure requires that deployment authors be members of a different team than the team managing infrastructure, the work pool interface may be preferred.
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/deployments/#dynamically-provisioning-infrastructure-with-work-pools","title":"Dynamically provisioning infrastructure with work pools","text":"

    Work pools allow Prefect to exercise greater control of the infrastructure on which flows run. Options for serverless work pools allow you to scale to zero when workflows aren't running. Prefect even provides you with the ability to provision cloud infrastructure via a single CLI command, if you use a Prefect Cloud push work pool option.

    With work pools:

    • You can configure and monitor infrastructure configuration within the Prefect UI.
    • Infrastructure is ephemeral and dynamically provisioned.
    • Prefect is more infrastructure-aware and therefore collects more event data from your infrastructure by default.
    • Highly decoupled setups are possible.

    You don't have to commit to one approach

    You are not required to use only one of these approaches for your deployments. You can mix and match approaches based on the needs of each flow. Further, you can change the deployment approach for a particular flow as its needs evolve. For example, you might use workers for your expensive machine learning pipelines, but use the serve mechanics for smaller, more frequent file-processing pipelines.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","infrastructure","storage","work pool","worker"],"boost":2},{"location":"concepts/events/","title":"Events","text":"

    An event is a notification of a change. Together, events form a feed of activity recording what's happening across your stack.

    Events power several features in Prefect Cloud, including flow run logs, audit logs, and automations.

    Events can represent API calls, state transitions, or changes in your execution environment or infrastructure.

    Events enable observability into your data stack via the event feed, and the configuration of Prefect's reactivity via automations.

    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#event-specification","title":"Event Specification","text":"

    Events adhere to a structured specification.

    Name Type Required? Description occurred String yes When the event happened event String yes The name of the event that happened resource Object yes The primary Resource this event concerns related Array no A list of additional Resources involved in this event payload Object no An open-ended set of data describing what happened id String yes The client-provided identifier of this event follows String no The ID of an event that is known to have occurred prior to this one.","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#event-grammar","title":"Event Grammar","text":"

    Generally, events have a consistent and informative grammar - an event describes a resource and an action that the resource took or that was taken on that resource. For example, events emitted by Prefect objects take the form of:

    prefect.block.write-method.called\nprefect-cloud.automation.action.executed\nprefect-cloud.user.logged-in\n
    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#event-sources","title":"Event Sources","text":"

    Events are automatically emitted by all Prefect objects, including flows, tasks, deployments, work queues, and logs. Prefect-emitted events will contain the prefect or prefect-cloud resource prefix. Events can also be sent to the Prefect events API via authenticated http request.

    The Prefect SDK provides a method that emits events, for use in arbitrary python code that may not be a task or flow. Running the following code will emit events to Prefect Cloud, which will validate and ingest the event data.

    from prefect.events import emit_event\n\ndef some_function(name: str=\"kiki\") -> None:\n    print(f\"hi {name}!\")\n    emit_event(event=f\"{name}.sent.event!\", resource={\"prefect.resource.id\": f\"coder.{name}\"})\n\nsome_function()\n

    Prefect Cloud offers programmable webhooks to receive HTTP requests from other systems and translate them into events within your workspace. Webhooks can emit pre-defined static events, dynamic events that use portions of the incoming HTTP request, or events derived from CloudEvents.

    Events emitted from any source will appear in the event feed, where you can visualize activity in context and configure automations to react to the presence or absence of it in the future.

    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#resources","title":"Resources","text":"

    Every event has a primary resource, which describes the object that emitted an event. Resources are used as quasi-stable identifiers for sources of events, and are constructed as dot-delimited strings, for example:

    prefect-cloud.automation.5b9c5c3d-6ca0-48d0-8331-79f4b65385b3.action.0\nacme.user.kiki.elt_script_1\nprefect.flow-run.e3755d32-cec5-42ca-9bcd-af236e308ba6\n

    Resources can optionally have additional arbitrary labels which can be used in event aggregation queries, such as:

    \"resource\": {\n    \"prefect.resource.id\": \"prefect-cloud.automation.5b9c5c3d-6ca0-48d0-8331-79f4b65385b3\",\n    \"prefect-cloud.action.type\": \"call-webhook\"\n    }\n

    Events can optionally contain related resources, used to associate the event with other resources, such as in the case that the primary resource acted on or with another resource:

    \"resource\": {\n    \"prefect.resource.id\": \"prefect-cloud.automation.5b9c5c3d-6ca0-48d0-8331-79f4b65385b3.action.0\",\n    \"prefect-cloud.action.type\": \"call-webhook\"\n  },\n\"related\": [\n  {\n      \"prefect.resource.id\": \"prefect-cloud.automation.5b9c5c3d-6ca0-48d0-8331-79f4b65385b3\",\n      \"prefect.resource.role\": \"automation\",\n      \"prefect-cloud.name\": \"webhook_body_demo\",\n      \"prefect-cloud.posture\": \"Reactive\"\n  }\n]\n
    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#events-in-the-cloud-ui","title":"Events in the Cloud UI","text":"

    Prefect Cloud provides an interactive dashboard to analyze and take action on events that occurred in your workspace on the event feed page.

    The event feed is the primary place to view, search, and filter events to understand activity across your stack. Each entry displays data on the resource, related resource, and event that took place.

    You can view more information about an event by clicking into it, where you can view the full details of an event's resource, related resources, and its payload.

    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/events/#reacting-to-events","title":"Reacting to events","text":"

    From an event page, you can configure an automation to trigger on the observation of matching events or a lack of matching events by clicking the automate button in the overflow menu:

    The default trigger configuration will fire every time it sees an event with a matching resource identifier. Advanced configuration is possible via custom triggers.

    ","tags":["UI","dashboard","Prefect Cloud","Observability","Events"],"boost":2},{"location":"concepts/filesystems/","title":"Filesystems","text":"

    A filesystem block is an object that allows you to read and write data from paths. Prefect provides multiple built-in file system types that cover a wide range of use cases.

    • LocalFileSystem
    • RemoteFileSystem
    • Azure
    • GitHub
    • GitLab
    • GCS
    • S3
    • SMB

    Additional file system types are available in Prefect Collections.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#local-filesystem","title":"Local filesystem","text":"

    The LocalFileSystem block enables interaction with the files in your current development environment.

    LocalFileSystem properties include:

    Property Description basepath String path to the location of files on the local filesystem. Access to files outside of the base path will not be allowed.
    from prefect.filesystems import LocalFileSystem\n\nfs = LocalFileSystem(basepath=\"/foo/bar\")\n

    Limited access to local file system

    Be aware that LocalFileSystem access is limited to the exact path provided. This file system may not be ideal for some use cases. The execution environment for your workflows may not have the same file system as the environment you are writing and deploying your code on.

    Use of this file system can limit the availability of results after a flow run has completed or prevent the code for a flow from being retrieved successfully at the start of a run.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#remote-file-system","title":"Remote file system","text":"

    The RemoteFileSystem block enables interaction with arbitrary remote file systems. Under the hood, RemoteFileSystem uses fsspec and supports any file system that fsspec supports.

    RemoteFileSystem properties include:

    Property Description basepath String path to the location of files on the remote filesystem. Access to files outside of the base path will not be allowed. settings Dictionary containing extra parameters required to access the remote file system.

    The file system is specified using a protocol:

    • s3://my-bucket/my-folder/ will use S3
    • gcs://my-bucket/my-folder/ will use GCS
    • az://my-bucket/my-folder/ will use Azure

    For example, to use it with Amazon S3:

    from prefect.filesystems import RemoteFileSystem\n\nblock = RemoteFileSystem(basepath=\"s3://my-bucket/folder/\")\nblock.save(\"dev\")\n

    You may need to install additional libraries to use some remote storage types.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#remotefilesystem-examples","title":"RemoteFileSystem examples","text":"

    How can we use RemoteFileSystem to store our flow code? The following is a use case where we use MinIO as a storage backend:

    from prefect.filesystems import RemoteFileSystem\n\nminio_block = RemoteFileSystem(\n    basepath=\"s3://my-bucket\",\n    settings={\n        \"key\": \"MINIO_ROOT_USER\",\n        \"secret\": \"MINIO_ROOT_PASSWORD\",\n        \"client_kwargs\": {\"endpoint_url\": \"http://localhost:9000\"},\n    },\n)\nminio_block.save(\"minio\")\n
    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#azure","title":"Azure","text":"

    The Azure file system block enables interaction with Azure Datalake and Azure Blob Storage. Under the hood, the Azure block uses adlfs.

    Azure properties include:

    Property Description bucket_path String path to the location of files on the remote filesystem. Access to files outside of the bucket path will not be allowed. azure_storage_connection_string Azure storage connection string. azure_storage_account_name Azure storage account name. azure_storage_account_key Azure storage account key. azure_storage_tenant_id Azure storage tenant ID. azure_storage_client_id Azure storage client ID. azure_storage_client_secret Azure storage client secret. azure_storage_anon Anonymous authentication, disable to use DefaultAzureCredential.

    To create a block:

    from prefect.filesystems import Azure\n\nblock = Azure(bucket_path=\"my-bucket/folder/\")\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb az/dev\n

    You need to install adlfs to use it.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#github","title":"GitHub","text":"

    The GitHub filesystem block enables interaction with GitHub repositories. This block is read-only and works with both public and private repositories.

    GitHub properties include:

    Property Description reference An optional reference to pin to, such as a branch name or tag. repository The URL of a GitHub repository to read from, in either HTTPS or SSH format. access_token A GitHub Personal Access Token (PAT) with repo scope.

    To create a block:

    from prefect.filesystems import GitHub\n\nblock = GitHub(\n    repository=\"https://github.com/my-repo/\",\n    access_token=<my_access_token> # only required for private repos\n)\nblock.get_directory(\"folder-in-repo\") # specify a subfolder of repo\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb github/dev -a\n
    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#gitlabrepository","title":"GitLabRepository","text":"

    The GitLabRepository block is read-only and works with private GitLab repositories.

    GitLabRepository properties include:

    Property Description reference An optional reference to pin to, such as a branch name or tag. repository The URL of a GitLab repository to read from, in either HTTPS or SSH format. credentials A GitLabCredentials block with Personal Access Token (PAT) with read_repository scope.

    To create a block:

    from prefect_gitlab.credentials import GitLabCredentials\nfrom prefect_gitlab.repositories import GitLabRepository\n\ngitlab_creds = GitLabCredentials(token=\"YOUR_GITLAB_ACCESS_TOKEN\")\ngitlab_repo = GitLabRepository(\n    repository=\"https://gitlab.com/yourorg/yourrepo.git\",\n    reference=\"main\",\n    credentials=gitlab_creds,\n)\ngitlab_repo.save(\"dev\")\n

    To use it in a deployment (and apply):

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb gitlab-repository/dev -a\n

    Note that to use this block, you need to install the prefect-gitlab collection.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#gcs","title":"GCS","text":"

    The GCS file system block enables interaction with Google Cloud Storage. Under the hood, GCS uses gcsfs.

    GCS properties include:

    Property Description bucket_path A GCS bucket path service_account_info The contents of a service account keyfile as a JSON string. project The project the GCS bucket resides in. If not provided, the project will be inferred from the credentials or environment.

    To create a block:

    from prefect.filesystems import GCS\n\nblock = GCS(bucket_path=\"my-bucket/folder/\")\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb gcs/dev\n

    You need to install gcsfsto use it.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#s3","title":"S3","text":"

    The S3 file system block enables interaction with Amazon S3. Under the hood, S3 uses s3fs.

    S3 properties include:

    Property Description bucket_path An S3 bucket path aws_access_key_id AWS Access Key ID aws_secret_access_key AWS Secret Access Key

    To create a block:

    from prefect.filesystems import S3\n\nblock = S3(bucket_path=\"my-bucket/folder/\")\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb s3/dev\n

    You need to install s3fsto use this block.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#smb","title":"SMB","text":"

    The SMB file system block enables interaction with SMB shared network storage. Under the hood, SMB uses smbprotocol. Used to connect to Windows-based SMB shares from Linux-based Prefect flows. The SMB file system block is able to copy files, but cannot create directories.

    SMB properties include:

    Property Description basepath String path to the location of files on the remote filesystem. Access to files outside of the base path will not be allowed. smb_host Hostname or IP address where SMB network share is located. smb_port Port for SMB network share (defaults to 445). smb_username SMB username with read/write permissions. smb_password SMB password.

    To create a block:

    from prefect.filesystems import SMB\n\nblock = SMB(basepath=\"my-share/folder/\")\nblock.save(\"dev\")\n

    To use it in a deployment:

    prefect deployment build path/to/flow.py:flow_name --name deployment_name --tag dev -sb smb/dev\n

    You need to install smbprotocol to use it.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#handling-credentials-for-cloud-object-storage-services","title":"Handling credentials for cloud object storage services","text":"

    If you leverage S3, GCS, or Azure storage blocks, and you don't explicitly configure credentials on the respective storage block, those credentials will be inferred from the environment. Make sure to set those either explicitly on the block or as environment variables, configuration files, or IAM roles within both the build and runtime environment for your deployments.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#filesystem-package-dependencies","title":"Filesystem package dependencies","text":"

    A Prefect installation and doesn't include filesystem-specific package dependencies such as s3fs, gcsfs or adlfs. This includes Prefect base Docker images.

    You must ensure that filesystem-specific libraries are installed in an execution environment where they will be used by flow runs.

    In Dockerized deployments using the Prefect base image, you can leverage the EXTRA_PIP_PACKAGES environment variable. Those dependencies will be installed at runtime within your Docker container or Kubernetes Job before the flow starts running.

    In Dockerized deployments using a custom image, you must include the filesystem-specific package dependency in your image.

    Here is an example from a deployment YAML file showing how to specify the installation of s3fs from into your image:

    infrastructure:\n  type: docker-container\n  env:\n    EXTRA_PIP_PACKAGES: s3fs  # could be gcsfs, adlfs, etc.\n

    You may specify multiple dependencies by providing a comma-delimted list.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#saving-and-loading-file-systems","title":"Saving and loading file systems","text":"

    Configuration for a file system can be saved to the Prefect API. For example:

    fs = RemoteFileSystem(basepath=\"s3://my-bucket/folder/\")\nfs.write_path(\"foo\", b\"hello\")\nfs.save(\"dev-s3\")\n

    This file system can be retrieved for later use with load.

    fs = RemoteFileSystem.load(\"dev-s3\")\nfs.read_path(\"foo\")  # b'hello'\n
    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/filesystems/#readable-and-writable-file-systems","title":"Readable and writable file systems","text":"

    Prefect provides two abstract file system types, ReadableFileSystem and WriteableFileSystem.

    • All readable file systems must implement read_path, which takes a file path to read content from and returns bytes.
    • All writeable file systems must implement write_path which takes a file path and content and writes the content to the file as bytes.

    A file system may implement both of these types.

    ","tags":["filesystems","storage","deployments","LocalFileSystem","RemoteFileSystem"],"boost":0.5},{"location":"concepts/flows/","title":"Flows","text":"

    Flows are the most central Prefect object. A flow is a container for workflow logic as-code and allows users to configure how their workflows behave. Flows are defined as Python functions, and any Python function is eligible to be a flow.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#flows-overview","title":"Flows overview","text":"

    Flows can be thought of as special types of functions. They can take inputs, perform work, and return an output. In fact, you can turn any function into a Prefect flow by adding the @flow decorator. When a function becomes a flow, its behavior changes, giving it the following advantages:

    • Every invocation of this function is tracked and all state transitions are reported to the API, allowing observation of flow execution.
    • Input arguments are automatically type checked and coerced to the appropriate types.
    • Retries can be performed on failure.
    • Timeouts can be enforced to prevent unintentional, long-running workflows.

    Flows also take advantage of automatic Prefect logging to capture details about flow runs such as run time and final state.

    Flows can include calls to tasks as well as to other flows, which Prefect calls \"subflows\" in this context. Flows may be defined within modules and imported for use as subflows in your flow definitions.

    Deployments elevate individual workflows from functions that you call manually to API-managed entities.

    Tasks must be called from flows

    All tasks must be called from within a flow. Tasks may not be called from other tasks.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#flow-runs","title":"Flow runs","text":"

    A flow run represents a single execution of the flow.

    You can create a flow run by calling the flow manually. For example, by running a Python script or importing the flow into an interactive session and calling it.

    You can also create a flow run by:

    • Using external schedulers such as cron to invoke a flow function
    • Creating a deployment on Prefect Cloud or a locally run Prefect server.
    • Creating a flow run for the deployment via a schedule, the Prefect UI, or the Prefect API.

    However you run the flow, the Prefect API monitors the flow run, capturing flow run state for observability.

    When you run a flow that contains tasks or additional flows, Prefect will track the relationship of each child run to the parent flow run.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#writing-flows","title":"Writing flows","text":"

    The @flow decorator is used to designate a flow:

    from prefect import flow\n\n@flow\ndef my_flow():\n    return\n

    There are no rigid rules for what code you include within a flow definition - all valid Python is acceptable.

    Flows are uniquely identified by name. You can provide a name parameter value for the flow. If you don't provide a name, Prefect uses the flow function name.

    @flow(name=\"My Flow\")\ndef my_flow():\n    return\n

    Flows can call tasks to allow Prefect to orchestrate and track more granular units of work:

    from prefect import flow, task\n\n@task\ndef print_hello(name):\n    print(f\"Hello {name}!\")\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    print_hello(name)\n

    Flows and tasks

    There's nothing stopping you from putting all of your code in a single flow function \u2014 Prefect will happily run it!

    However, organizing your workflow code into smaller flow and task units lets you take advantage of Prefect features like retries, more granular visibility into runtime state, the ability to determine final state regardless of individual task state, and more.

    In addition, if you put all of your workflow logic in a single flow function and any line of code fails, the entire flow will fail and must be retried from the beginning. This can be avoided by breaking up the code into multiple tasks.

    You may call any number of other tasks, subflows, and even regular Python functions within your flow. You can pass parameters to your flow function that will be used elsewhere in the workflow, and Prefect will report on the progress and final state of any invocation.

    Prefect encourages \"small tasks\" \u2014 each one should represent a single logical step of your workflow. This allows Prefect to better contain task failures.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#flow-settings","title":"Flow settings","text":"

    Flows allow a great deal of configuration by passing arguments to the decorator. Flows accept the following optional settings.

    Argument Description description An optional string description for the flow. If not provided, the description will be pulled from the docstring for the decorated function. name An optional name for the flow. If not provided, the name will be inferred from the function. retries An optional number of times to retry on flow run failure. retry_delay_seconds An optional number of seconds to wait before retrying the flow after failure. This is only applicable if retries is nonzero. flow_run_name An optional name to distinguish runs of this flow; this name can be provided as a string template with the flow's parameters as variables; this name can also be provided as a function that returns a string. task_runner An optional task runner to use for task execution within the flow when you .submit() tasks. If not provided and you .submit() tasks, the ConcurrentTaskRunner will be used. timeout_seconds An optional number of seconds indicating a maximum runtime for the flow. If the flow exceeds this runtime, it will be marked as failed. Flow execution may continue until the next task is called. validate_parameters Boolean indicating whether parameters passed to flows are validated by Pydantic. Default is True. version An optional version string for the flow. If not provided, we will attempt to create a version string as a hash of the file containing the wrapped function. If the file cannot be located, the version will be null.

    For example, you can provide a name value for the flow. Here we've also used the optional description argument and specified a non-default task runner.

    from prefect import flow\nfrom prefect.task_runners import SequentialTaskRunner\n\n@flow(name=\"My Flow\",\n      description=\"My flow using SequentialTaskRunner\",\n      task_runner=SequentialTaskRunner())\ndef my_flow():\n    return\n

    You can also provide the description as the docstring on the flow function.

    @flow(name=\"My Flow\",\n      task_runner=SequentialTaskRunner())\ndef my_flow():\n    \"\"\"My flow using SequentialTaskRunner\"\"\"\n    return\n

    You can distinguish runs of this flow by providing a flow_run_name. This setting accepts a string that can optionally contain templated references to the parameters of your flow. The name will be formatted using Python's standard string formatting syntax as can be seen here:

    import datetime\nfrom prefect import flow\n\n@flow(flow_run_name=\"{name}-on-{date:%A}\")\ndef my_flow(name: str, date: datetime.datetime):\n    pass\n\n# creates a flow run called 'marvin-on-Thursday'\nmy_flow(name=\"marvin\", date=datetime.datetime.utcnow())\n

    Additionally this setting also accepts a function that returns a string for the flow run name:

    import datetime\nfrom prefect import flow\n\ndef generate_flow_run_name():\n    date = datetime.datetime.utcnow()\n\n    return f\"{date:%A}-is-a-nice-day\"\n\n@flow(flow_run_name=generate_flow_run_name)\ndef my_flow(name: str):\n    pass\n\n# creates a flow run called 'Thursday-is-a-nice-day'\nmy_flow(name=\"marvin\")\n

    If you need access to information about the flow, use the prefect.runtime module. For example:

    from prefect import flow\nfrom prefect.runtime import flow_run\n\ndef generate_flow_run_name():\n    flow_name = flow_run.flow_name\n\n    parameters = flow_run.parameters\n    name = parameters[\"name\"]\n    limit = parameters[\"limit\"]\n\n    return f\"{flow_name}-with-{name}-and-{limit}\"\n\n@flow(flow_run_name=generate_flow_run_name)\ndef my_flow(name: str, limit: int = 100):\n    pass\n\n# creates a flow run called 'my-flow-with-marvin-and-100'\nmy_flow(name=\"marvin\")\n

    Note that validate_parameters will check that input values conform to the annotated types on the function. Where possible, values will be coerced into the correct type. For example, if a parameter is defined as x: int and \"5\" is passed, it will be resolved to 5. If set to False, no validation will be performed on flow parameters.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#separating-logic-into-tasks","title":"Separating logic into tasks","text":"

    The simplest workflow is just a @flow function that does all the work of the workflow.

    from prefect import flow\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    print(f\"Hello {name}!\")\n\nhello_world(\"Marvin\")\n

    When you run this flow, you'll see output like the following:

    $ python hello.py\n15:11:23.594 | INFO    | prefect.engine - Created flow run 'benevolent-donkey' for flow 'hello-world'\n15:11:23.594 | INFO    | Flow run 'benevolent-donkey' - Using task runner 'ConcurrentTaskRunner'\nHello Marvin!\n15:11:24.447 | INFO    | Flow run 'benevolent-donkey' - Finished in state Completed()\n

    A better practice is to create @task functions that do the specific work of your flow, and use your @flow function as the conductor that orchestrates the flow of your application:

    from prefect import flow, task\n\n@task(name=\"Print Hello\")\ndef print_hello(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    message = print_hello(name)\n\nhello_world(\"Marvin\")\n

    When you run this flow, you'll see the following output, which illustrates how the work is encapsulated in a task run.

    $ python hello.py\n15:15:58.673 | INFO    | prefect.engine - Created flow run 'loose-wolverine' for flow 'Hello Flow'\n15:15:58.674 | INFO    | Flow run 'loose-wolverine' - Using task runner 'ConcurrentTaskRunner'\n15:15:58.973 | INFO    | Flow run 'loose-wolverine' - Created task run 'Print Hello-84f0fe0e-0' for task 'Print Hello'\nHello Marvin!\n15:15:59.037 | INFO    | Task run 'Print Hello-84f0fe0e-0' - Finished in state Completed()\n15:15:59.568 | INFO    | Flow run 'loose-wolverine' - Finished in state Completed('All states completed.')\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#visualizing-flow-structure","title":"Visualizing flow structure","text":"

    You can get a quick sense of the structure of your flow using the .visualize() method on your flow. Calling this method will attempt to produce a schematic diagram of your flow and tasks without actually running your flow code.

    Functions and code not inside of flows or tasks will still be run when calling .visualize(). This may have unintended consequences. Place your code into tasks to avoid unintended execution.

    To use the visualize() method, Graphviz must be installed and on your PATH. Please install Graphviz from http://www.graphviz.org/download/. And note: just installing the graphviz python package is not sufficient.

    from prefect import flow, task\n\n@task(name=\"Print Hello\")\ndef print_hello(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@task(name=\"Print Hello Again\")\ndef print_hello_again(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    message = print_hello(name)\n    message2 = print_hello_again(message)\n\nhello_world.visualize()\n

    Prefect cannot automatically produce a schematic for dynamic workflows, such as those with loops or if/else control flow. In this case, you can provide tasks with mock return values for use in the visualize() call.

    from prefect import flow, task\n@task(viz_return_value=[4])\ndef get_list():\n    return [1, 2, 3]\n\n@task\ndef append_one(n):\n    return n.append(6)\n\n@flow\ndef viz_return_value_tracked():\n    l = get_list()\n    for num in range(3):\n        l.append(5)\n        append_one(l)\n\nviz_return_value_tracked.visualize()\n

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#composing-flows","title":"Composing flows","text":"

    A subflow run is created when a flow function is called inside the execution of another flow. The primary flow is the \"parent\" flow. The flow created within the parent is the \"child\" flow or \"subflow.\"

    Subflow runs behave like normal flow runs. There is a full representation of the flow run in the backend as if it had been called separately. When a subflow starts, it will create a new task runner for tasks within the subflow. When the subflow completes, the task runner is shut down.

    Subflows will block execution of the parent flow until completion. However, asynchronous subflows can be run in parallel by using AnyIO task groups or asyncio.gather.

    Subflows differ from normal flows in that they will resolve any passed task futures into data. This allows data to be passed from the parent flow to the child easily.

    The relationship between a child and parent flow is tracked by creating a special task run in the parent flow. This task run will mirror the state of the child flow run.

    A task that represents a subflow will be annotated as such in its state_details via the presence of a child_flow_run_id field. A subflow can be identified via the presence of a parent_task_run_id on state_details.

    You can define multiple flows within the same file. Whether running locally or via a deployment, you must indicate which flow is the entrypoint for a flow run.

    Cancelling subflow runs

    Inline subflow runs, specifically those created without run_deployment, cannot be cancelled without cancelling their parent flow run. If you may need to cancel a subflow run independent of its parent flow run, we recommend deploying it separately and starting it using the run_deployment function.

    from prefect import flow, task\n\n@task(name=\"Print Hello\")\ndef print_hello(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@flow(name=\"Subflow\")\ndef my_subflow(msg):\n    print(f\"Subflow says: {msg}\")\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    message = print_hello(name)\n    my_subflow(message)\n\nhello_world(\"Marvin\")\n

    You can also define flows or tasks in separate modules and import them for usage. For example, here's a simple subflow module:

    from prefect import flow, task\n\n@flow(name=\"Subflow\")\ndef my_subflow(msg):\n    print(f\"Subflow says: {msg}\")\n

    Here's a parent flow that imports and uses my_subflow() as a subflow:

    from prefect import flow, task\nfrom subflow import my_subflow\n\n@task(name=\"Print Hello\")\ndef print_hello(name):\n    msg = f\"Hello {name}!\"\n    print(msg)\n    return msg\n\n@flow(name=\"Hello Flow\")\ndef hello_world(name=\"world\"):\n    message = print_hello(name)\n    my_subflow(message)\n\nhello_world(\"Marvin\")\n

    Running the hello_world() flow (in this example from the file hello.py) creates a flow run like this:

    $ python hello.py\n15:19:21.651 | INFO    | prefect.engine - Created flow run 'daft-cougar' for flow 'Hello Flow'\n15:19:21.651 | INFO    | Flow run 'daft-cougar' - Using task runner 'ConcurrentTaskRunner'\n15:19:21.945 | INFO    | Flow run 'daft-cougar' - Created task run 'Print Hello-84f0fe0e-0' for task 'Print Hello'\nHello Marvin!\n15:19:22.055 | INFO    | Task run 'Print Hello-84f0fe0e-0' - Finished in state Completed()\n15:19:22.107 | INFO    | Flow run 'daft-cougar' - Created subflow run 'ninja-duck' for flow 'Subflow'\nSubflow says: Hello Marvin!\n15:19:22.794 | INFO    | Flow run 'ninja-duck' - Finished in state Completed()\n15:19:23.215 | INFO    | Flow run 'daft-cougar' - Finished in state Completed('All states completed.')\n

    Subflows or tasks?

    In Prefect you can call tasks or subflows to do work within your workflow, including passing results from other tasks to your subflow. So a common question is:

    \"When should I use a subflow instead of a task?\"

    We recommend writing tasks that do a discrete, specific piece of work in your workflow: calling an API, performing a database operation, analyzing or transforming a data point. Prefect tasks are well suited to parallel or distributed execution using distributed computation frameworks such as Dask or Ray. For troubleshooting, the more granular you create your tasks, the easier it is to find and fix issues should a task fail.

    Subflows enable you to group related tasks within your workflow. Here are some scenarios where you might choose to use a subflow rather than calling tasks individually:

    • Observability: Subflows, like any other flow run, have first-class observability within the Prefect UI and Prefect Cloud. You'll see subflow status in the Flow Runs dashboard rather than having to dig down into the tasks within a specific flow run. See Final state determination for some examples of leveraging task state within flows.
    • Conditional flows: If you have a group of tasks that run only under certain conditions, you can group them within a subflow and conditionally run the subflow rather than each task individually.
    • Parameters: Flows have first-class support for parameterization, making it easy to run the same group of tasks in different use cases by simply passing different parameters to the subflow in which they run.
    • Task runners: Subflows enable you to specify the task runner used for tasks within the flow. For example, if you want to optimize parallel execution of certain tasks with Dask, you can group them in a subflow that uses the Dask task runner. You can use a different task runner for each subflow.
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#parameters","title":"Parameters","text":"

    Flows can be called with both positional and keyword arguments. These arguments are resolved at runtime into a dictionary of parameters mapping name to value. These parameters are stored by the Prefect orchestration engine on the flow run object.

    Prefect API requires keyword arguments

    When creating flow runs from the Prefect API, parameter names must be specified when overriding defaults \u2014 they cannot be positional.

    Type hints provide an easy way to enforce typing on your flow parameters via pydantic. This means any pydantic model used as a type hint within a flow will be coerced automatically into the relevant object type:

    from prefect import flow\nfrom pydantic import BaseModel\n\nclass Model(BaseModel):\n    a: int\n    b: float\n    c: str\n\n@flow\ndef model_validator(model: Model):\n    print(model)\n

    Note that parameter values can be provided to a flow via API using a deployment. Flow run parameters sent to the API on flow calls are coerced to a serializable form. Type hints on your flow functions provide you a way of automatically coercing JSON provided values to their appropriate Python representation.

    For example, to automatically convert something to a datetime:

    from prefect import flow\nfrom datetime import datetime\n\n@flow\ndef what_day_is_it(date: datetime = None):\n    if date is None:\n        date = datetime.utcnow()\n    print(f\"It was {date.strftime('%A')} on {date.isoformat()}\")\n\nwhat_day_is_it(\"2021-01-01T02:00:19.180906\")\n# It was Friday on 2021-01-01T02:00:19.180906\n

    Parameters are validated before a flow is run. If a flow call receives invalid parameters, a flow run is created in a Failed state. If a flow run for a deployment receives invalid parameters, it will move from a Pending state to a Failed without entering a Running state.

    Flow run parameters cannot exceed 512kb in size

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#final-state-determination","title":"Final state determination","text":"

    Prerequisite

    Read the documentation about states before proceeding with this section.

    The final state of the flow is determined by its return value. The following rules apply:

    • If an exception is raised directly in the flow function, the flow run is marked as failed.
    • If the flow does not return a value (or returns None), its state is determined by the states of all of the tasks and subflows within it.
    • If any task run or subflow run failed, then the final flow run state is marked as FAILED.
    • If any task run was cancelled, then the final flow run state is marked as CANCELLED.
    • If a flow returns a manually created state, it is used as the state of the final flow run. This allows for manual determination of final state.
    • If the flow run returns any other object, then it is marked as completed.

    The following examples illustrate each of these cases:

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#raise-an-exception","title":"Raise an exception","text":"

    If an exception is raised within the flow function, the flow is immediately marked as failed.

    from prefect import flow\n\n@flow\ndef always_fails_flow():\n    raise ValueError(\"This flow immediately fails\")\n\nalways_fails_flow()\n

    Running this flow produces the following result:

    22:22:36.864 | INFO    | prefect.engine - Created flow run 'acrid-tuatara' for flow 'always-fails-flow'\n22:22:36.864 | INFO    | Flow run 'acrid-tuatara' - Starting 'ConcurrentTaskRunner'; submitted tasks will be run concurrently...\n22:22:37.060 | ERROR   | Flow run 'acrid-tuatara' - Encountered exception during execution:\nTraceback (most recent call last):...\nValueError: This flow immediately fails\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-none","title":"Return none","text":"

    A flow with no return statement is determined by the state of all of its task runs.

    from prefect import flow, task\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I fail successfully\")\n\n@task\ndef always_succeeds_task():\n    print(\"I'm fail safe!\")\n    return \"success\"\n\n@flow\ndef always_fails_flow():\n    always_fails_task.submit().result(raise_on_failure=False)\n    always_succeeds_task()\n\nif __name__ == \"__main__\":\n    always_fails_flow()\n

    Running this flow produces the following result:

    18:32:05.345 | INFO    | prefect.engine - Created flow run 'auburn-lionfish' for flow 'always-fails-flow'\n18:32:05.346 | INFO    | Flow run 'auburn-lionfish' - Starting 'ConcurrentTaskRunner'; submitted tasks will be run concurrently...\n18:32:05.582 | INFO    | Flow run 'auburn-lionfish' - Created task run 'always_fails_task-96e4be14-0' for task 'always_fails_task'\n18:32:05.582 | INFO    | Flow run 'auburn-lionfish' - Submitted task run 'always_fails_task-96e4be14-0' for execution.\n18:32:05.610 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Encountered exception during execution:\nTraceback (most recent call last):\n  ...\nValueError: I fail successfully\n18:32:05.638 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Finished in state Failed('Task run encountered an exception.')\n18:32:05.658 | INFO    | Flow run 'auburn-lionfish' - Created task run 'always_succeeds_task-9c27db32-0' for task 'always_succeeds_task'\n18:32:05.659 | INFO    | Flow run 'auburn-lionfish' - Executing 'always_succeeds_task-9c27db32-0' immediately...\nI'm fail safe!\n18:32:05.703 | INFO    | Task run 'always_succeeds_task-9c27db32-0' - Finished in state Completed()\n18:32:05.730 | ERROR   | Flow run 'auburn-lionfish' - Finished in state Failed('1/2 states failed.')\nTraceback (most recent call last):\n  ...\nValueError: I fail successfully\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-a-future","title":"Return a future","text":"

    If a flow returns one or more futures, the final state is determined based on the underlying states.

    from prefect import flow, task\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I fail successfully\")\n\n@task\ndef always_succeeds_task():\n    print(\"I'm fail safe!\")\n    return \"success\"\n\n@flow\ndef always_succeeds_flow():\n    x = always_fails_task.submit().result(raise_on_failure=False)\n    y = always_succeeds_task.submit(wait_for=[x])\n    return y\n\nif __name__ == \"__main__\":\n    always_succeeds_flow()\n

    Running this flow produces the following result \u2014 it succeeds because it returns the future of the task that succeeds:

    18:35:24.965 | INFO    | prefect.engine - Created flow run 'whispering-guan' for flow 'always-succeeds-flow'\n18:35:24.965 | INFO    | Flow run 'whispering-guan' - Starting 'ConcurrentTaskRunner'; submitted tasks will be run concurrently...\n18:35:25.204 | INFO    | Flow run 'whispering-guan' - Created task run 'always_fails_task-96e4be14-0' for task 'always_fails_task'\n18:35:25.205 | INFO    | Flow run 'whispering-guan' - Submitted task run 'always_fails_task-96e4be14-0' for execution.\n18:35:25.232 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Encountered exception during execution:\nTraceback (most recent call last):\n  ...\nValueError: I fail successfully\n18:35:25.265 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Finished in state Failed('Task run encountered an exception.')\n18:35:25.289 | INFO    | Flow run 'whispering-guan' - Created task run 'always_succeeds_task-9c27db32-0' for task 'always_succeeds_task'\n18:35:25.289 | INFO    | Flow run 'whispering-guan' - Submitted task run 'always_succeeds_task-9c27db32-0' for execution.\nI'm fail safe!\n18:35:25.335 | INFO    | Task run 'always_succeeds_task-9c27db32-0' - Finished in state Completed()\n18:35:25.362 | INFO    | Flow run 'whispering-guan' - Finished in state Completed('All states completed.')\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-multiple-states-or-futures","title":"Return multiple states or futures","text":"

    If a flow returns a mix of futures and states, the final state is determined by resolving all futures to states, then determining if any of the states are not COMPLETED.

    from prefect import task, flow\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I am bad task\")\n\n@task\ndef always_succeeds_task():\n    return \"foo\"\n\n@flow\ndef always_succeeds_flow():\n    return \"bar\"\n\n@flow\ndef always_fails_flow():\n    x = always_fails_task()\n    y = always_succeeds_task()\n    z = always_succeeds_flow()\n    return x, y, z\n

    Running this flow produces the following result. It fails because one of the three returned futures failed. Note that the final state is Failed, but the states of each of the returned futures is included in the flow state:

    20:57:51.547 | INFO    | prefect.engine - Created flow run 'impartial-gorilla' for flow 'always-fails-flow'\n20:57:51.548 | INFO    | Flow run 'impartial-gorilla' - Using task runner 'ConcurrentTaskRunner'\n20:57:51.645 | INFO    | Flow run 'impartial-gorilla' - Created task run 'always_fails_task-58ea43a6-0' for task 'always_fails_task'\n20:57:51.686 | INFO    | Flow run 'impartial-gorilla' - Created task run 'always_succeeds_task-c9014725-0' for task 'always_succeeds_task'\n20:57:51.727 | ERROR   | Task run 'always_fails_task-58ea43a6-0' - Encountered exception during execution:\nTraceback (most recent call last):...\nValueError: I am bad task\n20:57:51.787 | INFO    | Task run 'always_succeeds_task-c9014725-0' - Finished in state Completed()\n20:57:51.808 | INFO    | Flow run 'impartial-gorilla' - Created subflow run 'unbiased-firefly' for flow 'always-succeeds-flow'\n20:57:51.884 | ERROR   | Task run 'always_fails_task-58ea43a6-0' - Finished in state Failed('Task run encountered an exception.')\n20:57:52.438 | INFO    | Flow run 'unbiased-firefly' - Finished in state Completed()\n20:57:52.811 | ERROR   | Flow run 'impartial-gorilla' - Finished in state Failed('1/3 states failed.')\nFailed(message='1/3 states failed.', type=FAILED, result=(Failed(message='Task run encountered an exception.', type=FAILED, result=ValueError('I am bad task'), task_run_id=5fd4c697-7c4c-440d-8ebc-dd9c5bbf2245), Completed(message=None, type=COMPLETED, result='foo', task_run_id=df9b6256-f8ac-457c-ba69-0638ac9b9367), Completed(message=None, type=COMPLETED, result='bar', task_run_id=cfdbf4f1-dccd-4816-8d0f-128750017d0c)), flow_run_id=6d2ec094-001a-4cb0-a24e-d2051db6318d)\n

    Returning multiple states

    When returning multiple states, they must be contained in a set, list, or tuple. If other collection types are used, the result of the contained states will not be checked.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-a-manual-state","title":"Return a manual state","text":"

    If a flow returns a manually created state, the final state is determined based on the return value.

    from prefect import task, flow\nfrom prefect.states import Completed, Failed\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I fail successfully\")\n\n@task\ndef always_succeeds_task():\n    print(\"I'm fail safe!\")\n    return \"success\"\n\n@flow\ndef always_succeeds_flow():\n    x = always_fails_task.submit()\n    y = always_succeeds_task.submit()\n    if y.result() == \"success\":\n        return Completed(message=\"I am happy with this result\")\n    else:\n        return Failed(message=\"How did this happen!?\")\n\nif __name__ == \"__main__\":\n    always_succeeds_flow()\n

    Running this flow produces the following result.

    18:37:42.844 | INFO    | prefect.engine - Created flow run 'lavender-elk' for flow 'always-succeeds-flow'\n18:37:42.845 | INFO    | Flow run 'lavender-elk' - Starting 'ConcurrentTaskRunner'; submitted tasks will be run concurrently...\n18:37:43.125 | INFO    | Flow run 'lavender-elk' - Created task run 'always_fails_task-96e4be14-0' for task 'always_fails_task'\n18:37:43.126 | INFO    | Flow run 'lavender-elk' - Submitted task run 'always_fails_task-96e4be14-0' for execution.\n18:37:43.162 | INFO    | Flow run 'lavender-elk' - Created task run 'always_succeeds_task-9c27db32-0' for task 'always_succeeds_task'\n18:37:43.163 | INFO    | Flow run 'lavender-elk' - Submitted task run 'always_succeeds_task-9c27db32-0' for execution.\n18:37:43.175 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Encountered exception during execution:\nTraceback (most recent call last):\n  ...\nValueError: I fail successfully\nI'm fail safe!\n18:37:43.217 | ERROR   | Task run 'always_fails_task-96e4be14-0' - Finished in state Failed('Task run encountered an exception.')\n18:37:43.236 | INFO    | Task run 'always_succeeds_task-9c27db32-0' - Finished in state Completed()\n18:37:43.264 | INFO    | Flow run 'lavender-elk' - Finished in state Completed('I am happy with this result')\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#return-an-object","title":"Return an object","text":"

    If the flow run returns any other object, then it is marked as completed.

    from prefect import task, flow\n\n@task\ndef always_fails_task():\n    raise ValueError(\"I fail successfully\")\n\n@flow\ndef always_succeeds_flow():\n    always_fails_task().submit()\n    return \"foo\"\n\nif __name__ == \"__main__\":\n    always_succeeds_flow()\n

    Running this flow produces the following result.

    21:02:45.715 | INFO    | prefect.engine - Created flow run 'sparkling-pony' for flow 'always-succeeds-flow'\n21:02:45.715 | INFO    | Flow run 'sparkling-pony' - Using task runner 'ConcurrentTaskRunner'\n21:02:45.816 | INFO    | Flow run 'sparkling-pony' - Created task run 'always_fails_task-58ea43a6-0' for task 'always_fails_task'\n21:02:45.853 | ERROR   | Task run 'always_fails_task-58ea43a6-0' - Encountered exception during execution:\nTraceback (most recent call last):...\nValueError: I am bad task\n21:02:45.879 | ERROR   | Task run 'always_fails_task-58ea43a6-0' - Finished in state Failed('Task run encountered an exception.')\n21:02:46.593 | INFO    | Flow run 'sparkling-pony' - Finished in state Completed()\nCompleted(message=None, type=COMPLETED, result='foo', flow_run_id=7240e6f5-f0a8-4e00-9440-a7b33fb51153)\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#serving-a-flow","title":"Serving a flow","text":"

    The simplest way to create a deployment for your flow is by calling its serve method. This method creates a deployment for the flow and starts a long-running process that monitors for work from the Prefect server. When work is found, it is executed within its own isolated subprocess.

    hello_world.py
    from prefect import flow\n\n\n@flow(log_prints=True)\ndef hello_world(name: str = \"world\", goodbye: bool = False):\n    print(f\"Hello {name} from Prefect! \ud83e\udd17\")\n\n    if goodbye:\n        print(f\"Goodbye {name}!\")\n\n\nif __name__ == \"__main__\":\n    # creates a deployment and stays running to monitor for work instructions generated on the server\n\n    hello_world.serve(name=\"my-first-deployment\",\n                      tags=[\"onboarding\"],\n                      parameters={\"goodbye\": True},\n                      interval=60)\n

    This interface provides all of the configuration needed for a deployment with no strong infrastructure requirements:

    • schedules
    • event triggers
    • metadata such as tags and description
    • default parameter values

    Schedules are auto-paused on shutdown

    By default, stopping the process running flow.serve will pause the schedule for the deployment (if it has one). When running this in environments where restarts are expected use the pause_on_shutdown=False flag to prevent this behavior:

    if __name__ == \"__main__\":\n    hello_world.serve(name=\"my-first-deployment\",\n                      tags=[\"onboarding\"],\n                      parameters={\"goodbye\": True},\n                      pause_on_shutdown=False,\n                      interval=60)\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#serving-multiple-flows-at-once","title":"Serving multiple flows at once","text":"

    You can take this further and serve multiple flows with the same process using the serve utility along with the to_deployment method of flows:

    import time\nfrom prefect import flow, serve\n\n\n@flow\ndef slow_flow(sleep: int = 60):\n    \"Sleepy flow - sleeps the provided amount of time (in seconds).\"\n    time.sleep(sleep)\n\n\n@flow\ndef fast_flow():\n    \"Fastest flow this side of the Mississippi.\"\n    return\n\n\nif __name__ == \"__main__\":\n    slow_deploy = slow_flow.to_deployment(name=\"sleeper\", interval=45)\n    fast_deploy = fast_flow.to_deployment(name=\"fast\")\n    serve(slow_deploy, fast_deploy)\n

    The behavior and interfaces are identical to the single flow case.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#retrieve-a-flow-from-remote-storage","title":"Retrieve a flow from remote storage","text":"

    Flows can be retrieved from remote storage using the flow.from_source method.

    flow.from_source accepts a git repository URL and an entrypoint pointing to the flow to load from the repository:

    load_from_url.py
    from prefect import flow\n\nmy_flow = flow.from_source(\n    source=\"https://github.com/org/repo.git\",\n    entrypoint=\"flows.py:my_flow\"\n)\n\nif __name__ == \"__main__\":\n    my_flow()\n

    A flow entrypoint is the path to the file the flow is located in and the name of the flow function separated by a colon.

    If you need additional configuration, such as specifying a private repository, you can provide a GitRepository instead of URL:

    load_from_storage.py
    from prefect import flow\nfrom prefect.runner.storage import GitRepository\nfrom prefect.blocks.system import Secret\n\nmy_flow = flow.from_source(\n    source=GitRepository(\n        url=\"https://github.com/org/private-repo.git\",\n        branch=\"dev\",\n        credentials={\n            \"access_token\": Secret.load(\"github-access-token\").get()\n        }\n    ),\n    entrypoint=\"flows.py:my_flow\"\n)\n\nif __name__ == \"__main__\":\n    my_flow()\n

    You can serve loaded flows

    Flows loaded from remote storage can be served using the same serve method as local flows:

    serve_loaded_flow.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=\"https://github.com/org/repo.git\",\n        entrypoint=\"flows.py:my_flow\"\n    ).serve(name=\"my-deployment\")\n

    When you serve a flow loaded from remote storage, the serving process will periodically poll your remote storage for updates to the flow's code. This pattern allows you to update your flow code without restarting the serving process.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#pausing-or-suspending-a-flow-run","title":"Pausing or suspending a flow run","text":"

    Prefect provides you with the ability to halt a flow run with two functions that are similar, but slightly different. When a flow run is paused, code execution is stopped and the process continues to run. When a flow run is suspended, code execution is stopped and so is the process.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#pausing-a-flow-run","title":"Pausing a flow run","text":"

    Prefect enables pausing an in-progress flow run for manual approval. Prefect exposes this functionality via the pause_flow_run and resume_flow_run functions, as well as the Prefect UI.

    Timeouts

    Paused flow runs time out after one hour by default. After the timeout, the flow run will fail with a message saying it paused and never resumed. You can specify a different timeout period in seconds using the timeout parameter.

    Most simply, pause_flow_run can be called inside a flow:

    from prefect import task, flow, pause_flow_run, resume_flow_run\n\n@task\nasync def marvin_setup():\n    return \"a raft of ducks walk into a bar...\"\n\n\n@task\nasync def marvin_punchline():\n    return \"it's a wonder none of them ducked!\"\n\n\n@flow\nasync def inspiring_joke():\n    await marvin_setup()\n    await pause_flow_run(timeout=600)  # pauses for 10 minutes\n    await marvin_punchline()\n

    You can also implement conditional pauses:

    from prefect import task, flow, pause_flow_run\n\n@task\ndef task_one():\n    for i in range(3):\n        sleep(1)\n        print(i)\n\n@flow\ndef my_flow():\n    terminal_state = task_one.submit(return_state=True)\n    if terminal_state.type == StateType.COMPLETED:\n        print(\"Task one succeeded! Pausing flow run..\")\n        pause_flow_run(timeout=2) \n    else:\n        print(\"Task one failed. Skipping pause flow run..\")\n

    Calling this flow will block code execution after the first task and wait for resumption to deliver the punchline.

    await inspiring_joke()\n> \"a raft of ducks walk into a bar...\"\n

    Paused flow runs can be resumed by clicking the Resume button in the Prefect UI or calling the resume_flow_run utility via client code.

    resume_flow_run(FLOW_RUN_ID)\n

    The paused flow run will then finish!

    > \"it's a wonder none of them ducked!\"\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#suspending-a-flow-run","title":"Suspending a flow run","text":"

    Similar to pausing a flow run, Prefect enables suspending an in-progress flow run.

    The difference between pausing and suspending a flow run

    There is an important difference between pausing and suspending a flow run. When you pause a flow run, the flow code is still running but is blocked until someone resumes the flow. This is not the case with suspending a flow run! When you suspend a flow run, the flow exits completely and the infrastructure running it (e.g., a Kubernetes Job) tears down.

    This means that you can suspend flow runs to save costs instead of paying for long-running infrastructure. However, when the flow run resumes, the flow code will execute again from the beginning of the flow, so you should use tasks and task caching to avoid recomputing expensive operations.

    Prefect exposes this functionality via the suspend_flow_run and resume_flow_run functions, as well as the Prefect UI.

    When called inside of a flow suspend_flow_run will immediately suspend execution of the flow run. The flow run will be marked as Suspended and will not be resumed until resume_flow_run is called.

    Timeouts

    Suspended flow runs time out after one hour by default. After the timeout, the flow run will fail with a message saying it suspended and never resumed. You can specify a different timeout period in seconds using the timeout parameter or pass timeout=None for no timeout.

    Here is an example of a flow that does not block flow execution while paused. This flow will exit after one task, and will be rescheduled upon resuming. The stored result of the first task is retrieved instead of being rerun.

    from prefect import flow, pause_flow_run, task\n\n@task(persist_result=True)\ndef foo():\n    return 42\n\n@flow(persist_result=True)\ndef noblock_pausing():\n    x = foo.submit()\n    pause_flow_run(timeout=30, reschedule=True)\n    y = foo.submit()\n    z = foo(wait_for=[x])\n    alpha = foo(wait_for=[y])\n    omega = foo(wait_for=[x, y])\n

    Flow runs can be suspended out-of-process by calling suspend_flow_run(flow_run_id=<ID>) or selecting the Suspend button in the Prefect UI or Prefect Cloud.

    Suspended flow runs can be resumed by clicking the Resume button in the Prefect UI or calling the resume_flow_run utility via client code.

    resume_flow_run(FLOW_RUN_ID)\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#waiting-for-input-when-pausing-or-suspending-a-flow-run","title":"Waiting for input when pausing or suspending a flow run","text":"

    Experimental

    The wait_for_input parameter used in the pause_flow_run or suspend_flow_run functions is an experimental feature. The interface or behavior of this feature may change without warning in future releases.

    If you encounter any issues, please let us know in Slack or with a Github issue.

    When pausing or suspending a flow run you may want to wait for input from a user. Prefect provides a way to do this by leveraging the pause_flow_run and suspend_flow_run functions. These functions accept a wait_for_input argument, the value of which should be a subclass of prefect.input.RunInput, a pydantic model. When resuming the flow run, users are required to provide data for this model. Upon successful validation, the flow run resumes, and the return value of the pause_flow_run or suspend_flow_run is an instance of the model containing the provided data.

    Here is an example of a flow that pauses and waits for input from a user:

    from prefect import flow, get_run_logger, pause_flow_run\nfrom prefect.input import RunInput\n\n\nclass UserNameInput(RunInput):\n    name: str\n\n\n@flow\nasync def greet_user():\n    logger = get_run_logger()\n\n    user_input = await pause_flow_run(\n        wait_for_input=UserNameInput\n    )\n\n    logger.info(f\"Hello, {user_input.name}!\")\n
    Running this flow will create a flow run. The flow run will advance until code execution reaches pause_flow_run, at which point it will move into a Paused state. Execution will block and wait for resumption.

    When resuming the flow run, users will be prompted to provide a value for the name field of the UserNameInput model. Upon successful validation, the flow run will resume, and the return value of the pause_flow_run will be an instance of the UserNameInput model containing the provided data.

    For more in-depth information on receiving input from users when pausing and suspending flow runs, see the Creating human in the loop workflows guide.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#canceling-a-flow-run","title":"Canceling a flow run","text":"

    You may cancel a scheduled or in-progress flow run from the CLI, UI, REST API, or Python client.

    When cancellation is requested, the flow run is moved to a \"Cancelling\" state. If the deployment is a work pool-based deployemnt with a worker, then the worker monitors the state of flow runs and detects that cancellation has been requested. The worker then sends a signal to the flow run infrastructure, requesting termination of the run. If the run does not terminate after a grace period (default of 30 seconds), the infrastructure will be killed, ensuring the flow run exits.

    A deployment is required

    Flow run cancellation requires the flow run to be associated with a deployment. A monitoring process must be running to enforce the cancellation. Inline subflow runs, i.e. those created without run_deployment, cannot be cancelled without cancelling the parent flow run. If you may need to cancel a subflow run independent of its parent flow run, we recommend deploying it separately and starting it using the run_deployment function.

    Cancellation is robust to restarts of Prefect workers. To enable this, we attach metadata about the created infrastructure to the flow run. Internally, this is referred to as the infrastructure_pid or infrastructure identifier. Generally, this is composed of two parts:

    1. Scope: identifying where the infrastructure is running.
    2. ID: a unique identifier for the infrastructure within the scope.

    The scope is used to ensure that Prefect does not kill the wrong infrastructure. For example, workers running on multiple machines may have overlapping process IDs but should not have a matching scope.

    The identifiers for infrastructure types:

    • Processes: The machine hostname and the PID.
    • Docker Containers: The Docker API URL and container ID.
    • Kubernetes Jobs: The Kubernetes cluster name and the job name.

    While the cancellation process is robust, there are a few issues than can occur:

    • If the infrastructure block for the flow run has been removed or altered, cancellation may not work.
    • If the infrastructure block for the flow run does not have support for cancellation, cancellation will not work.
    • If the identifier scope does not match when attempting to cancel a flow run the worker will be unable to cancel the flow run. Another worker may attempt cancellation.
    • If the infrastructure associated with the run cannot be found or has already been killed, the worker will mark the flow run as cancelled.
    • If the infrastructre_pid is missing from the flow run will be marked as cancelled but cancellation cannot be enforced.
    • If the worker runs into an unexpected error during cancellation the flow run may or may not be cancelled depending on where the error occurred. The worker will try again to cancel the flow run. Another worker may attempt cancellation.

    Enhanced cancellation

    We are working on improving cases where cancellation can fail. You can try the improved cancellation experience by enabling the PREFECT_EXPERIMENTAL_ENABLE_ENHANCED_CANCELLATION setting on your worker or agents:

    prefect config set PREFECT_EXPERIMENTAL_ENABLE_ENHANCED_CANCELLATION=True\n

    If you encounter any issues, please let us know in SlackSlack or with a Github issue.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#cancel-via-the-cli","title":"Cancel via the CLI","text":"

    From the command line in your execution environment, you can cancel a flow run by using the prefect flow-run cancel CLI command, passing the ID of the flow run.

    prefect flow-run cancel 'a55a4804-9e3c-4042-8b59-b3b6b7618736'\n
    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/flows/#cancel-via-the-ui","title":"Cancel via the UI","text":"

    From the UI you can cancel a flow run by navigating to the flow run's detail page and clicking the Cancel button in the upper right corner.

    ","tags":["flows","subflows","workflows","scripts","parameters","states","final state"],"boost":2},{"location":"concepts/infrastructure/","title":"Infrastructure","text":"

    Workers are recommended

    Infrastructure blocks are part of the agent based deployment model. Work Pools and Workers simplify the specification of each flow's infrastructure and runtime environment. If you have existing agents, you can upgrade from agents to workers to significantly enhance the experience of deploying flows.

    Users may specify an infrastructure block when creating a deployment. This block will be used to specify infrastructure for flow runs created by the deployment at runtime.

    Infrastructure can only be used with a deployment. When you run a flow directly by calling the flow yourself, you are responsible for the environment in which the flow executes.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#infrastructure-overview","title":"Infrastructure overview","text":"

    Prefect uses infrastructure to create the environment for a user's flow to execute.

    Infrastructure is attached to a deployment and is propagated to flow runs created for that deployment. Infrastructure is deserialized by the agent and it has two jobs:

    • Create execution environment infrastructure for the flow run.
    • Run a Python command to start the prefect.engine in the infrastructure, which retrieves the flow from storage and executes the flow.

    The engine acquires and calls the flow. Infrastructure doesn't know anything about how the flow is stored, it's just passing a flow run ID to the engine.

    Infrastructure is specific to the environments in which flows will run. Prefect currently provides the following infrastructure types:

    • Process runs flows in a local subprocess.
    • DockerContainer runs flows in a Docker container.
    • KubernetesJob runs flows in a Kubernetes Job.
    • ECSTask runs flows in an Amazon ECS Task.
    • Cloud Run runs flows in a Google Cloud Run Job.
    • Container Instance runs flows in an Azure Container Instance.

    What about tasks?

    Flows and tasks can both use configuration objects to manage the environment in which code runs.

    Flows use infrastructure.

    Tasks use task runners. For more on how task runners work, see Task Runners.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#using-infrastructure","title":"Using infrastructure","text":"

    You may create customized infrastructure blocks through the Prefect UI or Prefect Cloud Blocks page or create them in code and save them to the API using the blocks .save() method.

    Once created, there are two distinct ways to use infrastructure in a deployment:

    • Starting with Prefect defaults \u2014 this is what happens when you pass the -i or --infra flag and provide a type when building deployment files.
    • Pre-configure infrastructure settings as blocks and base your deployment infrastructure on those settings \u2014 by passing -ib or --infra-block and a block slug when building deployment files.

    For example, when creating your deployment files, the supported Prefect infrastrucure types are:

    • process
    • docker-container
    • kubernetes-job
    • ecs-task
    • cloud-run-job
    • container-instance-job
    $ prefect deployment build ./my_flow.py:my_flow -n my-flow-deployment -t test -i docker-container -sb s3/my-bucket --override env.EXTRA_PIP_PACKAGES=s3fs\nFound flow 'my-flow'\nSuccessfully uploaded 2 files to s3://bucket-full-of-sunshine\nDeployment YAML created at '/Users/terry/test/flows/infra/deployment.yaml'.\n

    In this example we specify the DockerContainer infrastructure in addition to a preconfigured AWS S3 bucket storage block.

    The default deployment YAML filename may be edited as needed to add an infrastructure type or infrastructure settings.

    ###\n### A complete description of a Prefect Deployment for flow 'my-flow'\n###\nname: my-flow-deployment\ndescription: null\nversion: e29de5d01b06d61b4e321d40f34a480c\n# The work queue that will handle this deployment's runs\nwork_queue_name: default\nwork_pool_name: default-agent-pool\ntags:\n- test\nparameters: {}\nschedule: null\nis_schedule_active: true\ninfra_overrides:\n  env.EXTRA_PIP_PACKAGES: s3fs\ninfrastructure:\n  type: docker-container\n  env: {}\n  labels: {}\n  name: null\n  command:\n  - python\n  - -m\n  - prefect.engine\n  image: prefecthq/prefect:2-latest\n  image_pull_policy: null\n  networks: []\n  network_mode: null\n  auto_remove: false\n  volumes: []\n  stream_output: true\n  memswap_limit: null\n  mem_limit: null\n  privileged: false\n  block_type_slug: docker-container\n  _block_type_slug: docker-container\n\n###\n### DO NOT EDIT BELOW THIS LINE\n###\nflow_name: my-flow\nmanifest_path: my_flow-manifest.json\nstorage:\n  bucket_path: bucket-full-of-sunshine\n  aws_access_key_id: '**********'\n  aws_secret_access_key: '**********'\n  _is_anonymous: true\n  _block_document_name: anonymous-xxxxxxxx-f1ff-4265-b55c-6353a6d65333\n  _block_document_id: xxxxxxxx-06c2-4c3c-a505-4a8db0147011\n  block_type_slug: s3\n  _block_type_slug: s3\npath: ''\nentrypoint: my_flow.py:my-flow\nparameter_openapi_schema:\n  title: Parameters\n  type: object\n  properties: {}\n  required: null\n  definitions: null\ntimestamp: '2023-02-08T23:00:14.974642+00:00'\n

    Editing deployment YAML

    Note the big DO NOT EDIT comment in the deployment YAML: In practice, anything above this block can be freely edited before running prefect deployment apply to create the deployment on the API.

    Once the deployment exists, any flow runs that this deployment starts will use DockerContainer infrastructure.

    You can also create custom infrastructure blocks \u2014 either in the Prefect UI for in code via the API \u2014 and use the settings in the block to configure your infastructure. For example, here we specify settings for Kubernetes infrastructure in a block named k8sdev.

    from prefect.infrastructure import KubernetesJob, KubernetesImagePullPolicy\n\nk8s_job = KubernetesJob(\n    namespace=\"dev\",\n    image=\"prefecthq/prefect:2.0.0-python3.9\",\n    image_pull_policy=KubernetesImagePullPolicy.IF_NOT_PRESENT,\n)\nk8s_job.save(\"k8sdev\")\n

    Now we can apply the infrastrucure type and settings in the block by specifying the block slug kubernetes-job/k8sdev as the infrastructure type when building a deployment:

    prefect deployment build flows/k8s_example.py:k8s_flow --name k8sdev --tag k8s -sb s3/dev -ib kubernetes-job/k8sdev\n

    See Deployments for more information about deployment build options.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#configuring-infrastructure","title":"Configuring infrastructure","text":"

    Every infrastrcture type has type-specific options.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#process","title":"Process","text":"

    Process infrastructure runs a command in a new process.

    Current environment variables and Prefect settings will be included in the created process. Configured environment variables will override any current environment variables.

    Process supports the following settings:

    Attributes Description command A list of strings specifying the command to start the flow run. In most cases you should not override this. env Environment variables to set for the new process. labels Labels for the process. Labels are for metadata purposes only and cannot be attached to the process itself. name A name for the process. For display purposes only.","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#dockercontainer","title":"DockerContainer","text":"

    DockerContainer infrastructure executes flow runs in a container.

    Requirements for DockerContainer:

    • Docker Engine must be available.
    • You must configure remote Storage. Local storage is not supported for Docker.
    • The API must be available from within the flow run container. To facilitate connections to locally hosted APIs, localhost and 127.0.0.1 will be replaced with host.docker.internal.
    • The ephemeral Prefect API won't work with Docker and Kubernetes. You must have a Prefect server or Prefect Cloud API endpoint set in your agent's configuration.

    DockerContainer supports the following settings:

    Attributes Description auto_remove Bool indicating whether the container will be removed on completion. If False, the container will remain after exit for inspection. command A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this. env Environment variables to set for the container. image An optional string specifying the name of a Docker image to use. Defaults to the Prefect image. If the image is stored anywhere other than a public Docker Hub registry, use a corresponding registry block, e.g. DockerRegistry or ensure otherwise that your execution layer is authenticated to pull the image from the image registry. image_pull_policy Specifies if the image should be pulled. One of 'ALWAYS', 'NEVER', 'IF_NOT_PRESENT'. image_registry A DockerRegistry block containing credentials to use if image is stored in a private image registry. labels An optional dictionary of labels, mapping name to value. name An optional name for the container. networks An optional list of strings specifying Docker networks to connect the container to. network_mode Set the network mode for the created container. Defaults to 'host' if a local API url is detected, otherwise the Docker default of 'bridge' is used. If 'networks' is set, this cannot be set. stream_output Bool indicating whether to stream output from the subprocess to local standard output. volumes An optional list of volume mount strings in the format of \"local_path:container_path\".

    Prefect automatically sets a Docker image matching the Python and Prefect version you're using at deployment time. You can see all available images at Docker Hub.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#kubernetesjob","title":"KubernetesJob","text":"

    KubernetesJob infrastructure executes flow runs in a Kubernetes Job.

    Requirements for KubernetesJob:

    • kubectl must be available.
    • You must configure remote Storage. Local storage is not supported for Kubernetes.
    • The ephemeral Prefect API won't work with Docker and Kubernetes. You must have a Prefect server or Prefect Cloud API endpoint set in your agent's configuration.

    The Prefect CLI command prefect kubernetes manifest server automatically generates a Kubernetes manifest with default settings for Prefect deployments. By default, it simply prints out the YAML configuration for a manifest. You can pipe this output to a file of your choice and edit as necessary.

    KubernetesJob supports the following settings:

    Attributes Description cluster_config An optional Kubernetes cluster config to use for this job. command A list of strings specifying the command to run in the container to start the flow run. In most cases you should not override this. customizations A list of JSON 6902 patches to apply to the base Job manifest. Alternatively, a valid JSON string is allowed (handy for deployments CLI). env Environment variables to set for the container. finished_job_ttl The number of seconds to retain jobs after completion. If set, finished jobs will be cleaned up by Kubernetes after the given delay. If None (default), jobs will need to be manually removed. image String specifying the tag of a Docker image to use for the Job. image_pull_policy The Kubernetes image pull policy to use for job containers. job The base manifest for the Kubernetes Job. job_watch_timeout_seconds Number of seconds to watch for job creation before timing out (defaults to None). labels Dictionary of labels to add to the Job. name An optional name for the job. namespace String signifying the Kubernetes namespace to use. pod_watch_timeout_seconds Number of seconds to watch for pod creation before timing out (default 60). service_account_name An optional string specifying which Kubernetes service account to use. stream_output Bool indicating whether to stream output from the subprocess to local standard output.","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#kubernetesjob-overrides-and-customizations","title":"KubernetesJob overrides and customizations","text":"

    When creating deployments using KubernetesJob infrastructure, the infra_overrides parameter expects a dictionary. For a KubernetesJob, the customizations parameter expects a list.

    Containers expect a list of objects, even if there is only one. For any patches applying to the container, the path value should be a list, for example: /spec/templates/spec/containers/0/resources

    A Kubernetes-Job infrastructure block defined in Python:

    customizations = [\n {\n     \"op\": \"add\",\n     \"path\": \"/spec/template/spec/containers/0/resources\",\n     \"value\": {\n         \"requests\": {\n             \"cpu\": \"2000m\",\n             \"memory\": \"4gi\"\n         },\n         \"limits\": {\n             \"cpu\": \"4000m\",\n             \"memory\": \"8Gi\",\n             \"nvidia.com/gpu\": \"1\"\n      }\n  },\n }\n]\n\nk8s_job = KubernetesJob(\n        namespace=namespace,\n        image=image_name,\n        image_pull_policy=KubernetesImagePullPolicy.ALWAYS,\n        finished_job_ttl=300,\n        job_watch_timeout_seconds=600,\n        pod_watch_timeout_seconds=600,\n        service_account_name=\"prefect-server\",\n        customizations=customizations,\n    )\nk8s_job.save(\"devk8s\")\n

    A Deployment with infra-overrides defined in Python:

    infra_overrides={ \n    \"customizations\": [\n            {\n                \"op\": \"add\",\n                \"path\": \"/spec/template/spec/containers/0/resources\",\n                \"value\": {\n                    \"requests\": {\n                        \"cpu\": \"2000m\",\n                        \"memory\": \"4gi\"\n                    },\n                    \"limits\": {\n                        \"cpu\": \"4000m\",\n                        \"memory\": \"8Gi\",\n                        \"nvidia.com/gpu\": \"1\"\n                }\n            },\n        }\n    ]\n}\n\n# Load an already created K8s Block\nk8sjob = k8s_job.load(\"devk8s\")\n\ndeployment = Deployment.build_from_flow(\n    flow=my_flow,\n    name=\"s3-example\",\n    version=2,\n    work_queue_name=\"aws\",\n    infrastructure=k8sjob,\n    storage=storage,\n    infra_overrides=infra_overrides,\n)\n\ndeployment.apply()\n
    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#ecstask","title":"ECSTask","text":"

    ECSTask infrastructure runs your flow in an ECS Task.

    Requirements for ECSTask:

    • The ephemeral Prefect API won't work with ECS directly. You must have a Prefect server or Prefect Cloud API endpoint set in your agent's configuration.
    • The prefect-aws collection must be installed within the agent environment: pip install prefect-aws
    • The ECSTask and AwsCredentials blocks must be registered within the agent environment: prefect block register -m prefect_aws.ecs
    • You must configure remote Storage. Local storage is not supported for ECS tasks. The most commonly used type of storage with ECSTask is S3. If you leverage that type of block, make sure that s3fs is installed within your agent and flow run environment. The easiest way to satisfy all the installation-related points mentioned above is to include the following commands in your Dockerfile:
    FROM prefecthq/prefect:2-python3.9  # example base image \nRUN pip install s3fs prefect-aws\n

    Make sure to allocate enough CPU and memory to your agent, and consider adding retries

    When you start a Prefect agent on AWS ECS Fargate, allocate as much CPU and memory as needed for your workloads. Your agent needs enough resources to appropriately provision infrastructure for your flow runs and to monitor their execution. Otherwise, your flow runs may get stuck in a Pending state. Alternatively, set a work-queue concurrency limit to ensure that the agent will not try to process all runs at the same time.

    Some API calls to provision infrastructure may fail due to unexpected issues on the client side (for example, transient errors such as ConnectionError, HTTPClientError, or RequestTimeout), or due to server-side rate limiting from the AWS service. To mitigate those issues, we recommend adding environment variables such as AWS_MAX_ATTEMPTS (can be set to an integer value such as 10) and AWS_RETRY_MODE (can be set to a string value including standard or adaptive modes). Those environment variables must be added within the agent environment, e.g. on your ECS service running the agent, rather than on the ECSTask infrastructure block.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/infrastructure/#docker-images","title":"Docker images","text":"

    Learn about options for Prefect-maintained Docker images in the Docker guide.

    ","tags":["orchestration","infrastructure","flow run infrastructure","deployments","Kubernetes","Docker","ECS","Cloud Run","Container Instances"],"boost":0.5},{"location":"concepts/results/","title":"Results","text":"

    Results represent the data returned by a flow or a task.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#retrieving-results","title":"Retrieving results","text":"

    When calling flows or tasks, the result is returned directly:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return 1\n\n@flow\ndef my_flow():\n    task_result = my_task()\n    return task_result + 1\n\nresult = my_flow()\nassert result == 2\n

    When working with flow and task states, the result can be retrieved with the State.result() method:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return 1\n\n@flow\ndef my_flow():\n    state = my_task(return_state=True)\n    return state.result() + 1\n\nstate = my_flow(return_state=True)\nassert state.result() == 2\n

    When submitting tasks to a runner, the result can be retrieved with the Future.result() method:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return 1\n\n@flow\ndef my_flow():\n    future = my_task.submit()\n    return future.result() + 1\n\nresult = my_flow()\nassert result == 2\n
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#handling-failures","title":"Handling failures","text":"

    Sometimes your flows or tasks will encounter an exception. Prefect captures all exceptions in order to report states to the orchestrator, but we do not hide them from you (unless you ask us to) as your program needs to know if an unexpected error has occurred.

    When calling flows or tasks, the exceptions are raised as in normal Python:

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    try:\n        my_task()\n    except ValueError:\n        print(\"Oh no! The task failed.\")\n\n    return True\n\nmy_flow()\n

    If you would prefer to check for a failed task without using try/except, you may ask Prefect to return the state:

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    state = my_task(return_state=True)\n\n    if state.is_failed():\n        print(\"Oh no! The task failed. Falling back to '1'.\")\n        result = 1\n    else:\n        result = state.result()\n\n    return result + 1\n\nresult = my_flow()\nassert result == 2\n

    If you retrieve the result from a failed state, the exception will be raised. For this reason, it's often best to check if the state is failed first.

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    state = my_task(return_state=True)\n\n    try:\n        result = state.result()\n    except ValueError:\n        print(\"Oh no! The state raised the error!\")\n\n    return True\n\nmy_flow()\n

    When retrieving the result from a state, you can ask Prefect not to raise exceptions:

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    state = my_task(return_state=True)\n\n    maybe_result = state.result(raise_on_failure=False)\n    if isinstance(maybe_result, ValueError):\n        print(\"Oh no! The task failed. Falling back to '1'.\")\n        result = 1\n    else:\n        result = maybe_result\n\n    return result + 1\n\nresult = my_flow()\nassert result == 2\n

    When submitting tasks to a runner, Future.result() works the same as State.result():

    from prefect import flow, task\n\n@task\ndef my_task():\n    raise ValueError()\n\n@flow\ndef my_flow():\n    future = my_task.submit()\n\n    try:\n        future.result()\n    except ValueError:\n        print(\"Ah! Futures will raise the failure as well.\")\n\n    # You can ask it not to raise the exception too\n    maybe_result = future.result(raise_on_failure=False)\n    print(f\"Got {type(maybe_result)}\")\n\n    return True\n\nmy_flow()\n
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#working-with-async-results","title":"Working with async results","text":"

    When calling flows or tasks, the result is returned directly:

    import asyncio\nfrom prefect import flow, task\n\n@task\nasync def my_task():\n    return 1\n\n@flow\nasync def my_flow():\n    task_result = await my_task()\n    return task_result + 1\n\nresult = asyncio.run(my_flow())\nassert result == 2\n

    When working with flow and task states, the result can be retrieved with the State.result() method:

    import asyncio\nfrom prefect import flow, task\n\n@task\nasync def my_task():\n    return 1\n\n@flow\nasync def my_flow():\n    state = await my_task(return_state=True)\n    result = await state.result(fetch=True)\n    return result + 1\n\nasync def main():\n    state = await my_flow(return_state=True)\n    assert await state.result(fetch=True) == 2\n\nasyncio.run(main())\n

    Resolving results

    Prefect 2.6.0 added automatic retrieval of persisted results. Prior to this version, State.result() did not require an await. For backwards compatibility, when used from an asynchronous context, State.result() returns a raw result type.

    You may opt-in to the new behavior by passing fetch=True as shown in the example above. If you would like this behavior to be used automatically, you may enable the PREFECT_ASYNC_FETCH_STATE_RESULT setting. If you do not opt-in to this behavior, you will see a warning.

    You may also opt-out by setting fetch=False. This will silence the warning, but you will need to retrieve your result manually from the result type.

    When submitting tasks to a runner, the result can be retrieved with the Future.result() method:

    import asyncio\nfrom prefect import flow, task\n\n@task\nasync def my_task():\n    return 1\n\n@flow\nasync def my_flow():\n    future = await my_task.submit()\n    result = await future.result()\n    return result + 1\n\nresult = asyncio.run(my_flow())\nassert result == 2\n
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#persisting-results","title":"Persisting results","text":"

    The Prefect API does not store your results except in special cases. Instead, the result is persisted to a storage location in your infrastructure and Prefect stores a reference to the result.

    The following Prefect features require results to be persisted:

    • Task cache keys
    • Flow run retries

    If results are not persisted, these features may not be usable.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#configuring-persistence-of-results","title":"Configuring persistence of results","text":"

    Persistence of results requires a serializer and a storage location. Prefect sets defaults for these, and you should not need to adjust them until you want to customize behavior. You can configure results on the flow and task decorators with the following options:

    • persist_result: Whether the result should be persisted to storage.
    • result_storage: Where to store the result when persisted.
    • result_serializer: How to convert the result to a storable form.
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#toggling-persistence","title":"Toggling persistence","text":"

    Persistence of the result of a task or flow can be configured with the persist_result option. The persist_result option defaults to a null value, which will automatically enable persistence if it is needed for a Prefect feature used by the flow or task. Otherwise, persistence is disabled by default.

    For example, the following flow has retries enabled. Flow retries require that all task results are persisted, so the task's result will be persisted:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return \"hello world!\"\n\n@flow(retries=2)\ndef my_flow():\n    # This task does not have persistence toggled off and it is needed for the flow feature,\n    # so Prefect will persist its result at runtime\n    my_task()\n

    Flow retries do not require the flow's result to be persisted, so it will not be.

    In this next example, one task has caching enabled. Task caching requires that the given task's result is persisted:

    from prefect import flow, task\nfrom datetime import timedelta\n\n@task(cache_key_fn=lambda: \"always\", cache_expiration=timedelta(seconds=20))\ndef my_task():\n    # This task uses caching so its result will be persisted by default\n    return \"hello world!\"\n\n\n@task\ndef my_other_task():\n    ...\n\n@flow\ndef my_flow():\n    # This task uses a feature that requires result persistence\n    my_task()\n\n    # This task does not use a feature that requires result persistence and the\n    # flow does not use any features that require task result persistence so its\n    # result will not be persisted by default\n    my_other_task()\n

    Persistence of results can be manually toggled on or off:

    from prefect import flow, task\n\n@flow(persist_result=True)\ndef my_flow():\n    # This flow will persist its result even if not necessary for a feature.\n    ...\n\n@task(persist_result=False)\ndef my_task():\n    # This task will never persist its result.\n    # If persistence needed for a feature, an error will be raised.\n    ...\n

    Toggling persistence manually will always override any behavior that Prefect would infer.

    You may also change Prefect's default persistence behavior with the PREFECT_RESULTS_PERSIST_BY_DEFAULT setting. To persist results by default, even if they are not needed for a feature change the value to a truthy value:

    prefect config set PREFECT_RESULTS_PERSIST_BY_DEFAULT=true\n

    Task and flows with persist_result=False will not persist their results even if PREFECT_RESULTS_PERSIST_BY_DEFAULT is true.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-storage-location","title":"Result storage location","text":"

    The result storage location can be configured with the result_storage option. The result_storage option defaults to a null value, which infers storage from the context. Generally, this means that tasks will use the result storage configured on the flow unless otherwise specified. If there is no context to load the storage from and results must be persisted, results will be stored in the path specified by the PREFECT_LOCAL_STORAGE_PATH setting (defaults to ~/.prefect/storage).

    from prefect import flow, task\nfrom prefect.filesystems import LocalFileSystem, S3\n\n@flow(persist_result=True)\ndef my_flow():\n    my_task()  # This task will use the flow's result storage\n\n@task(persist_result=True)\ndef my_task():\n    ...\n\nmy_flow()  # The flow has no result storage configured and no parent, the local file system will be used.\n\n\n# Reconfigure the flow to use a different storage type\nnew_flow = my_flow.with_options(result_storage=S3(bucket_path=\"my-bucket\"))\n\nnew_flow()  # The flow and task within it will use S3 for result storage.\n

    You can configure this to use a specific storage using one of the following:

    • A storage instance, e.g. LocalFileSystem(basepath=\".my-results\")
    • A storage slug, e.g. 's3/dev-s3-block'
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-storage-key","title":"Result storage key","text":"

    The path of the result file in the result storage can be configured with the result_storage_key. The result_storage_key option defaults to a null value, which generates a unique identifier for each result.

    from prefect import flow, task\nfrom prefect.filesystems import LocalFileSystem, S3\n\n@flow(result_storage=S3(bucket_path=\"my-bucket\"))\ndef my_flow():\n    my_task()\n\n@task(persist_result=True, result_storage_key=\"my_task.json\")\ndef my_task():\n    ...\n\nmy_flow()  # The task's result will be persisted to 's3://my-bucket/my_task.json'\n

    Result storage keys are formatted with access to all of the modules in prefect.runtime and the run's parameters. In the following example, we will run a flow with three runs of the same task. Each task run will write its result to a unique file based on the name parameter.

    from prefect import flow, task\n\n@flow()\ndef my_flow():\n    hello_world()\n    hello_world(name=\"foo\")\n    hello_world(name=\"bar\")\n\n@task(persist_result=True, result_storage_key=\"hello-{parameters[name]}.json\")\ndef hello_world(name: str = \"world\"):\n    return f\"hello {name}\"\n\nmy_flow()\n

    After running the flow, we can see three persisted result files in our storage directory:

    $ ls ~/.prefect/storage | grep \"hello-\"\nhello-bar.json\nhello-foo.json\nhello-world.json\n

    In the next example, we include metadata about the flow run from the prefect.runtime.flow_run module:

    from prefect import flow, task\n\n@flow\ndef my_flow():\n    hello_world()\n\n@task(persist_result=True, result_storage_key=\"{flow_run.flow_name}_{flow_run.name}_hello.json\")\ndef hello_world(name: str = \"world\"):\n    return f\"hello {name}\"\n\nmy_flow()\n

    After running this flow, we can see a result file templated with the name of the flow and the flow run:

    \u276f ls ~/.prefect/storage | grep \"my-flow\"    \nmy-flow_industrious-trout_hello.json\n

    If a result exists at a given storage key in the storage location, it will be overwritten.

    Result storage keys can only be configured on tasks at this time.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-serializer","title":"Result serializer","text":"

    The result serializer can be configured with the result_serializer option. The result_serializer option defaults to a null value, which infers the serializer from the context. Generally, this means that tasks will use the result serializer configured on the flow unless otherwise specified. If there is no context to load the serializer from, the serializer defined by PREFECT_RESULTS_DEFAULT_SERIALIZER will be used. This setting defaults to Prefect's pickle serializer.

    You may configure the result serializer using:

    • A type name, e.g. \"json\" or \"pickle\" \u2014 this corresponds to an instance with default values
    • An instance, e.g. JSONSerializer(jsonlib=\"orjson\")
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#compressing-results","title":"Compressing results","text":"

    Prefect provides a CompressedSerializer which can be used to wrap other serializers to provide compression over the bytes they generate. The compressed serializer uses lzma compression by default. We test other compression schemes provided in the Python standard library such as bz2 and zlib, but you should be able to use any compression library that provides compress and decompress methods.

    You may configure compression of results using:

    • A type name, prefixed with compressed/ e.g. \"compressed/json\" or \"compressed/pickle\"
    • An instance e.g. CompressedSerializer(serializer=\"pickle\", compressionlib=\"lzma\")

    Note that the \"compressed/<serializer-type>\" shortcut will only work for serializers provided by Prefect. If you are using custom serializers, you must pass a full instance.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#storage-of-results-in-prefect","title":"Storage of results in Prefect","text":"

    The Prefect API does not store your results in most cases for the following reasons:

    • Results can be large and slow to send to and from the API.
    • Results often contain private information or data.
    • Results would need to be stored in the database or complex logic implemented to hydrate from another source.

    There are a few cases where Prefect will store your results directly in the database. This is an optimization to reduce the overhead of reading and writing to result storage.

    The following data types will be stored by the API without persistence to storage:

    • booleans (True, False)
    • nulls (None)

    If persist_result is set to False, these values will never be stored.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#tracking-results","title":"Tracking results","text":"

    The Prefect API tracks metadata about your results. The value of your result is only stored in specific cases. Result metadata can be seen in the UI on the \"Results\" page for flows.

    Prefect tracks the following result metadata:

    • Data type
    • Storage location (if persisted)
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#caching-of-results-in-memory","title":"Caching of results in memory","text":"

    When running your workflows, Prefect will keep the results of all tasks and flows in memory so they can be passed downstream. In some cases, it is desirable to override this behavior. For example, if you are returning a large amount of data from a task it can be costly to keep it memory for the entire duration of the flow run.

    Flows and tasks both include an option to drop the result from memory with cache_result_in_memory:

    @flow(cache_result_in_memory=False)\ndef foo():\n    return \"pretend this is large data\"\n\n@task(cache_result_in_memory=False)\ndef bar():\n    return \"pretend this is biiiig data\"\n

    When cache_result_in_memory is disabled, the result of your flow or task will be persisted by default. The result will then be pulled from storage when needed.

    @flow\ndef foo():\n    result = bar()\n    state = bar(return_state=True)\n\n    # The result will be retrieved from storage here\n    state.result()\n\n    future = bar.submit()\n    # The result will be retrieved from storage here\n    future.result()\n\n@task(cache_result_in_memory=False)\ndef bar():\n    # This result will persisted\n    return \"pretend this is biiiig data\"\n

    If both cache_result_in_memory and persistence are disabled, your results will not be available downstream.

    @task(persist_result=False, cache_result_in_memory=False)\ndef bar():\n    return \"pretend this is biiiig data\"\n\n@flow\ndef foo():\n    # Raises an error\n    result = bar()\n\n    # This is oaky\n    state = bar(return_state=True)\n\n    # Raises an error\n    state.result()\n\n    # This is okay\n    future = bar.submit()\n\n    # Raises an error\n    future.result()\n
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-storage-types","title":"Result storage types","text":"

    Result storage is responsible for reading and writing serialized data to an external location. At this time, any file system block can be used for result storage.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-serializer-types","title":"Result serializer types","text":"

    A result serializer is responsible for converting your Python object to and from bytes. This is necessary to store the object outside of Python and retrieve it later.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#pickle-serializer","title":"Pickle serializer","text":"

    Pickle is a standard Python protocol for encoding arbitrary Python objects. We supply a custom pickle serializer at prefect.serializers.PickleSerializer. Prefect's pickle serializer uses the cloudpickle project by default to support more object types. Alternative pickle libraries can be specified:

    from prefect.serializers import PickleSerializer\n\nPickleSerializer(picklelib=\"custompickle\")\n

    Benefits of the pickle serializer:

    • Many object types are supported.
    • Objects can define custom pickle support.

    Drawbacks of the pickle serializer:

    • When nested attributes of an object cannot be pickled, it is hard to determine the cause.
    • When deserializing objects, your Python and pickle library versions must match the one used at serialization time.
    • Serialized objects cannot be easily shared across different programming languages.
    • Serialized objects are not human readable.
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#json-serializer","title":"JSON serializer","text":"

    We supply a custom JSON serializer at prefect.serializers.JSONSerializer. Prefect's JSON serializer uses custom hooks by default to support more object types. Specifically, we add support for all types supported by Pydantic.

    By default, we use the standard Python json library. Alternative JSON libraries can be specified:

    from prefect.serializers import JSONSerializer\n\nJSONSerializer(jsonlib=\"orjson\")\n

    Benefits of the JSON serializer:

    • Serialized objects are human readable.
    • Serialized objects can often be shared across different programming languages.
    • Deserialization of serialized objects is generally version agnostic.

    Drawbacks of the JSON serializer:

    • Supported types are limited.
    • Implementing support for additional types must be done at the serializer level.
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#result-types","title":"Result types","text":"

    Prefect uses internal result types to capture information about the result attached to a state. The following types are used:

    • UnpersistedResult: Stores result metadata but the value is only available when created.
    • LiteralResult: Stores simple values inline.
    • PersistedResult: Stores a reference to a result persisted to storage.

    All result types include a get() method that can be called to return the value of the result. This is done behind the scenes when the result() method is used on states or futures.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#unpersisted-results","title":"Unpersisted results","text":"

    Unpersisted results are used to represent results that have not been and will not be persisted beyond the current flow run. The value associated with the result is stored in memory, but will not be available later. Result metadata is attached to this object for storage in the API and representation in the UI.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#literal-results","title":"Literal results","text":"

    Literal results are used to represent results stored in the Prefect database. The values contained by these results must always be JSON serializable.

    Example:

    result = LiteralResult(value=None)\nresult.json()\n# {\"type\": \"result\", \"value\": \"null\"}\n

    Literal results reduce the overhead required to persist simple results.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#persisted-results","title":"Persisted results","text":"

    The persisted result type contains all of the information needed to retrieve the result from storage. This includes:

    • Storage: A reference to the result storage that can be used to read the serialized result.
    • Key: Indicates where this specific result is in storage.

    Persisted result types also contain metadata for inspection without retrieving the result:

    • Serializer type: The name of the result serializer type.

    The get() method on result references retrieves the data from storage, deserializes it, and returns the original object. The get() operation will cache the resolved object to reduce the overhead of subsequent calls.

    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/results/#persisted-result-blob","title":"Persisted result blob","text":"

    When results are persisted to storage, they are always written as a JSON document. The schema for this is described by the PersistedResultBlob type. The document contains:

    • The serialized data of the result.
    • A full description of result serializer that can be used to deserialize the result data.
    • The Prefect version used to create the result.
    ","tags":["flows","subflows","tasks","states","results"],"boost":2},{"location":"concepts/schedules/","title":"Schedules","text":"

    Scheduling is one of the primary reasons for using an orchestrator such as Prefect. Prefect allows you to use schedules to automatically create new flow runs for deployments.

    Prefect Cloud can also schedule flow runs through event-driven automations.

    Schedules tell the Prefect API how to create new flow runs for you automatically on a specified cadence.

    You can add a schedule to any flow deployment. The Prefect Scheduler service periodically reviews every deployment and creates new flow runs according to the schedule configured for the deployment.

    There are several recommended ways to create a schedule for a deployment:

    • Through the Prefect UI
    • Via a the cron, interval, or rrule parameters if building your deployment via the serve method of the Flow object or the serve utility for managing multiple flows simultaneously
    • If using worker-based deployments
    • Through the interactive prefect deploy command
    • With the deployments -> schedule section of the prefect.yaml file )
    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-the-ui","title":"Creating schedules through the UI","text":"

    You can add a schedule on the Deployments tab of the UI.

    Under Schedule select the Add button. If you don't see the Schedule section, expand the three dot menu on the top right of the page.

    Then select Interval or Cron to create a schedule.

    Once a schedule has been created, a number of scheduled flow runs will be visible in the UI. The schedule is viewable in human-friendly text on the Deployments page. You can edit the schedule by selecting the Edit button on the Deployment page.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#schedule-types","title":"Schedule types","text":"

    Prefect supports several types of schedules that cover a wide range of use cases and offer a large degree of customization:

    • Cron is most appropriate for users who are already familiar with cron from previous use.
    • Interval is best suited for deployments that need to run at some consistent cadence that isn't related to absolute time.
    • RRule is best suited for deployments that rely on calendar logic for simple recurring schedules, irregular intervals, exclusions, or day-of-month adjustments.
    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-the-serve-method","title":"Creating schedules through the serve method","text":"

    As seen in the Quickstart, you can create a schedule by passing a cron, interval, or rrule parameters to the Flow.serve method or the serve utility.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-the-interactive-prefect-deploy-command","title":"Creating schedules through the interactive prefect deploy command","text":"

    If you are using worker-based deployments, you can create a schedule through the interactive prefect deploy command. You will be prompted to choose which type of schedule to create.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-a-prefectyaml-files-deployments-schedule-section","title":"Creating schedules through a prefect.yaml file's deployments -> schedule section","text":"

    If you save the prefect.yaml file from the prefect deploy command, you will see it has a schedule section for your deployment. Alternatively, you can create a prefect.yaml file from a recipe or from scratch and add a schedule section to it.

    deployments:\n  ...\n  schedule:\n    cron: 0 0 * * *\n    timezone: America/Chicago\n

    Let's discuss the three schedule types in more detail.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#cron","title":"Cron","text":"

    A schedule may be specified with a cron pattern. Users may also provide a timezone to enforce DST behaviors.

    Cron uses croniter to specify datetime iteration with a cron-like format.

    Cron properties include:

    Property Description cron A valid cron string. (Required) day_or Boolean indicating how croniter handles day and day_of_week entries. Default is True. timezone String name of a time zone. (See the IANA Time Zone Database for valid time zones.)

    The day_or property defaults to True, matching cron, which connects those values using OR. If False, the values are connected using AND. This behaves like fcron and enables you to, for example, define a job that executes each 2nd Friday of a month by setting the days of month and the weekday.

    Supported croniter features

    While Prefect supports most features of croniter for creating cron-like schedules, we do not currently support \"R\" random or \"H\" hashed keyword expressions or the schedule jittering possible with those expressions.

    Daylight saving time considerations

    If the timezone is a DST-observing one, then the schedule will adjust itself appropriately.

    The cron rules for DST are based on schedule times, not intervals. This means that an hourly cron schedule fires on every new schedule hour, not every elapsed hour. For example, when clocks are set back, this results in a two-hour pause as the schedule will fire the first time 1am is reached and the first time 2am is reached, 120 minutes later.

    Longer schedules, such as one that fires at 9am every morning, will adjust for DST automatically.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#interval","title":"Interval","text":"

    An Interval schedule creates new flow runs on a regular interval measured in seconds. Intervals are computed from an optional anchor_date. For example, here's how you can create a schedule for every 10 minutes in the deployment YAML file.

    schedule:\n  interval: 600\n  timezone: America/Chicago \n

    Interval properties include:

    Property Description interval datetime.timedelta indicating the time between flow runs. (Required) anchor_date datetime.datetime indicating the starting or \"anchor\" date to begin the schedule. If no anchor_date is supplied, the current UTC time is used. timezone String name of a time zone, used to enforce localization behaviors like DST boundaries. (See the IANA Time Zone Database for valid time zones.)

    Note that the anchor_date does not indicate a \"start time\" for the schedule, but rather a fixed point in time from which to compute intervals. If the anchor date is in the future, then schedule dates are computed by subtracting the interval from it. Note that in this example, we import the Pendulum Python package for easy datetime manipulation. Pendulum isn\u2019t required, but it\u2019s a useful tool for specifying dates.

    Daylight saving time considerations

    If the schedule's anchor_date or timezone are provided with a DST-observing timezone, then the schedule will adjust itself appropriately. Intervals greater than 24 hours will follow DST conventions, while intervals of less than 24 hours will follow UTC intervals.

    For example, an hourly schedule will fire every UTC hour, even across DST boundaries. When clocks are set back, this will result in two runs that appear to both be scheduled for 1am local time, even though they are an hour apart in UTC time.

    For longer intervals, like a daily schedule, the interval schedule will adjust for DST boundaries so that the clock-hour remains constant. This means that a daily schedule that always fires at 9am will observe DST and continue to fire at 9am in the local time zone.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#rrule","title":"RRule","text":"

    An RRule scheduling supports iCal recurrence rules (RRules), which provide convenient syntax for creating repetitive schedules. Schedules can repeat on a frequency from yearly down to every minute.

    RRule uses the dateutil rrule module to specify iCal recurrence rules.

    RRules are appropriate for any kind of calendar-date manipulation, including simple repetition, irregular intervals, exclusions, week day or day-of-month adjustments, and more. RRules can represent complex logic like:

    • The last weekday of each month
    • The fourth Thursday of November
    • Every other day of the week

    RRule properties include:

    Property Description rrule String representation of an RRule schedule. See the rrulestr examples for syntax. timezone String name of a time zone. See the IANA Time Zone Database for valid time zones.

    You may find it useful to use an RRule string generator such as the iCalendar.org RRule Tool to help create valid RRules.

    For example, the following RRule schedule creates flow runs on Monday, Wednesday, and Friday until July 30, 2024.

    schedule:\n  rrule: 'FREQ=WEEKLY;BYDAY=MO,WE,FR;UNTIL=20240730T040000Z'\n

    Max RRule length

    Note the max supported character length of an rrulestr is 6500 characters

    Daylight saving time considerations

    Note that as a calendar-oriented standard, RRules are sensitive to the initial timezone provided. A 9am daily schedule with a DST-aware start date will maintain a local 9am time through DST boundaries. A 9am daily schedule with a UTC start date will maintain a 9am UTC time.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#creating-schedules-through-a-python-deployment-creation-file","title":"Creating schedules through a Python deployment creation file","text":"

    When you create a deployment with through a Python file with flow.serve(), serve, flow.deploy(), or deploy you can specify the schedule. Just add the keyword argument cron, interval, or rrule.

    interval: An interval on which to execute the new deployment. Accepts either a number\n    or a timedelta object. If a number is given, it will be interpreted as seconds.\ncron: A cron schedule of when to execute runs of this deployment.\nrrule: An rrule schedule of when to execute runs of this deployment.\nschedule: A schedule object defining when to execute runs of this deployment. Used to\n  define additional scheduling options like `timezone`.\n

    Here's an example of creating a cron schedule with serve for a deployment flow that will run every minute of every day:

    my_flow.serve(name=\"flowing\", cron=\"* * * * *\")\n

    Here's an example of creating an interval schedule with serve for a deployment flow that will run every 10 minutes with an anchor date and a timezone:

    from datetime import timedelta, datetime\nfrom prefect.client.schemas.schedules import IntervalSchedule\n\nmy_flow.serve(name=\"flowing\", schedule=IntervalSchedule(interval=timedelta(minutes=10), anchor_date=datetime(2023, 1, 1, 0, 0), timezone=\"America/Chicago\"))\n

    Block and agent-based deployments with Python files are not a recommended way to create deployments. However, if you are using that deployment creation method you can create a schedule by passing a schedule parameter to the Deployment.build_from_flow method.

    Here's how you create the equivalent schedule in a Python deployment file, with a timezone specified.

    from prefect.server.schemas.schedules import CronSchedule\n\ncron_demo = Deployment.build_from_flow(\n    pipeline,\n    \"etl\",\n    schedule=(CronSchedule(cron=\"0 0 * * *\", timezone=\"America/Chicago\"))\n)\n

    IntervalSchedule and RRuleSchedule are the other two Python class schedule options.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/schedules/#the-scheduler-service","title":"The Scheduler service","text":"

    The Scheduler service is started automatically when prefect server start is run and it is a built-in service of Prefect Cloud.

    By default, the Scheduler service visits deployments on a 60-second loop, though recently-modified deployments will be visited more frequently. The Scheduler evaluates each deployment's schedule and creates new runs appropriately. For typical deployments, it will create the next three runs, though more runs will be scheduled if the next 3 would all start in the next hour.

    More specifically, the Scheduler tries to create the smallest number of runs that satisfy the following constraints, in order:

    • No more than 100 runs will be scheduled.
    • Runs will not be scheduled more than 100 days in the future.
    • At least 3 runs will be scheduled.
    • Runs will be scheduled until at least one hour in the future.

    These behaviors can all be adjusted through the relevant settings that can be viewed with the terminal command prefect config view --show-defaults:

    PREFECT_API_SERVICES_SCHEDULER_DEPLOYMENT_BATCH_SIZE='100'\nPREFECT_API_SERVICES_SCHEDULER_ENABLED='True'\nPREFECT_API_SERVICES_SCHEDULER_INSERT_BATCH_SIZE='500'\nPREFECT_API_SERVICES_SCHEDULER_LOOP_SECONDS='60.0'\nPREFECT_API_SERVICES_SCHEDULER_MIN_RUNS='3'\nPREFECT_API_SERVICES_SCHEDULER_MAX_RUNS='100'\nPREFECT_API_SERVICES_SCHEDULER_MIN_SCHEDULED_TIME='1:00:00'\nPREFECT_API_SERVICES_SCHEDULER_MAX_SCHEDULED_TIME='100 days, 0:00:00'\n

    See the Settings docs for more information on altering your settings.

    These settings mean that if a deployment has an hourly schedule, the default settings will create runs for the next 4 days (or 100 hours). If it has a weekly schedule, the default settings will maintain the next 14 runs (up to 100 days in the future).

    The Scheduler does not affect execution

    The Prefect Scheduler service only creates new flow runs and places them in Scheduled states. It is not involved in flow or task execution.

    If you change a schedule, previously scheduled flow runs that have not started are removed, and new scheduled flow runs are created to reflect the new schedule.

    To remove all scheduled runs for a flow deployment, you can remove the schedule via the UI.

    ","tags":["flows","flow runs","deployments","schedules","scheduling","cron","RRule","iCal"],"boost":2},{"location":"concepts/states/","title":"States","text":"","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#overview","title":"Overview","text":"

    States are rich objects that contain information about the status of a particular task run or flow run. While you don't need to know the details of the states to use Prefect, you can give your workflows superpowers by taking advantage of it.

    At any moment, you can learn anything you need to know about a task or flow by examining its current state or the history of its states. For example, a state could tell you that a task:

    • is scheduled to make a third run attempt in an hour

    • succeeded and what data it produced

    • was scheduled to run, but later cancelled

    • used the cached result of a previous run instead of re-running

    • failed because it timed out

    By manipulating a relatively small number of task states, Prefect flows can harness the complexity that emerges in workflows.

    Only runs have states

    Though we often refer to the \"state\" of a flow or a task, what we really mean is the state of a flow run or a task run. Flows and tasks are templates that describe what a system does; only when we run the system does it also take on a state. So while we might refer to a task as \"running\" or being \"successful\", we really mean that a specific instance of the task is in that state.

    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#state-types","title":"State Types","text":"

    States have names and types. State types are canonical, with specific orchestration rules that apply to transitions into and out of each state type. A state's name, is often, but not always, synonymous with its type. For example, a task run that is running for the first time has a state with the name Running and the type RUNNING. However, if the task retries, that same task run will have the name Retrying and the type RUNNING. Each time the task run transitions into the RUNNING state, the same orchestration rules are applied.

    There are terminal state types from which there are no orchestrated transitions to any other state type.

    • COMPLETED
    • CANCELLED
    • FAILED
    • CRASHED

    The full complement of states and state types includes:

    Name Type Terminal? Description Scheduled SCHEDULED No The run will begin at a particular time in the future. Late SCHEDULED No The run's scheduled start time has passed, but it has not transitioned to PENDING (5 seconds by default). AwaitingRetry SCHEDULED No The run did not complete successfully because of a code issue and had remaining retry attempts. Pending PENDING No The run has been submitted to run, but is waiting on necessary preconditions to be satisfied. Running RUNNING No The run code is currently executing. Retrying RUNNING No The run code is currently executing after previously not complete successfully. Paused PAUSED No The run code has stopped executing until it receives manual approval to proceed. Cancelling CANCELLING No The infrastructure on which the code was running is being cleaned up. Cancelled CANCELLED Yes The run did not complete because a user determined that it should not. Completed COMPLETED Yes The run completed successfully. Failed FAILED Yes The run did not complete because of a code issue and had no remaining retry attempts. Crashed CRASHED Yes The run did not complete because of an infrastructure issue.","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#returned-values","title":"Returned values","text":"

    When calling a task or a flow, there are three types of returned values:

    • Data: A Python object (such as int, str, dict, list, and so on).
    • State: A Prefect object indicating the state of a flow or task run.
    • PrefectFuture: A Prefect object that contains both data and State.

    Returning data\u200a is the default behavior any time you call your_task().

    Returning Prefect State occurs anytime you call your task or flow with the argument return_state=True.

    Returning PrefectFuture is achieved by calling your_task.submit().

    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#return-data","title":"Return Data","text":"

    By default, running a task will return data:

    from prefect import flow, task \n\n@task \ndef add_one(x):\n    return x + 1\n\n@flow \ndef my_flow():\n    result = add_one(1) # return int\n

    The same rule applies for a subflow:

    @flow \ndef subflow():\n    return 42 \n\n@flow \ndef my_flow():\n    result = subflow() # return data\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#return-prefect-state","title":"Return Prefect State","text":"

    To return a State instead, add return_state=True as a parameter of your task call.

    @flow \ndef my_flow():\n    state = add_one(1, return_state=True) # return State\n

    To get data from a State, call .result().

    @flow \ndef my_flow():\n    state = add_one(1, return_state=True) # return State\n    result = state.result() # return int\n

    The same rule applies for a subflow:

    @flow \ndef subflow():\n    return 42 \n\n@flow \ndef my_flow():\n    state = subflow(return_state=True) # return State\n    result = state.result() # return int\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#return-a-prefectfuture","title":"Return a PrefectFuture","text":"

    To get a PrefectFuture, add .submit() to your task call.

    @flow \ndef my_flow():\n    future = add_one.submit(1) # return PrefectFuture\n

    To get data from a PrefectFuture, call .result().

    @flow \ndef my_flow():\n    future = add_one.submit(1) # return PrefectFuture\n    result = future.result() # return data\n

    To get a State from a PrefectFuture, call .wait().

    @flow \ndef my_flow():\n    future = add_one.submit(1) # return PrefectFuture\n    state = future.wait() # return State\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#final-state-determination","title":"Final state determination","text":"

    The final state of a flow is determined by its return value. The following rules apply:

    • If an exception is raised directly in the flow function, the flow run is marked as FAILED.
    • If the flow does not return a value (or returns None), its state is determined by the states of all of the tasks and subflows within it.
    • If any task run or subflow run failed and none were cancelled, then the final flow run state is marked as FAILED.
    • If any task run or subflow run was cancelled, then the final flow run state is marked as CANCELLED.
    • If a flow returns a manually created state, it is used as the state of the final flow run. This allows for manual determination of final state.
    • If the flow run returns any other object, then it is marked as successfully completed.

    See the Final state determination section of the Flows documentation for further details and examples.

    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#state-change-hooks","title":"State Change Hooks","text":"

    State change hooks execute code in response to changes in flow or task run states, enabling you to define actions for specific state transitions in a workflow.

    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#a-simple-example","title":"A simple example","text":"
    from prefect import flow\n\ndef my_success_hook(flow, flow_run, state):\n    print(\"Flow run succeeded!\")\n\n@flow(on_completion=[my_success_hook])\ndef my_flow():\n    return 42\n\nmy_flow()\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#create-and-use-hooks","title":"Create and use hooks","text":"","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#available-state-change-hooks","title":"Available state change hooks","text":"Type Flow Task Description on_completion \u2713 \u2713 Executes when a flow or task run enters a Completed state. on_failure \u2713 \u2713 Executes when a flow or task run enters a Failed state. on_cancellation \u2713 - Executes when a flow run enters a Cancelling state. on_crashed \u2713 - Executes when a flow run enters a Crashed state.","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#create-flow-run-state-change-hooks","title":"Create flow run state change hooks","text":"
    def my_flow_hook(flow: Flow, flow_run: FlowRun, state: State):\n    \"\"\"This is the required signature for a flow run state\n    change hook. This hook can only be passed into flows.\n    \"\"\"\n\n# pass hook as a list of callables\n@flow(on_completion=[my_flow_hook])\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#create-task-run-state-change-hooks","title":"Create task run state change hooks","text":"
    def my_task_hook(task: Task, task_run: TaskRun, state: State):\n    \"\"\"This is the required signature for a task run state change\n    hook. This hook can only be passed into tasks.\n    \"\"\"\n\n# pass hook as a list of callables\n@task(on_failure=[my_task_hook])\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#use-multiple-state-change-hooks","title":"Use multiple state change hooks","text":"

    State change hooks are versatile, allowing you to specify multiple state change hooks for the same state transition, or to use the same state change hook for different transitions:

    def my_success_hook(task, task_run, state):\n    print(\"Task run succeeded!\")\n\ndef my_failure_hook(task, task_run, state):\n    print(\"Task run failed!\")\n\ndef my_succeed_or_fail_hook(task, task_run, state):\n    print(\"If the task run succeeds or fails, this hook runs.\")\n\n@task(\n    on_completion=[my_success_hook, my_succeed_or_fail_hook],\n    on_failure=[my_failure_hook, my_succeed_or_fail_hook]\n)\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#pass-kwargs-to-your-hooks","title":"Pass kwargs to your hooks","text":"

    The Prefect engine will call your hooks for you upon the state change, passing in the flow, flow run, and state objects.

    However, you can define your hook to have additional default arguments:

    from prefect import flow\n\ndata = {}\n\ndef my_hook(flow, flow_run, state, my_arg=\"custom_value\"):\n    data.update(my_arg=my_arg, state=state)\n\n@flow(on_completion=[my_hook])\ndef lazy_flow():\n    pass\n\nstate = lazy_flow(return_state=True)\n\nassert data == {\"my_arg\": \"custom_value\", \"state\": state}\n

    ... or define your hook to accept arbitrary keyword arguments:

    from functools import partial\nfrom prefect import flow, task\n\ndata = {}\n\ndef my_hook(task, task_run, state, **kwargs):\n    data.update(state=state, **kwargs)\n\n@task\ndef bad_task():\n    raise ValueError(\"meh\")\n\n@flow\ndef ok_with_failure_flow(x: str = \"foo\", y: int = 42):\n    bad_task_with_a_hook = bad_task.with_options(\n        on_failure=[partial(my_hook, **dict(x=x, y=y))]\n    )\n    # return a tuple of \"bar\" and the task run state\n    # to avoid raising the task's exception\n    return \"bar\", bad_task_with_a_hook(return_state=True)\n\n_, task_run_state = ok_with_failure_flow()\n\nassert data == {\"x\": \"foo\", \"y\": 42, \"state\": task_run_state}\n
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/states/#more-examples-of-state-change-hooks","title":"More examples of state change hooks","text":"
    • Send a notification when a flow run fails
    • Delete a Cloud Run job when a flow crashes
    ","tags":["orchestration","flow runs","task runs","states","status","state change hooks","triggers"],"boost":2},{"location":"concepts/storage/","title":"Storage","text":"

    Storage blocks are not recommended

    Storage blocks are part of the legacy block-based deployment model. Instead, using serve or runner-based Python creation methods or workers and work pools with prefect deploy via the CLI are the recommended options for creating a deployment. Flow code storage can be specified in the Python file with serve or runner-based Python creation methods; alternatively, with the work pools and workers style of flow deployment, you can specify flow code storage during the interactive prefect deploy CLI experience and in its resulting prefect.yaml file.

    Storage lets you configure how flow code for deployments is persisted and retrieved by Prefect workers (or legacy agents). Anytime you build a block-based deployment, a storage block is used to upload the entire directory containing your workflow code (along with supporting files) to its configured location. This helps ensure portability of your relative imports, configuration files, and more. Note that your environment dependencies (for example, external Python packages) still need to be managed separately.

    If no storage is explicitly configured, Prefect will use LocalFileSystem storage by default. Local storage works fine for many local flow run scenarios, especially when testing and getting started. However, due to the inherent lack of portability, many use cases are better served by using remote storage such as S3 or Google Cloud Storage.

    Prefect supports creating multiple storage configurations and switching between storage as needed.

    Storage uses blocks

    Blocks are the Prefect technology underlying storage, and enables you to do so much more.

    In addition to creating storage blocks via the Prefect CLI, you can now create storage blocks and other kinds of block configuration objects via the Prefect UI and Prefect Cloud.

    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/storage/#configuring-storage-for-a-deployment","title":"Configuring storage for a deployment","text":"

    When building a deployment for a workflow, you have two options for configuring workflow storage:

    • Use the default local storage
    • Preconfigure a storage block to use
    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/storage/#using-the-default","title":"Using the default","text":"

    Anytime you call prefect deployment build without providing the --storage-block flag, a default LocalFileSystem block will be used. Note that this block will always use your present working directory as its basepath (which is usually desirable). You can see the block's settings by inspecting the deployment.yaml file that Prefect creates after calling prefect deployment build.

    While you generally can't run a deployment stored on a local file system on other machines, any agent running on the same machine will be able to successfully run your deployment.

    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/storage/#supported-storage-blocks","title":"Supported storage blocks","text":"

    Current options for deployment storage blocks include:

    Storage Description Required Library Local File System Store code in a run's local file system. Remote File System Store code in a any filesystem supported by fsspec. AWS S3 Storage Store code in an AWS S3 bucket. s3fs Azure Storage Store code in Azure Datalake and Azure Blob Storage. adlfs GitHub Storage Store code in a GitHub repository. Google Cloud Storage Store code in a Google Cloud Platform (GCP) Cloud Storage bucket. gcsfs SMB Store code in SMB shared network storage. smbprotocol GitLab Repository Store code in a GitLab repository. prefect-gitlab Bitbucket Repository Store code in a Bitbucket repository. prefect-bitbucket

    Accessing files may require storage filesystem libraries

    Note that the appropriate filesystem library supporting the storage location must be installed prior to building a deployment with a storage block or accessing the storage location from flow scripts.

    For example, the AWS S3 Storage block requires the s3fs library.

    See Filesystem package dependencies for more information about configuring filesystem libraries in your execution environment.

    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/storage/#configuring-a-block","title":"Configuring a block","text":"

    You can create these blocks either via the UI or via Python.

    You can create, edit, and manage storage blocks in the Prefect UI and Prefect Cloud. On a Prefect server, blocks are created in the server's database. On Prefect Cloud, blocks are created on a workspace.

    To create a new block, select the + button. Prefect displays a library of block types you can configure to create blocks to be used by your flows.

    Select Add + to configure a new storage block based on a specific block type. Prefect displays a Create page that enables specifying storage settings.

    You can also create blocks using the Prefect Python API:

    from prefect.filesystems import S3\n\nblock = S3(bucket_path=\"my-bucket/a-sub-directory\", \n           aws_access_key_id=\"foo\", \n           aws_secret_access_key=\"bar\"\n)\nblock.save(\"example-block\")\n

    This block configuration is now available to be used by anyone with appropriate access to your Prefect API. We can use this block to build a deployment by passing its slug to the prefect deployment build command. The storage block slug is formatted as block-type/block-name. In this case, s3/example-block for an AWS S3 Bucket block named example-block. See block identifiers for details.

    prefect deployment build ./flows/my_flow.py:my_flow --name \"Example Deployment\" --storage-block s3/example-block\n

    This command will upload the contents of your flow's directory to the designated storage location, then the full deployment specification will be persisted to a newly created deployment.yaml file. For more information, see Deployments.

    ","tags":["storage","databases","database configuration","configuration","settings","AWS S3","Azure Blob Storage","Google Cloud Storage","SMB"],"boost":0.5},{"location":"concepts/task-runners/","title":"Task runners","text":"

    Task runners enable you to engage specific executors for Prefect tasks, such as for concurrent, parallel, or distributed execution of tasks.

    Task runners are not required for task execution. If you call a task function directly, the task executes as a regular Python function, without a task runner, and produces whatever result is returned by the function.

    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#task-runner-overview","title":"Task runner overview","text":"

    Calling a task function from within a flow, using the default task settings, executes the function sequentially. Execution of the task function blocks execution of the flow until the task completes. This means, by default, calling multiple tasks in a flow causes them to run in order.

    However, that's not the only way to run tasks!

    You can use the .submit() method on a task function to submit the task to a task runner. Using a task runner enables you to control whether tasks run sequentially, concurrently, or if you want to take advantage of a parallel or distributed execution library such as Dask or Ray.

    Using the .submit() method to submit a task also causes the task run to return a PrefectFuture, a Prefect object that contains both any data returned by the task function and a State, a Prefect object indicating the state of the task run.

    Prefect currently provides the following built-in task runners:

    • SequentialTaskRunner can run tasks sequentially.
    • ConcurrentTaskRunner can run tasks concurrently, allowing tasks to switch when blocking on IO. Tasks will be submitted to a thread pool maintained by anyio.

    In addition, the following Prefect-developed task runners for parallel or distributed task execution may be installed as Prefect Integrations.

    • DaskTaskRunner can run tasks requiring parallel execution using dask.distributed.
    • RayTaskRunner can run tasks requiring parallel execution using Ray.

    Concurrency versus parallelism

    The words \"concurrency\" and \"parallelism\" may sound the same, but they mean different things in computing.

    Concurrency refers to a system that can do more than one thing simultaneously, but not at the exact same time. It may be more accurate to think of concurrent execution as non-blocking: within the restrictions of resources available in the execution environment and data dependencies between tasks, execution of one task does not block execution of other tasks in a flow.

    Parallelism refers to a system that can do more than one thing at the exact same time. Again, within the restrictions of resources available, parallel execution can run tasks at the same time, such as for operations mapped across a dataset.

    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#using-a-task-runner","title":"Using a task runner","text":"

    You do not need to specify a task runner for a flow unless your tasks require a specific type of execution.

    To configure your flow to use a specific task runner, import a task runner and assign it as an argument for the flow when the flow is defined.

    Remember to call .submit() when using a task runner

    Make sure you use .submit() to run your task with a task runner. Calling the task directly, without .submit(), from within a flow will run the task sequentially instead of using a specified task runner.

    For example, you can use ConcurrentTaskRunner to allow tasks to switch when they would block.

    from prefect import flow, task\nfrom prefect.task_runners import ConcurrentTaskRunner\nimport time\n\n@task\ndef stop_at_floor(floor):\n    print(f\"elevator moving to floor {floor}\")\n    time.sleep(floor)\n    print(f\"elevator stops on floor {floor}\")\n\n@flow(task_runner=ConcurrentTaskRunner())\ndef elevator():\n    for floor in range(10, 0, -1):\n        stop_at_floor.submit(floor)\n

    If you specify an uninitialized task runner class, a task runner instance of that type is created with the default settings. You can also pass additional configuration parameters for task runners that accept parameters, such as DaskTaskRunner and RayTaskRunner.

    Default task runner

    If you don't specify a task runner for a flow and you call a task with .submit() within the flow, Prefect uses the default ConcurrentTaskRunner.

    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#running-tasks-sequentially","title":"Running tasks sequentially","text":"

    Sometimes, it's useful to force tasks to run sequentially to make it easier to reason about the behavior of your program. Switching to the SequentialTaskRunner will force submitted tasks to run sequentially rather than concurrently.

    Synchronous and asynchronous tasks

    The SequentialTaskRunner works with both synchronous and asynchronous task functions. Asynchronous tasks are Python functions defined using async def rather than def.

    The following example demonstrates using the SequentialTaskRunner to ensure that tasks run sequentially. In the example, the flow glass_tower runs the task stop_at_floor for floors one through 38, in that order.

    from prefect import flow, task\nfrom prefect.task_runners import SequentialTaskRunner\nimport random\n\n@task\ndef stop_at_floor(floor):\n    situation = random.choice([\"on fire\",\"clear\"])\n    print(f\"elevator stops on {floor} which is {situation}\")\n\n@flow(task_runner=SequentialTaskRunner(),\n      name=\"towering-infernflow\",\n      )\ndef glass_tower():\n    for floor in range(1, 39):\n        stop_at_floor.submit(floor)\n\nglass_tower()\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#using-multiple-task-runners","title":"Using multiple task runners","text":"

    Each flow can only have a single task runner, but sometimes you may want a subset of your tasks to run using a specific task runner. In this case, you can create subflows for tasks that need to use a different task runner.

    For example, you can have a flow (in the example below called sequential_flow) that runs its tasks locally using the SequentialTaskRunner. If you have some tasks that can run more efficiently in parallel on a Dask cluster, you could create a subflow (such as dask_subflow) to run those tasks using the DaskTaskRunner.

    from prefect import flow, task\nfrom prefect.task_runners import SequentialTaskRunner\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@task\ndef hello_local():\n    print(\"Hello!\")\n\n@task\ndef hello_dask():\n    print(\"Hello from Dask!\")\n\n@flow(task_runner=SequentialTaskRunner())\ndef sequential_flow():\n    hello_local.submit()\n    dask_subflow()\n    hello_local.submit()\n\n@flow(task_runner=DaskTaskRunner())\ndef dask_subflow():\n    hello_dask.submit()\n\nif __name__ == \"__main__\":\n    sequential_flow()\n

    Guarding main

    Note that you should guard the main function by using if __name__ == \"__main__\" to avoid issues with parallel processing.

    This script outputs the following logs demonstrating the use of the Dask task runner:

    120:14:29.785 | INFO    | prefect.engine - Created flow run 'ivory-caiman' for flow 'sequential-flow'\n20:14:29.785 | INFO    | Flow run 'ivory-caiman' - Starting 'SequentialTaskRunner'; submitted tasks will be run sequentially...\n20:14:29.880 | INFO    | Flow run 'ivory-caiman' - Created task run 'hello_local-7633879f-0' for task 'hello_local'\n20:14:29.881 | INFO    | Flow run 'ivory-caiman' - Executing 'hello_local-7633879f-0' immediately...\nHello!\n20:14:29.904 | INFO    | Task run 'hello_local-7633879f-0' - Finished in state Completed()\n20:14:29.952 | INFO    | Flow run 'ivory-caiman' - Created subflow run 'nimble-sparrow' for flow 'dask-subflow'\n20:14:29.953 | INFO    | prefect.task_runner.dask - Creating a new Dask cluster with `distributed.deploy.local.LocalCluster`\n20:14:31.862 | INFO    | prefect.task_runner.dask - The Dask dashboard is available at http://127.0.0.1:8787/status\n20:14:31.901 | INFO    | Flow run 'nimble-sparrow' - Created task run 'hello_dask-2b96d711-0' for task 'hello_dask'\n20:14:32.370 | INFO    | Flow run 'nimble-sparrow' - Submitted task run 'hello_dask-2b96d711-0' for execution.\nHello from Dask!\n20:14:33.358 | INFO    | Flow run 'nimble-sparrow' - Finished in state Completed('All states completed.')\n20:14:33.368 | INFO    | Flow run 'ivory-caiman' - Created task run 'hello_local-7633879f-1' for task 'hello_local'\n20:14:33.368 | INFO    | Flow run 'ivory-caiman' - Executing 'hello_local-7633879f-1' immediately...\nHello!\n20:14:33.386 | INFO    | Task run 'hello_local-7633879f-1' - Finished in state Completed()\n20:14:33.399 | INFO    | Flow run 'ivory-caiman' - Finished in state Completed('All states completed.')\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#using-results-from-submitted-tasks","title":"Using results from submitted tasks","text":"

    When you use .submit() to submit a task to a task runner, the task runner creates a PrefectFuture for access to the state and result of the task.

    A PrefectFuture is an object that provides access to a computation happening in a task runner \u2014 even if that computation is happening on a remote system.

    In the following example, we save the return value of calling .submit() on the task say_hello to the variable future, and then we print the type of the variable:

    from prefect import flow, task\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@flow\ndef hello_world():\n    future = say_hello.submit(\"Marvin\")\n    print(f\"variable 'future' is type {type(future)}\")\n\nhello_world()\n

    When you run this code, you'll see that the variable future is a PrefectFuture:

    variable 'future' is type <class 'prefect.futures.PrefectFuture'>\n

    When you pass a future into a task, Prefect waits for the \"upstream\" task \u2014 the one that the future references \u2014 to reach a final state before starting the downstream task.

    This means that the downstream task won't receive the PrefectFuture you passed as an argument. Instead, the downstream task will receive the value that the upstream task returned.

    Take a look at how this works in the following example

    from prefect import flow, task\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@task\ndef print_result(result):\n    print(type(result))\n    print(result)\n\n@flow(name=\"hello-flow\")\ndef hello_world():\n    future = say_hello.submit(\"Marvin\")\n    print_result.submit(future)\n\nhello_world()\n
    <class 'str'>\nHello Marvin!\n

    Futures have a few useful methods. For example, you can get the return value of the task run with .result():

    from prefect import flow, task\n\n@task\ndef my_task():\n    return 42\n\n@flow\ndef my_flow():\n    future = my_task.submit()\n    result = future.result()\n    print(result)\n\nmy_flow()\n

    The .result() method will wait for the task to complete before returning the result to the caller. If the task run fails, .result() will raise the task run's exception. You may disable this behavior with the raise_on_failure option:

    from prefect import flow, task\n\n@task\ndef my_task():\n    return \"I'm a task!\"\n\n\n@flow\ndef my_flow():\n    future = my_task.submit()\n    result = future.result(raise_on_failure=False)\n    if future.get_state().is_failed():\n        # `result` is an exception! handle accordingly\n        ...\n    else:\n        # `result` is the expected return value of our task\n        ...\n

    You can retrieve the current state of the task run associated with the PrefectFuture using .get_state():

    @flow\ndef my_flow():\n    future = my_task.submit()\n    state = future.get_state()\n

    You can also wait for a task to complete by using the .wait() method:

    @flow\ndef my_flow():\n    future = my_task.submit()\n    final_state = future.wait()\n

    You can include a timeout in the wait call to perform logic if the task has not finished in a given amount of time:

    @flow\ndef my_flow():\n    future = my_task.submit()\n    final_state = future.wait(1)  # Wait one second max\n    if final_state:\n        # Take action if the task is done\n        result = final_state.result()\n    else:\n        ... # Task action if the task is still running\n

    You may also use the wait_for=[] parameter when calling a task, specifying upstream task dependencies. This enables you to control task execution order for tasks that do not share data dependencies.

    @task\ndef task_a():\n    pass\n\n@task\ndef task_b():\n    pass\n\n@task\ndef task_c():\n    pass\n\n@task\ndef task_d():\n    pass\n\n@flow\ndef my_flow():\n    a = task_a.submit()\n    b = task_b.submit()\n    # Wait for task_a and task_b to complete\n    c = task_c.submit(wait_for=[a, b])\n    # task_d will wait for task_c to complete\n    # Note: If waiting for one task it must still be in a list.\n    d = task_d(wait_for=[c])\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#when-to-use-result-in-flows","title":"When to use .result() in flows","text":"

    The simplest pattern for writing a flow is either only using tasks or only using pure Python functions. When you need to mix the two, use .result().

    Using only tasks:

    from prefect import flow, task\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@task\ndef say_nice_to_meet_you(hello_greeting):\n    return f\"{hello_greeting} Nice to meet you :)\"\n\n@flow\ndef hello_world():\n    hello = say_hello.submit(\"Marvin\")\n    nice_to_meet_you = say_nice_to_meet_you.submit(hello)\n\nhello_world()\n

    Using only Python functions:

    from prefect import flow, task\n\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\ndef say_nice_to_meet_you(hello_greeting):\n    return f\"{hello_greeting} Nice to meet you :)\"\n\n@flow\ndef hello_world():\n    # because this is just a Python function, calls will not be tracked\n    hello = say_hello(\"Marvin\") \n    nice_to_meet_you = say_nice_to_meet_you(hello)\n\nhello_world()\n

    Mixing tasks and Python functions:

    from prefect import flow, task\n\ndef say_hello_extra_nicely_to_marvin(hello): # not a task or flow!\n    if hello == \"Hello Marvin!\":\n        return \"HI MARVIN!\"\n    return hello\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@task\ndef say_nice_to_meet_you(hello_greeting):\n    return f\"{hello_greeting} Nice to meet you :)\"\n\n@flow\ndef hello_world():\n    # run a task and get the result\n    hello = say_hello.submit(\"Marvin\").result()\n\n    # not calling a task or flow\n    special_greeting = say_hello_extra_nicely_to_marvin(hello)\n\n    # pass our modified greeting back into a task\n    nice_to_meet_you = say_nice_to_meet_you.submit(special_greeting)\n\n    print(nice_to_meet_you.result())\n\nhello_world()\n

    Note that .result() also limits Prefect's ability to track task dependencies. In the \"mixed\" example above, Prefect will not be aware that say_hello is upstream of nice_to_meet_you.

    Calling .result() is blocking

    When calling .result(), be mindful your flow function will have to wait until the task run is completed before continuing.

    from prefect import flow, task\n\n@task\ndef say_hello(name):\n    return f\"Hello {name}!\"\n\n@task\ndef do_important_stuff():\n    print(\"Doing lots of important stuff!\")\n\n@flow\ndef hello_world():\n    # blocks until `say_hello` has finished\n    result = say_hello.submit(\"Marvin\").result() \n    do_important_stuff.submit()\n\nhello_world()\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#running-tasks-on-dask","title":"Running tasks on Dask","text":"

    The DaskTaskRunner is a parallel task runner that submits tasks to the dask.distributed scheduler. By default, a temporary Dask cluster is created for the duration of the flow run. If you already have a Dask cluster running, either local or cloud hosted, you can provide the connection URL via the address kwarg.

    1. Make sure the prefect-dask collection is installed: pip install prefect-dask.
    2. In your flow code, import DaskTaskRunner from prefect_dask.task_runners.
    3. Assign it as the task runner when the flow is defined using the task_runner=DaskTaskRunner argument.

    For example, this flow uses the DaskTaskRunner configured to access an existing Dask cluster at http://my-dask-cluster.

    from prefect import flow\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@flow(task_runner=DaskTaskRunner(address=\"http://my-dask-cluster\"))\ndef my_flow():\n    ...\n

    DaskTaskRunner accepts the following optional parameters:

    Parameter Description address Address of a currently running Dask scheduler. cluster_class The cluster class to use when creating a temporary Dask cluster. It can be either the full class name (for example, \"distributed.LocalCluster\"), or the class itself. cluster_kwargs Additional kwargs to pass to the cluster_class when creating a temporary Dask cluster. adapt_kwargs Additional kwargs to pass to cluster.adapt when creating a temporary Dask cluster. Note that adaptive scaling is only enabled if adapt_kwargs are provided. client_kwargs Additional kwargs to use when creating a dask.distributed.Client.

    Multiprocessing safety

    Note that, because the DaskTaskRunner uses multiprocessing, calls to flows in scripts must be guarded with if __name__ == \"__main__\": or you will encounter warnings and errors.

    If you don't provide the address of a Dask scheduler, Prefect creates a temporary local cluster automatically. The number of workers used is based on the number of cores on your machine. The default provides a mix of processes and threads that should work well for most workloads. If you want to specify this explicitly, you can pass values for n_workers or threads_per_worker to cluster_kwargs.

    # Use 4 worker processes, each with 2 threads\nDaskTaskRunner(\n    cluster_kwargs={\"n_workers\": 4, \"threads_per_worker\": 2}\n)\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#using-a-temporary-cluster","title":"Using a temporary cluster","text":"

    The DaskTaskRunner is capable of creating a temporary cluster using any of Dask's cluster-manager options. This can be useful when you want each flow run to have its own Dask cluster, allowing for per-flow adaptive scaling.

    To configure, you need to provide a cluster_class. This can be:

    • A string specifying the import path to the cluster class (for example, \"dask_cloudprovider.aws.FargateCluster\")
    • The cluster class itself
    • A function for creating a custom cluster.

    You can also configure cluster_kwargs, which takes a dictionary of keyword arguments to pass to cluster_class when starting the flow run.

    For example, to configure a flow to use a temporary dask_cloudprovider.aws.FargateCluster with 4 workers running with an image named my-prefect-image:

    DaskTaskRunner(\n    cluster_class=\"dask_cloudprovider.aws.FargateCluster\",\n    cluster_kwargs={\"n_workers\": 4, \"image\": \"my-prefect-image\"},\n)\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#connecting-to-an-existing-cluster","title":"Connecting to an existing cluster","text":"

    Multiple Prefect flow runs can all use the same existing Dask cluster. You might manage a single long-running Dask cluster (maybe using the Dask Helm Chart) and configure flows to connect to it during execution. This has a few downsides when compared to using a temporary cluster (as described above):

    • All workers in the cluster must have dependencies installed for all flows you intend to run.
    • Multiple flow runs may compete for resources. Dask tries to do a good job sharing resources between tasks, but you may still run into issues.

    That said, you may prefer managing a single long-running cluster.

    To configure a DaskTaskRunner to connect to an existing cluster, pass in the address of the scheduler to the address argument:

    # Connect to an existing cluster running at a specified address\nDaskTaskRunner(address=\"tcp://...\")\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#adaptive-scaling","title":"Adaptive scaling","text":"

    One nice feature of using a DaskTaskRunner is the ability to scale adaptively to the workload. Instead of specifying n_workers as a fixed number, this lets you specify a minimum and maximum number of workers to use, and the dask cluster will scale up and down as needed.

    To do this, you can pass adapt_kwargs to DaskTaskRunner. This takes the following fields:

    • maximum (int or None, optional): the maximum number of workers to scale to. Set to None for no maximum.
    • minimum (int or None, optional): the minimum number of workers to scale to. Set to None for no minimum.

    For example, here we configure a flow to run on a FargateCluster scaling up to at most 10 workers.

    DaskTaskRunner(\n    cluster_class=\"dask_cloudprovider.aws.FargateCluster\",\n    adapt_kwargs={\"maximum\": 10}\n)\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#dask-annotations","title":"Dask annotations","text":"

    Dask annotations can be used to further control the behavior of tasks.

    For example, we can set the priority of tasks in the Dask scheduler:

    import dask\nfrom prefect import flow, task\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@task\ndef show(x):\n    print(x)\n\n\n@flow(task_runner=DaskTaskRunner())\ndef my_flow():\n    with dask.annotate(priority=-10):\n        future = show.submit(1)  # low priority task\n\n    with dask.annotate(priority=10):\n        future = show.submit(2)  # high priority task\n

    Another common use case is resource annotations:

    import dask\nfrom prefect import flow, task\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@task\ndef show(x):\n    print(x)\n\n# Create a `LocalCluster` with some resource annotations\n# Annotations are abstract in dask and not inferred from your system.\n# Here, we claim that our system has 1 GPU and 1 process available per worker\n@flow(\n    task_runner=DaskTaskRunner(\n        cluster_kwargs={\"n_workers\": 1, \"resources\": {\"GPU\": 1, \"process\": 1}}\n    )\n)\n\ndef my_flow():\n    with dask.annotate(resources={'GPU': 1}):\n        future = show(0)  # this task requires 1 GPU resource on a worker\n\n    with dask.annotate(resources={'process': 1}):\n        # These tasks each require 1 process on a worker; because we've \n        # specified that our cluster has 1 process per worker and 1 worker,\n        # these tasks will run sequentially\n        future = show(1)\n        future = show(2)\n        future = show(3)\n\n\nif __name__ == \"__main__\":\n    my_flow()\n
    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/task-runners/#running-tasks-on-ray","title":"Running tasks on Ray","text":"

    The RayTaskRunner \u2014 installed separately as a Prefect Collection \u2014 is a parallel task runner that submits tasks to Ray. By default, a temporary Ray instance is created for the duration of the flow run. If you already have a Ray instance running, you can provide the connection URL via an address argument.

    Remote storage and Ray tasks

    We recommend configuring remote storage for task execution with the RayTaskRunner. This ensures tasks executing in Ray have access to task result storage, particularly when accessing a Ray instance outside of your execution environment.

    To configure your flow to use the RayTaskRunner:

    1. Make sure the prefect-ray collection is installed: pip install prefect-ray.
    2. In your flow code, import RayTaskRunner from prefect_ray.task_runners.
    3. Assign it as the task runner when the flow is defined using the task_runner=RayTaskRunner argument.

    For example, this flow uses the RayTaskRunner configured to access an existing Ray instance at ray://192.0.2.255:8786.

    from prefect import flow\nfrom prefect_ray.task_runners import RayTaskRunner\n\n@flow(task_runner=RayTaskRunner(address=\"ray://192.0.2.255:8786\"))\ndef my_flow():\n    ... \n

    RayTaskRunner accepts the following optional parameters:

    Parameter Description address Address of a currently running Ray instance, starting with the ray:// URI. init_kwargs Additional kwargs to use when calling ray.init.

    Note that Ray Client uses the ray:// URI to indicate the address of a Ray instance. If you don't provide the address of a Ray instance, Prefect creates a temporary instance automatically.

    Ray environment limitations

    While we're excited about adding support for parallel task execution via Ray to Prefect, there are some inherent limitations with Ray you should be aware of:

    Ray's support for Python 3.11 is experimental.

    Ray support for non-x86/64 architectures such as ARM/M1 processors with installation from pip alone and will be skipped during installation of Prefect. It is possible to manually install the blocking component with conda. See the Ray documentation for instructions.

    See the Ray installation documentation for further compatibility information.

    ","tags":["tasks","task runners","executors","PrefectFuture","submit","concurrent execution","sequential execution","parallel execution","Dask","Ray"],"boost":2},{"location":"concepts/tasks/","title":"Tasks","text":"

    A task is a function that represents a discrete unit of work in a Prefect workflow. Tasks are not required \u2014 you may define Prefect workflows that consist only of flows, using regular Python statements and functions. Tasks enable you to encapsulate elements of your workflow logic in observable units that can be reused across flows and subflows.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#tasks-overview","title":"Tasks overview","text":"

    Tasks are functions: they can take inputs, perform work, and return an output. A Prefect task can do almost anything a Python function can do.

    Tasks are special because they receive metadata about upstream dependencies and the state of those dependencies before they run, even if they don't receive any explicit data inputs from them. This gives you the opportunity to, for example, have a task wait on the completion of another task before executing.

    Tasks also take advantage of automatic Prefect logging to capture details about task runs such as runtime, tags, and final state.

    You can define your tasks within the same file as your flow definition, or you can define tasks within modules and import them for use in your flow definitions. All tasks must be called from within a flow. Tasks may not be called from other tasks.

    Calling a task from a flow

    Use the @task decorator to designate a function as a task. Calling the task from within a flow function creates a new task run:

    from prefect import flow, task\n\n@task\ndef my_task():\n    print(\"Hello, I'm a task\")\n\n@flow\ndef my_flow():\n    my_task()\n

    Tasks are uniquely identified by a task key, which is a hash composed of the task name, the fully-qualified name of the function, and any tags. If the task does not have a name specified, the name is derived from the task function.

    How big should a task be?

    Prefect encourages \"small tasks\" \u2014 each one should represent a single logical step of your workflow. This allows Prefect to better contain task failures.

    To be clear, there's nothing stopping you from putting all of your code in a single task \u2014 Prefect will happily run it! However, if any line of code fails, the entire task will fail and must be retried from the beginning. This can be avoided by splitting the code into multiple dependent tasks.

    Calling a task's function from another task

    Prefect does not allow triggering task runs from other tasks. If you want to call your task's function directly, you can use task.fn().

    from prefect import flow, task\n\n@task\ndef my_first_task(msg):\n    print(f\"Hello, {msg}\")\n\n@task\ndef my_second_task(msg):\n    my_first_task.fn(msg)\n\n@flow\ndef my_flow():\n    my_second_task(\"Trillian\")\n

    Note that in the example above you are only calling the task's function without actually generating a task run. Prefect won't track task execution in your Prefect backend if you call the task function this way. You also won't be able to use features such as retries with this function call.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#task-arguments","title":"Task arguments","text":"

    Tasks allow for customization through optional arguments:

    Argument Description name An optional name for the task. If not provided, the name will be inferred from the function name. description An optional string description for the task. If not provided, the description will be pulled from the docstring for the decorated function. tags An optional set of tags to be associated with runs of this task. These tags are combined with any tags defined by a prefect.tags context at task runtime. cache_key_fn An optional callable that, given the task run context and call parameters, generates a string key. If the key matches a previous completed state, that state result will be restored instead of running the task again. cache_expiration An optional amount of time indicating how long cached states for this task should be restorable; if not provided, cached states will never expire. retries An optional number of times to retry on task run failure. retry_delay_seconds An optional number of seconds to wait before retrying the task after failure. This is only applicable if retries is nonzero. log_prints An optional boolean indicating whether to log print statements. persist_result An optional boolean indicating whether to persist the result of the task run to storage.

    See all possible parameters in the Python SDK API docs.

    For example, you can provide a name value for the task. Here we've used the optional description argument as well.

    @task(name=\"hello-task\", \n      description=\"This task says hello.\")\ndef my_task():\n    print(\"Hello, I'm a task\")\n

    You can distinguish runs of this task by providing a task_run_name; this setting accepts a string that can optionally contain templated references to the keyword arguments of your task. The name will be formatted using Python's standard string formatting syntax as can be seen here:

    import datetime\nfrom prefect import flow, task\n\n@task(name=\"My Example Task\", \n      description=\"An example task for a tutorial.\",\n      task_run_name=\"hello-{name}-on-{date:%A}\")\ndef my_task(name, date):\n    pass\n\n@flow\ndef my_flow():\n    # creates a run with a name like \"hello-marvin-on-Thursday\"\n    my_task(name=\"marvin\", date=datetime.datetime.utcnow())\n

    Additionally this setting also accepts a function that returns a string to be used for the task run name:

    import datetime\nfrom prefect import flow, task\n\ndef generate_task_name():\n    date = datetime.datetime.utcnow()\n    return f\"{date:%A}-is-a-lovely-day\"\n\n@task(name=\"My Example Task\",\n      description=\"An example task for a tutorial.\",\n      task_run_name=generate_task_name)\ndef my_task(name):\n    pass\n\n@flow\ndef my_flow():\n    # creates a run with a name like \"Thursday-is-a-lovely-day\"\n    my_task(name=\"marvin\")\n

    If you need access to information about the task, use the prefect.runtime module. For example:

    from prefect import flow\nfrom prefect.runtime import flow_run, task_run\n\ndef generate_task_name():\n    flow_name = flow_run.flow_name\n    task_name = task_run.task_name\n\n    parameters = task_run.parameters\n    name = parameters[\"name\"]\n    limit = parameters[\"limit\"]\n\n    return f\"{flow_name}-{task_name}-with-{name}-and-{limit}\"\n\n@task(name=\"my-example-task\",\n      description=\"An example task for a tutorial.\",\n      task_run_name=generate_task_name)\ndef my_task(name: str, limit: int = 100):\n    pass\n\n@flow\ndef my_flow(name: str):\n    # creates a run with a name like \"my-flow-my-example-task-with-marvin-and-100\"\n    my_task(name=\"marvin\")\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#tags","title":"Tags","text":"

    Tags are optional string labels that enable you to identify and group tasks other than by name or flow. Tags are useful for:

    • Filtering task runs by tag in the UI and via the Prefect REST API.
    • Setting concurrency limits on task runs by tag.

    Tags may be specified as a keyword argument on the task decorator.

    @task(name=\"hello-task\", tags=[\"test\"])\ndef my_task():\n    print(\"Hello, I'm a task\")\n

    You can also provide tags as an argument with a tags context manager, specifying tags when the task is called rather than in its definition.

    from prefect import flow, task\nfrom prefect import tags\n\n@task\ndef my_task():\n    print(\"Hello, I'm a task\")\n\n@flow\ndef my_flow():\n    with tags(\"test\"):\n        my_task()\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#retries","title":"Retries","text":"

    Prefect can automatically retry tasks on failure. In Prefect, a task fails if its Python function raises an exception.

    To enable retries, pass retries and retry_delay_seconds parameters to your task. If the task fails, Prefect will retry it up to retries times, waiting retry_delay_seconds seconds between each attempt. If the task fails on the final retry, Prefect marks the task as crashed if the task raised an exception or failed if it returned a string.

    Retries don't create new task runs

    A new task run is not created when a task is retried. A new state is added to the state history of the original task run.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#a-real-world-example-making-an-api-request","title":"A real-world example: making an API request","text":"

    Consider the real-world problem of making an API request. In this example, we'll use the httpx library to make an HTTP request.

    import httpx\n\nfrom prefect import flow, task\n\n\n@task(retries=2, retry_delay_seconds=5)\ndef get_data_task(\n    url: str = \"https://api.brittle-service.com/endpoint\"\n) -> dict:\n    response = httpx.get(url)\n\n    # If the response status code is anything but a 2xx, httpx will raise\n    # an exception. This task doesn't handle the exception, so Prefect will\n    # catch the exception and will consider the task run failed.\n    response.raise_for_status()\n\n    return response.json()\n\n\n@flow\ndef get_data_flow():\n    get_data_task()\n

    In this task, if the HTTP request to the brittle API receives any status code other than a 2xx (200, 201, etc.), Prefect will retry the task a maximum of two times, waiting five seconds in between retries.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#custom-retry-behavior","title":"Custom retry behavior","text":"

    The retry_delay_seconds option accepts a list of delays for more custom retry behavior. The following task will wait for successively increasing intervals of 1, 10, and 100 seconds, respectively, before the next attempt starts:

    from prefect import task\n\n@task(retries=3, retry_delay_seconds=[1, 10, 100])\ndef some_task_with_manual_backoff_retries():\n   ...\n

    The retry_condition_fn option accepts a callable that returns a boolean. If the callable returns True, the task will be retried. If the callable returns False, the task will not be retried. The callable accepts three arguments \u2014 the task, the task run, and the state of the task run. The following task will retry on HTTP status codes other than 401 or 404:

    import httpx\nfrom prefect import flow, task\n\ndef retry_handler(task, task_run, state) -> bool:\n    \"\"\"This is a custom retry handler to handle when we want to retry a task\"\"\"\n    try:\n        # Attempt to get the result of the task\n        state.result()\n    except httpx.HTTPStatusError as exc:\n        # Retry on any HTTP status code that is not 401 or 404\n        do_not_retry_on_these_codes = [401, 404]\n        return exc.response.status_code not in do_not_retry_on_these_codes\n    except httpx.ConnectError:\n        # Do not retry\n        return False\n\n    # For any other exception, retry\n    return True\n\n@task(retries=1, retry_condition_fn=retry_handler)\ndef my_api_call_task(url):\n    response = httpx.get(url)\n    response.raise_for_status()\n    return response.json()\n\n@flow\ndef get_data_flow(url):\n    my_api_call_task(url=url)\n\nif __name__ == \"__main__\":\n    get_data_flow(url=\"https://httpbin.org/status/503\")\n

    Additionally, you can pass a callable that accepts the number of retries as an argument and returns a list. Prefect includes an exponential_backoff utility that will automatically generate a list of retry delays that correspond to an exponential backoff retry strategy. The following flow will wait for 10, 20, then 40 seconds before each retry.

    from prefect import task\nfrom prefect.tasks import exponential_backoff\n\n@task(retries=3, retry_delay_seconds=exponential_backoff(backoff_factor=10))\ndef some_task_with_exponential_backoff_retries():\n   ...\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#advanced-topic-adding-jitter","title":"Advanced topic: adding \"jitter\"","text":"

    While using exponential backoff, you may also want to add jitter to the delay times. Jitter is a random amount of time added to retry periods that helps prevent \"thundering herd\" scenarios, which is when many tasks all retry at the exact same time, potentially overwhelming systems.

    The retry_jitter_factor option can be used to add variance to the base delay. For example, a retry delay of 10 seconds with a retry_jitter_factor of 0.5 will be allowed to delay up to 15 seconds. Large values of retry_jitter_factor provide more protection against \"thundering herds,\" while keeping the average retry delay time constant. For example, the following task adds jitter to its exponential backoff so the retry delays will vary up to a maximum delay time of 20, 40, and 80 seconds respectively.

    from prefect import task\nfrom prefect.tasks import exponential_backoff\n\n@task(\n    retries=3,\n    retry_delay_seconds=exponential_backoff(backoff_factor=10),\n    retry_jitter_factor=1,\n)\ndef some_task_with_exponential_backoff_retries():\n   ...\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#configuring-retry-behavior-globally-with-settings","title":"Configuring retry behavior globally with settings","text":"

    You can also set retries and retry delays by using the following global settings. These settings will not override the retries or retry_delay_seconds that are set in the flow or task decorator.

    prefect config set PREFECT_FLOW_DEFAULT_RETRIES=2\nprefect config set PREFECT_TASK_DEFAULT_RETRIES=2\nprefect config set PREFECT_FLOW_DEFAULT_RETRY_DELAY_SECONDS = [1, 10, 100]\nprefect config set PREFECT_TASK_DEFAULT_RETRY_DELAY_SECONDS = [1, 10, 100]\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#caching","title":"Caching","text":"

    Caching refers to the ability of a task run to reflect a finished state without actually running the code that defines the task. This allows you to efficiently reuse results of tasks that may be expensive to run with every flow run, or reuse cached results if the inputs to a task have not changed.

    To determine whether a task run should retrieve a cached state, we use \"cache keys\". A cache key is a string value that indicates if one run should be considered identical to another. When a task run with a cache key finishes, we attach that cache key to the state. When each task run starts, Prefect checks for states with a matching cache key. If a state with an identical key is found, Prefect will use the cached state instead of running the task again.

    To enable caching, specify a cache_key_fn \u2014 a function that returns a cache key \u2014 on your task. You may optionally provide a cache_expiration timedelta indicating when the cache expires. If you do not specify a cache_expiration, the cache key does not expire.

    You can define a task that is cached based on its inputs by using the Prefect task_input_hash. This is a task cache key implementation that hashes all inputs to the task using a JSON or cloudpickle serializer. If the task inputs do not change, the cached results are used rather than running the task until the cache expires.

    Note that, if any arguments are not JSON serializable, the pickle serializer is used as a fallback. If cloudpickle fails, task_input_hash returns a null key indicating that a cache key could not be generated for the given inputs.

    In this example, until the cache_expiration time ends, as long as the input to hello_task() remains the same when it is called, the cached return value is returned. In this situation the task is not rerun. However, if the input argument value changes, hello_task() runs using the new input.

    from datetime import timedelta\nfrom prefect import flow, task\nfrom prefect.tasks import task_input_hash\n\n@task(cache_key_fn=task_input_hash, cache_expiration=timedelta(days=1))\ndef hello_task(name_input):\n    # Doing some work\n    print(\"Saying hello\")\n    return \"hello \" + name_input\n\n@flow\ndef hello_flow(name_input):\n    hello_task(name_input)\n

    Alternatively, you can provide your own function or other callable that returns a string cache key. A generic cache_key_fn is a function that accepts two positional arguments:

    • The first argument corresponds to the TaskRunContext, which stores task run metadata in the attributes task_run_id, flow_run_id, and task.
    • The second argument corresponds to a dictionary of input values to the task. For example, if your task is defined with signature fn(x, y, z) then the dictionary will have keys \"x\", \"y\", and \"z\" with corresponding values that can be used to compute your cache key.

    Note that the cache_key_fn is not defined as a @task.

    Task cache keys

    By default, a task cache key is limited to 2000 characters, specified by the PREFECT_API_TASK_CACHE_KEY_MAX_LENGTH setting.

    from prefect import task, flow\n\ndef static_cache_key(context, parameters):\n    # return a constant\n    return \"static cache key\"\n\n@task(cache_key_fn=static_cache_key)\ndef cached_task():\n    print('running an expensive operation')\n    return 42\n\n@flow\ndef test_caching():\n    cached_task()\n    cached_task()\n    cached_task()\n

    In this case, there's no expiration for the cache key, and no logic to change the cache key, so cached_task() only runs once.

    >>> test_caching()\nrunning an expensive operation\n>>> test_caching()\n>>> test_caching()\n

    When each task run requested to enter a Running state, it provided its cache key computed from the cache_key_fn. The Prefect backend identified that there was a COMPLETED state associated with this key and instructed the run to immediately enter the same COMPLETED state, including the same return values.

    A real-world example might include the flow run ID from the context in the cache key so only repeated calls in the same flow run are cached.

    def cache_within_flow_run(context, parameters):\n    return f\"{context.task_run.flow_run_id}-{task_input_hash(context, parameters)}\"\n\n@task(cache_key_fn=cache_within_flow_run)\ndef cached_task():\n    print('running an expensive operation')\n    return 42\n

    Task results, retries, and caching

    Task results are cached in memory during a flow run and persisted to the location specified by the PREFECT_LOCAL_STORAGE_PATH setting. As a result, task caching between flow runs is currently limited to flow runs with access to that local storage path.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#refreshing-the-cache","title":"Refreshing the cache","text":"

    Sometimes, you want a task to update the data associated with its cache key instead of using the cache. This is a cache \"refresh\".

    The refresh_cache option can be used to enable this behavior for a specific task:

    import random\n\n\ndef static_cache_key(context, parameters):\n    # return a constant\n    return \"static cache key\"\n\n\n@task(cache_key_fn=static_cache_key, refresh_cache=True)\ndef caching_task():\n    return random.random()\n

    When this task runs, it will always update the cache key instead of using the cached value. This is particularly useful when you have a flow that is responsible for updating the cache.

    If you want to refresh the cache for all tasks, you can use the PREFECT_TASKS_REFRESH_CACHE setting. Setting PREFECT_TASKS_REFRESH_CACHE=true will change the default behavior of all tasks to refresh. This is particularly useful if you want to rerun a flow without cached results.

    If you have tasks that should not refresh when this setting is enabled, you may explicitly set refresh_cache to False. These tasks will never refresh the cache \u2014 if a cache key exists it will be read, not updated. Note that, if a cache key does not exist yet, these tasks can still write to the cache.

    @task(cache_key_fn=static_cache_key, refresh_cache=False)\ndef caching_task():\n    return random.random()\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#timeouts","title":"Timeouts","text":"

    Task timeouts are used to prevent unintentional long-running tasks. When the duration of execution for a task exceeds the duration specified in the timeout, a timeout exception will be raised and the task will be marked as failed. In the UI, the task will be visibly designated as TimedOut. From the perspective of the flow, the timed-out task will be treated like any other failed task.

    Timeout durations are specified using the timeout_seconds keyword argument.

    from prefect import task, get_run_logger\nimport time\n\n@task(timeout_seconds=1)\ndef show_timeouts():\n    logger = get_run_logger()\n    logger.info(\"I will execute\")\n    time.sleep(5)\n    logger.info(\"I will not execute\")\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#task-results","title":"Task results","text":"

    Depending on how you call tasks, they can return different types of results and optionally engage the use of a task runner.

    Any task can return:

    • Data\u200a, such as int, str, dict, list, and so on \u2014 \u200athis is the default behavior any time you call your_task().
    • PrefectFuture \u2014 \u200athis is achieved by calling your_task.submit(). A PrefectFuture contains both data and State
    • Prefect State \u200a\u2014 anytime you call your task or flow with the argument return_state=True, it will directly return a state you can use to build custom behavior based on a state change you care about, such as task or flow failing or retrying.

    To run your task with a task runner, you must call the task with .submit().

    See state returned values for examples.

    Task runners are optional

    If you just need the result from a task, you can simply call the task from your flow. For most workflows, the default behavior of calling a task directly and receiving a result is all you'll need.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#wait-for","title":"Wait for","text":"

    To create a dependency between two tasks that do not exchange data, but one needs to wait for the other to finish, use the special wait_for keyword argument:

    @task\ndef task_1():\n    pass\n\n@task\ndef task_2():\n    pass\n\n@flow\ndef my_flow():\n    x = task_1()\n\n    # task 2 will wait for task_1 to complete\n    y = task_2(wait_for=[x])\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#map","title":"Map","text":"

    Prefect provides a .map() implementation that automatically creates a task run for each element of its input data. Mapped tasks represent the computations of many individual children tasks.

    The simplest Prefect map takes a tasks and applies it to each element of its inputs.

    from prefect import flow, task\n\n@task\ndef print_nums(nums):\n    for n in nums:\n        print(n)\n\n@task\ndef square_num(num):\n    return num**2\n\n@flow\ndef map_flow(nums):\n    print_nums(nums)\n    squared_nums = square_num.map(nums) \n    print_nums(squared_nums)\n\nmap_flow([1,2,3,5,8,13])\n

    Prefect also supports unmapped arguments, allowing you to pass static values that don't get mapped over.

    from prefect import flow, task\n\n@task\ndef add_together(x, y):\n    return x + y\n\n@flow\ndef sum_it(numbers, static_value):\n    futures = add_together.map(numbers, static_value)\n    return futures\n\nsum_it([1, 2, 3], 5)\n

    If your static argument is an iterable, you'll need to wrap it with unmapped to tell Prefect that it should be treated as a static value.

    from prefect import flow, task, unmapped\n\n@task\ndef sum_plus(x, static_iterable):\n    return x + sum(static_iterable)\n\n@flow\ndef sum_it(numbers, static_iterable):\n    futures = sum_plus.map(numbers, static_iterable)\n    return futures\n\nsum_it([4, 5, 6], unmapped([1, 2, 3]))\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#async-tasks","title":"Async tasks","text":"

    Prefect also supports asynchronous task and flow definitions by default. All of the standard rules of async apply:

    import asyncio\n\nfrom prefect import task, flow\n\n@task\nasync def print_values(values):\n    for value in values:\n        await asyncio.sleep(1) # yield\n        print(value, end=\" \")\n\n@flow\nasync def async_flow():\n    await print_values([1, 2])  # runs immediately\n    coros = [print_values(\"abcd\"), print_values(\"6789\")]\n\n    # asynchronously gather the tasks\n    await asyncio.gather(*coros)\n\nasyncio.run(async_flow())\n

    Note, if you are not using asyncio.gather, calling .submit() is required for asynchronous execution on the ConcurrentTaskRunner.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#task-run-concurrency-limits","title":"Task run concurrency limits","text":"

    There are situations in which you want to actively prevent too many tasks from running simultaneously. For example, if many tasks across multiple flows are designed to interact with a database that only allows 10 connections, you want to make sure that no more than 10 tasks that connect to this database are running at any given time.

    Prefect has built-in functionality for achieving this: task concurrency limits.

    Task concurrency limits use task tags. You can specify an optional concurrency limit as the maximum number of concurrent task runs in a Running state for tasks with a given tag. The specified concurrency limit applies to any task to which the tag is applied.

    If a task has multiple tags, it will run only if all tags have available concurrency.

    Tags without explicit limits are considered to have unlimited concurrency.

    0 concurrency limit aborts task runs

    Currently, if the concurrency limit is set to 0 for a tag, any attempt to run a task with that tag will be aborted instead of delayed.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#execution-behavior","title":"Execution behavior","text":"

    Task tag limits are checked whenever a task run attempts to enter a Running state.

    If there are no concurrency slots available for any one of your task's tags, the transition to a Running state will be delayed and the client is instructed to try entering a Running state again in 30 seconds (or the value specified by the PREFECT_TASK_RUN_TAG_CONCURRENCY_SLOT_WAIT_SECONDS setting).

    Concurrency limits in subflows

    Using concurrency limits on task runs in subflows can cause deadlocks. As a best practice, configure your tags and concurrency limits to avoid setting limits on task runs in subflows.

    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#configuring-concurrency-limits","title":"Configuring concurrency limits","text":"

    Flow run concurrency limits are set at a work pool and/or work queue level

    While task run concurrency limits are configured via tags (as shown below), flow run concurrency limits are configured via work pools and/or work queues.

    You can set concurrency limits on as few or as many tags as you wish. You can set limits through:

    • Prefect CLI
    • Prefect API by using PrefectClient Python client
    • Prefect server UI or Prefect Cloud
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#cli","title":"CLI","text":"

    You can create, list, and remove concurrency limits by using Prefect CLI concurrency-limit commands.

    $ prefect concurrency-limit [command] [arguments]\n
    Command Description create Create a concurrency limit by specifying a tag and limit. delete Delete the concurrency limit set on the specified tag. inspect View details about a concurrency limit set on the specified tag. ls View all defined concurrency limits.

    For example, to set a concurrency limit of 10 on the 'small_instance' tag:

    $ prefect concurrency-limit create small_instance 10\n

    To delete the concurrency limit on the 'small_instance' tag:

    $ prefect concurrency-limit delete small_instance\n

    To view details about the concurrency limit on the 'small_instance' tag:

    $ prefect concurrency-limit inspect small_instance\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/tasks/#python-client","title":"Python client","text":"

    To update your tag concurrency limits programmatically, use PrefectClient.orchestration.create_concurrency_limit.

    create_concurrency_limit takes two arguments:

    • tag specifies the task tag on which you're setting a limit.
    • concurrency_limit specifies the maximum number of concurrent task runs for that tag.

    For example, to set a concurrency limit of 10 on the 'small_instance' tag:

    from prefect import get_client\n\nasync with get_client() as client:\n    # set a concurrency limit of 10 on the 'small_instance' tag\n    limit_id = await client.create_concurrency_limit(\n        tag=\"small_instance\", \n        concurrency_limit=10\n        )\n

    To remove all concurrency limits on a tag, use PrefectClient.delete_concurrency_limit_by_tag, passing the tag:

    async with get_client() as client:\n    # remove a concurrency limit on the 'small_instance' tag\n    await client.delete_concurrency_limit_by_tag(tag=\"small_instance\")\n

    If you wish to query for the currently set limit on a tag, use PrefectClient.read_concurrency_limit_by_tag, passing the tag:

    To see all of your limits across all of your tags, use PrefectClient.read_concurrency_limits.

    async with get_client() as client:\n    # query the concurrency limit on the 'small_instance' tag\n    limit = await client.read_concurrency_limit_by_tag(tag=\"small_instance\")\n
    ","tags":["tasks","task runs","functions","retries","caching","cache keys","cache key functions","tags","results","async","asynchronous execution","map","concurrency","concurrency limits","task concurrency"],"boost":2},{"location":"concepts/work-pools/","title":"Work Pools & Workers","text":"

    Work pools and workers bridge the Prefect orchestration environment with your execution environment. When a deployment creates a flow run, it is submitted to a specific work pool for scheduling. A worker running in the execution environment can poll its respective work pool for new runs to execute, or the work pool can submit flow runs to serverless infrastructure directly, depending on your configuration.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-pool-overview","title":"Work pool overview","text":"

    Work pools organize work for execution. Work pools have types corresponding to the infrastructure that will execute the flow code, as well as the delivery method of work to that environment. Pull work pools require workers (or less ideally, agents) to poll the work pool for flow runs to execute. Push work pools can submit runs directly to your serverless infrastructure providers such as Google Cloud Run, Azure Container Instances, and AWS ECS without the need for an agent or worker. Managed work pools are administered by Prefect and handle the submission and execution of code on your behalf.

    Work pools are like pub/sub topics

    It's helpful to think of work pools as a way to coordinate (potentially many) deployments with (potentially many) workers through a known channel: the pool itself. This is similar to how \"topics\" are used to connect producers and consumers in a pub/sub or message-based system. By switching a deployment's work pool, users can quickly change the worker that will execute their runs, making it easy to promote runs through environments or even debug locally.

    In addition, users can control aspects of work pool behavior, such as how many runs the pool allows to be run concurrently or pausing delivery entirely. These options can be modified at any time, and any workers requesting work for a specific pool will only see matching flow runs.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-pool-configuration","title":"Work pool configuration","text":"

    You can configure work pools by using:

    • Prefect CLI commands
    • Prefect Python API
    • Prefect UI

    To manage work pools in the UI, click the Work Pools icon. This displays a list of currently configured work pools.

    You can pause a work pool from this page by using the toggle.

    Select the + button to create a new work pool. You'll be able to specify the details for work served by this work pool.

    To create a work pool via the Prefect CLI, use the prefect work-pool create command:

    prefect work-pool create [OPTIONS] NAME\n

    NAME is a required, unique name for the work pool.

    Optional configuration parameters you can specify to filter work on the pool include:

    Option Description --paused If provided, the work pool will be created in a paused state. --type The type of infrastructure that can execute runs from this work pool. --set-as-default Whether to use the created work pool as the local default for deployment. --base-job-template The path to a JSON file containing the base job template to use. If unspecified, Prefect will use the default base job template for the given worker type.

    For example, to create a work pool called test-pool, you would run this command:

    prefect work-pool create test-pool\n
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-pool-types","title":"Work pool types","text":"

    If you don't use the --type flag to specify an infrastructure type, you are prompted to select from the following options:

    Prefect CloudPrefect server instance Infrastructure Type Description Prefect Agent Execute flow runs on heterogeneous infrastructure using infrastructure blocks. Local Subprocess Execute flow runs as subprocesses on a worker. Works well for local execution when first getting started. AWS Elastic Container Service Execute flow runs within containers on AWS ECS. Works with EC2 and Fargate clusters. Requires an AWS account. Azure Container Instances Execute flow runs within containers on Azure's Container Instances service. Requires an Azure account. Docker Execute flow runs within Docker containers. Works well for managing flow execution environments via Docker images. Requires access to a running Docker daemon. Google Cloud Run Execute flow runs within containers on Google Cloud Run. Requires a Google Cloud Platform account. Google Cloud Run V2 Execute flow runs within containers on Google Cloud Run (V2 API). Requires a Google Cloud Platform account. Google Vertex AI Execute flow runs within containers on Google Vertex AI. Requires a Google Cloud Platform account. Kubernetes Execute flow runs within jobs scheduled on a Kubernetes cluster. Requires a Kubernetes cluster. Google Cloud Run - Push Execute flow runs within containers on Google Cloud Run. Requires a Google Cloud Platform account. Flow runs are pushed directly to your environment, without the need for a Prefect worker. AWS Elastic Container Service - Push Execute flow runs within containers on AWS ECS. Works with existing ECS clusters and serverless execution via AWS Fargate. Requires an AWS account. Flow runs are pushed directly to your environment, without the need for a Prefect worker. Azure Container Instances - Push Execute flow runs within containers on Azure's Container Instances service. Requires an Azure account. Flow runs are pushed directly to your environment, without the need for a Prefect worker. Prefect Managed Execute flow runs within containers on Prefect managed infrastructure. Infrastructure Type Description Prefect Agent Execute flow runs on heterogeneous infrastructure using infrastructure blocks. Local Subprocess Execute flow runs as subprocesses on a worker. Works well for local execution when first getting started. AWS Elastic Container Service Execute flow runs within containers on AWS ECS. Works with EC2 and Fargate clusters. Requires an AWS account. Azure Container Instances Execute flow runs within containers on Azure's Container Instances service. Requires an Azure account. Docker Execute flow runs within Docker containers. Works well for managing flow execution environments via Docker images. Requires access to a running Docker daemon. Google Cloud Run Execute flow runs within containers on Google Cloud Run. Requires a Google Cloud Platform account. Google Cloud Run V2 Execute flow runs within containers on Google Cloud Run (V2 API). Requires a Google Cloud Platform account. Google Vertex AI Execute flow runs within containers on Google Vertex AI. Requires a Google Cloud Platform account. Kubernetes Execute flow runs within jobs scheduled on a Kubernetes cluster. Requires a Kubernetes cluster.

    On success, the command returns the details of the newly created work pool.

    Created work pool with properties:\n    name - 'test-pool'\n    id - a51adf8c-58bb-4949-abe6-1b87af46eabd\n    concurrency limit - None\n\nStart a worker to pick up flows from the work pool:\n    prefect worker start -p 'test-pool'\n\nInspect the work pool:\n    prefect work-pool inspect 'test-pool'\n

    Set a work pool as the default for new deployments by adding the --set-as-default flag.

    Which would result in output similar to the following:

    Set 'test-pool' as default work pool for profile 'default'\n\nTo change your default work pool, run:\n\n        prefect config set PREFECT_DEFAULT_WORK_POOL_NAME=<work-pool-name>\n

    To update a work pool via the Prefect CLI, use the prefect work-pool update command:

    prefect work-pool update [OPTIONS] NAME\n

    NAME is the name of the work pool to update.

    Optional configuration parameters you can specify to update the work pool include:

    Option Description --base-job-template The path to a JSON file containing the base job template to use. If unspecified, Prefect will use the default base job template for the given worker type. --description A description of the work pool. --concurrency-limit The maximum number of flow runs to run simultaneously in the work pool.

    Managing work pools in CI/CD

    You can version control your base job template by committing it as a JSON file to your repository and control updates to your work pools' base job templates by using the prefect work-pool update command in your CI/CD pipeline. For example, you could use the following command to update a work pool's base job template to the contents of a file named base-job-template.json:

    prefect work-pool update --base-job-template base-job-template.json my-work-pool\n
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#base-job-template","title":"Base job template","text":"

    Each work pool has a base job template that allows the customization of the behavior of the worker executing flow runs from the work pool.

    The base job template acts as a contract defining the configuration passed to the worker for each flow run and the options available to deployment creators to customize worker behavior per deployment.

    A base job template comprises a job_configuration section and a variables section.

    The variables section defines the fields available to be customized per deployment. The variables section follows the OpenAPI specification, which allows work pool creators to place limits on provided values (type, minimum, maximum, etc.).

    The job configuration section defines how values provided for fields in the variables section should be translated into the configuration given to a worker when executing a flow run.

    The values in the job_configuration can use placeholders to reference values provided in the variables section. Placeholders are declared using double curly braces, e.g., {{ variable_name }}. job_configuration values can also be hard-coded if the value should not be customizable.

    Each worker type is configured with a default base job template, making it easy to start with a work pool. The default base template defines fields that can be edited on a per-deployment basis or for the entire work pool via the Prefect API and UI.

    For example, if we create a process work pool named 'above-ground' via the CLI:

    prefect work-pool create --type process above-ground\n

    We see these configuration options available in the Prefect UI:

    For a process work pool with the default base job template, we can set environment variables for spawned processes, set the working directory to execute flows, and control whether the flow run output is streamed to workers' standard output. You can also see an example of JSON formatted base job template with the 'Advanced' tab.

    You can examine the default base job template for a given worker type by running:

    prefect work-pool get-default-base-job-template --type process\n
    {\n  \"job_configuration\": {\n    \"command\": \"{{ command }}\",\n    \"env\": \"{{ env }}\",\n    \"labels\": \"{{ labels }}\",\n    \"name\": \"{{ name }}\",\n    \"stream_output\": \"{{ stream_output }}\",\n    \"working_dir\": \"{{ working_dir }}\"\n  },\n  \"variables\": {\n    \"type\": \"object\",\n    \"properties\": {\n      \"name\": {\n        \"title\": \"Name\",\n        \"description\": \"Name given to infrastructure created by a worker.\",\n        \"type\": \"string\"\n      },\n      \"env\": {\n        \"title\": \"Environment Variables\",\n        \"description\": \"Environment variables to set when starting a flow run.\",\n        \"type\": \"object\",\n        \"additionalProperties\": {\n          \"type\": \"string\"\n        }\n      },\n      \"labels\": {\n        \"title\": \"Labels\",\n        \"description\": \"Labels applied to infrastructure created by a worker.\",\n        \"type\": \"object\",\n        \"additionalProperties\": {\n          \"type\": \"string\"\n        }\n      },\n      \"command\": {\n        \"title\": \"Command\",\n        \"description\": \"The command to use when starting a flow run. In most cases, this should be left blank and the command will be automatically generated by the worker.\",\n        \"type\": \"string\"\n      },\n      \"stream_output\": {\n        \"title\": \"Stream Output\",\n        \"description\": \"If enabled, workers will stream output from flow run processes to local standard output.\",\n        \"default\": true,\n        \"type\": \"boolean\"\n      },\n      \"working_dir\": {\n        \"title\": \"Working Directory\",\n        \"description\": \"If provided, workers will open flow run processes within the specified path as the working directory. Otherwise, a temporary directory will be created.\",\n        \"type\": \"string\",\n        \"format\": \"path\"\n      }\n    }\n  }\n}\n

    You can override each of these attributes on a per-deployment basis. When deploying a flow, you can specify these overrides in the work_pool.job_variables section of a deployment.yaml.

    If we wanted to turn off streaming output for a specific deployment, we could add the following to our deployment.yaml:

    work_pool:\n    name: above-ground  \n    job_variables:\n        stream_output: false\n

    Advanced Customization of the Base Job Template

    For advanced use cases, you can create work pools with fully customizable job templates. This customization is available when creating or editing a work pool on the 'Advanced' tab within the UI or when updating a work pool via the Prefect CLI.

    Advanced customization is useful anytime the underlying infrastructure supports a high degree of customization. In these scenarios a work pool job template allows you to expose a minimal and easy-to-digest set of options to deployment authors. Additionally, these options are the only customizable aspects for deployment infrastructure, which can be useful for restricting functionality in secure environments. For example, the kubernetes worker type allows users to specify a custom job template that can be used to configure the manifest that workers use to create jobs for flow execution.

    For more information and advanced configuration examples, see the Kubernetes Worker documentation.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#viewing-work-pools","title":"Viewing work pools","text":"

    At any time, users can see and edit configured work pools in the Prefect UI.

    To view work pools with the Prefect CLI, you can:

    • List (ls) all available pools
    • Inspect (inspect) the details of a single pool
    • Preview (preview) scheduled work for a single pool

    prefect work-pool ls lists all configured work pools for the server.

    prefect work-pool ls\n

    For example:

                                   Work pools\n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name       \u2503    Type        \u2503                                   ID \u2503 Concurrency Limit \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 barbeque   \u2502 docker         \u2502 72c0a101-b3e2-4448-b5f8-a8c5184abd17 \u2502 None              \u2502\n\u2502 k8s-pool   \u2502 kubernetes     \u2502 7b6e3523-d35b-4882-84a7-7a107325bb3f \u2502 None              \u2502\n\u2502 test-pool  \u2502 prefect-agent  \u2502 a51adf8c-58bb-4949-abe6-1b87af46eabd \u2502 None              |\n| my-pool    \u2502 process        \u2502 cd6ff9e8-bfd8-43be-9be3-69375f7a11cd \u2502 None              \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n                       (**) denotes a paused pool\n

    prefect work-pool inspect provides all configuration metadata for a specific work pool by ID.

    prefect work-pool inspect 'test-pool'\n

    Outputs information similar to the following:

    Workpool(\n    id='a51adf8c-58bb-4949-abe6-1b87af46eabd',\n    created='2 minutes ago',\n    updated='2 minutes ago',\n    name='test-pool',\n    filter=None,\n)\n

    prefect work-pool preview displays scheduled flow runs for a specific work pool by ID for the upcoming hour. The optional --hours flag lets you specify the number of hours to look ahead.

    prefect work-pool preview 'test-pool' --hours 12\n
    \u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Scheduled Star\u2026 \u2503 Run ID                     \u2503 Name         \u2503 Deployment ID               \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 2022-02-26 06:\u2026 \u2502 741483d4-dc90-4913-b88d-0\u2026 \u2502 messy-petrel \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 05:\u2026 \u2502 14e23a19-a51b-4833-9322-5\u2026 \u2502 unselfish-g\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 04:\u2026 \u2502 deb44d4d-5fa2-4f70-a370-e\u2026 \u2502 solid-ostri\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 03:\u2026 \u2502 07374b5c-121f-4c8d-9105-b\u2026 \u2502 sophisticat\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 02:\u2026 \u2502 545bc975-b694-4ece-9def-8\u2026 \u2502 gorgeous-mo\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 01:\u2026 \u2502 704f2d67-9dfa-4fb8-9784-4\u2026 \u2502 sassy-hedge\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-26 00:\u2026 \u2502 691312f0-d142-4218-b617-a\u2026 \u2502 sincere-moo\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-25 23:\u2026 \u2502 7cb3ff96-606b-4d8c-8a33-4\u2026 \u2502 curious-cat\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-25 22:\u2026 \u2502 3ea559fe-cb34-43b0-8090-1\u2026 \u2502 primitive-f\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2502 2022-02-25 21:\u2026 \u2502 96212e80-426d-4bf4-9c49-e\u2026 \u2502 phenomenal-\u2026 \u2502 156edead-fe6a-4783-a618-21\u2026 \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n                                   (**) denotes a late run\n
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-pool-status","title":"Work Pool Status","text":"

    Work pools have three statuses: READY, NOT_READY, and PAUSED. A work pool is considered ready if it has at least one online worker sending heartbeats to the work pool. If a work pool has no online workers, it is considered not ready to execute work. A work pool can be placed in a paused status manually by a user or via an automation. When a paused work pool is unpaused, it will be reassigned the appropriate status based on whether any workers are sending heartbeats.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#pausing-and-deleting-work-pools","title":"Pausing and deleting work pools","text":"

    A work pool can be paused at any time to stop the delivery of work to workers. Workers will not receive any work when polling a paused pool.

    To pause a work pool through the Prefect CLI, use the prefect work-pool pause command:

    prefect work-pool pause 'test-pool'\n

    To resume a work pool through the Prefect CLI, use the prefect work-pool resume command with the work pool name.

    To delete a work pool through the Prefect CLI, use the prefect work-pool delete command with the work pool name.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#managing-concurrency","title":"Managing concurrency","text":"

    Each work pool can optionally restrict concurrent runs of matching flows.

    For example, a work pool with a concurrency limit of 5 will only release new work if fewer than 5 matching runs are currently in a Running or Pending state. If 3 runs are Running or Pending, polling the pool for work will only result in 2 new runs, even if there are many more available, to ensure that the concurrency limit is not exceeded.

    When using the prefect work-pool Prefect CLI command to configure a work pool, the following subcommands set concurrency limits:

    • set-concurrency-limit sets a concurrency limit on a work pool.
    • clear-concurrency-limit clears any concurrency limits from a work pool.
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#work-queues","title":"Work queues","text":"

    Advanced topic

    Work queues do not require manual creation or configuration, because Prefect will automatically create them whenever needed. Managing work queues offers advanced control over how runs are executed.

    Each work pool has a \"default\" queue that all work will be sent to by default. Additional queues can be added to a work pool. Work queues enable greater control over work delivery through fine grained priority and concurrency. Each work queue has a priority indicated by a unique positive integer. Lower numbers take greater priority in the allocation of work. Accordingly, new queues can be added without changing the rank of the higher-priority queues (e.g. no matter how many queues you add, the queue with priority 1 will always be the highest priority).

    Work queues can also have their own concurrency limits. Note that each queue is also subject to the global work pool concurrency limit, which cannot be exceeded.

    Together work queue priority and concurrency enable precise control over work. For example, a pool may have three queues: A \"low\" queue with priority 10 and no concurrency limit, a \"high\" queue with priority 5 and a concurrency limit of 3, and a \"critical\" queue with priority 1 and a concurrency limit of 1. This arrangement would enable a pattern in which there are two levels of priority, \"high\" and \"low\" for regularly scheduled flow runs, with the remaining \"critical\" queue for unplanned, urgent work, such as a backfill.

    Priority is evaluated to determine the order in which flow runs are submitted for execution. If all flow runs are capable of being executed with no limitation due to concurrency or otherwise, priority is still used to determine order of submission, but there is no impact to execution. If not all flow runs can be executed, usually as a result of concurrency limits, priority is used to determine which queues receive precedence to submit runs for execution.

    Priority for flow run submission proceeds from the highest priority to the lowest priority. In the preceding example, all work from the \"critical\" queue (priority 1) will be submitted, before any work is submitted from \"high\" (priority 5). Once all work has been submitted from priority queue \"critical\", work from the \"high\" queue will begin submission.

    If new flow runs are received on the \"critical\" queue while flow runs are still in scheduled on the \"high\" and \"low\" queues, flow run submission goes back to ensuring all scheduled work is first satisfied from the highest priority queue, until it is empty, in waterfall fashion.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#local-debugging","title":"Local debugging","text":"

    As long as your deployment's infrastructure block supports it, you can use work pools to temporarily send runs to a worker running on your local machine for debugging by running prefect worker start -p my-local-machine and updating the deployment's work pool to my-local-machine.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#worker-overview","title":"Worker overview","text":"

    Workers are lightweight polling services that retrieve scheduled runs from a work pool and execute them.

    Workers are similar to agents, but offer greater control over infrastructure configuration and the ability to route work to specific types of execution environments.

    Workers each have a type corresponding to the execution environment to which they will submit flow runs. Workers are only able to poll work pools that match their type. As a result, when deployments are assigned to a work pool, you know in which execution environment scheduled flow runs for that deployment will run.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#worker-types","title":"Worker types","text":"

    Below is a list of available worker types. Note that most worker types will require installation of an additional package.

    Worker Type Description Required Package process Executes flow runs in subprocesses kubernetes Executes flow runs as Kubernetes jobs prefect-kubernetes docker Executes flow runs within Docker containers prefect-docker ecs Executes flow runs as ECS tasks prefect-aws cloud-run Executes flow runs as Google Cloud Run jobs prefect-gcp vertex-ai Executes flow runs as Google Cloud Vertex AI jobs prefect-gcp azure-container-instance Execute flow runs in ACI containers prefect-azure

    If you don\u2019t see a worker type that meets your needs, consider developing a new worker type!

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#worker-options","title":"Worker options","text":"

    Workers poll for work from one or more queues within a work pool. If the worker references a work queue that doesn't exist, it will be created automatically. The worker CLI is able to infer the worker type from the work pool. Alternatively, you can also specify the worker type explicitly. If you supply the worker type to the worker CLI, a work pool will be created automatically if it doesn't exist (using default job settings).

    Configuration parameters you can specify when starting a worker include:

    Option Description --name, -n The name to give to the started worker. If not provided, a unique name will be generated. --pool, -p The work pool the started worker should poll. --work-queue, -q One or more work queue names for the worker to pull from. If not provided, the worker will pull from all work queues in the work pool. --type, -t The type of worker to start. If not provided, the worker type will be inferred from the work pool. --prefetch-seconds The amount of time before a flow run's scheduled start time to begin submission. Default is the value of PREFECT_WORKER_PREFETCH_SECONDS. --run-once Only run worker polling once. By default, the worker runs forever. --limit, -l The maximum number of flow runs to start simultaneously. --with-healthcheck Start a healthcheck server for the worker. --install-policy Install policy to use workers from Prefect integration packages.

    You must start a worker within an environment that can access or create the infrastructure needed to execute flow runs. The worker will deploy flow runs to the infrastructure corresponding to the worker type. For example, if you start a worker with type kubernetes, the worker will deploy flow runs to a Kubernetes cluster.

    Prefect must be installed in execution environments

    Prefect must be installed in any environment (virtual environment, Docker container, etc.) where you intend to run the worker or execute a flow run.

    PREFECT_API_URL and PREFECT_API_KEYsettings for workers

    PREFECT_API_URL must be set for the environment in which your worker is running. You must also have a user or service account with the Worker role, which can be configured by setting the PREFECT_API_KEY.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#worker-status","title":"Worker status","text":"

    Workers have two statuses: ONLINE and OFFLINE. A worker is online if it sends regular heartbeat messages to the Prefect API. If a worker has missed three heartbeats, it is considered offline. By default, a worker is considered offline a maximum of 90 seconds after it stopped sending heartbeats, but the threshold can be configured via the PREFECT_WORKER_HEARTBEAT_SECONDS setting.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#starting-a-worker","title":"Starting a worker","text":"

    Use the prefect worker start CLI command to start a worker. You must pass at least the work pool name. If the work pool does not exist, it will be created if the --type flag is used.

    prefect worker start -p [work pool name]\n

    For example:

    prefect worker start -p \"my-pool\"\n

    Results in output like this:

    Discovered worker type 'process' for work pool 'my-pool'.\nWorker 'ProcessWorker 65716280-96f8-420b-9300-7e94417f2673' started!\n

    In this case, Prefect automatically discovered the worker type from the work pool. To create a work pool and start a worker in one command, use the --type flag:

    prefect worker start -p \"my-pool\" --type \"process\"\n
    Worker 'ProcessWorker d24f3768-62a9-4141-9480-a056b9539a25' started!\n06:57:53.289 | INFO    | prefect.worker.process.processworker d24f3768-62a9-4141-9480-a056b9539a25 - Worker pool 'my-pool' created.\n

    In addition, workers can limit the number of flow runs they will start simultaneously with the --limit flag. For example, to limit a worker to five concurrent flow runs:

    prefect worker start --pool \"my-pool\" --limit 5\n
    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#configuring-prefetch","title":"Configuring prefetch","text":"

    By default, the worker begins submitting flow runs a short time (10 seconds) before they are scheduled to run. This behavior allows time for the infrastructure to be created so that the flow run can start on time.

    In some cases, infrastructure will take longer than 10 seconds to start the flow run. The prefetch can be increased using the --prefetch-seconds option or the PREFECT_WORKER_PREFETCH_SECONDS setting.

    If this value is more than the amount of time it takes for the infrastructure to start, the flow run will wait until its scheduled start time.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#polling-for-work","title":"Polling for work","text":"

    Workers poll for work every 15 seconds by default. This interval is configurable in your profile settings with the PREFECT_WORKER_QUERY_SECONDS setting.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#install-policy","title":"Install policy","text":"

    The Prefect CLI can install the required package for Prefect-maintained worker types automatically. You can configure this behavior with the --install-policy option. The following are valid install policies

    Install Policy Description always Always install the required package. Will update the required package to the most recent version if already installed. if-not-present Install the required package if it is not already installed. never Never install the required package. prompt Prompt the user to choose whether to install the required package. This is the default install policy. If prefect worker start is run non-interactively, the prompt install policy will behave the same as never.","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"concepts/work-pools/#additional-resources","title":"Additional resources","text":"

    See how to daemonize a Prefect worker in this guide.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","schedules","concurrency limits","priority","work queues"],"boost":2},{"location":"contributing/overview/","title":"Contributing","text":"

    Thanks for considering contributing to Prefect!

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#setting-up-a-development-environment","title":"Setting up a development environment","text":"

    First, you'll need to download the source code and install an editable version of the Python package:

    # Clone the repository\ngit clone https://github.com/PrefectHQ/prefect.git\ncd prefect\n\n# We recommend using a virtual environment\n\npython -m venv .venv\nsource .venv/bin/activate\n\n# Install the package with development dependencies\n\npip install -e \".[dev]\"\n\n# Setup pre-commit hooks for required formatting\n\npre-commit install\n

    If you don't want to install the pre-commit hooks, you can manually install the formatting dependencies with:

    pip install $(./scripts/precommit-versions.py)\n

    You'll need to run black and ruff before a contribution can be accepted.

    After installation, you can run the test suite with pytest:

    # Run all the tests\npytest tests\n\n# Run a subset of tests\n\npytest tests/test_flows.py\n

    Building the Prefect UI

    If you intend to run a local Prefect server during development, you must first build the UI. See UI development for instructions.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#prefect-code-of-conduct","title":"Prefect Code of Conduct","text":"","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#our-pledge","title":"Our Pledge","text":"

    In the interest of fostering an open and welcoming environment, we as contributors and maintainers pledge to making participation in our project and our community a harassment-free experience for everyone, regardless of age, body size, disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, religion, or sexual identity and orientation.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#our-standards","title":"Our Standards","text":"

    Examples of behavior that contributes to creating a positive environment include:

    • Using welcoming and inclusive language
    • Being respectful of differing viewpoints and experiences
    • Gracefully accepting constructive criticism
    • Focusing on what is best for the community
    • Showing empathy towards other community members

    Examples of unacceptable behavior by participants include:

    • The use of sexualized language or imagery and unwelcome sexual attention or advances
    • Trolling, insulting/derogatory comments, and personal or political attacks
    • Public or private harassment
    • Publishing others' private information, such as a physical or electronic address, without explicit permission
    • Other conduct which could reasonably be considered inappropriate in a professional setting
    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#our-responsibilities","title":"Our Responsibilities","text":"

    Project maintainers are responsible for clarifying the standards of acceptable behavior and are expected to take appropriate and fair corrective action in response to any instances of unacceptable behavior.

    Project maintainers have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, or to ban temporarily or permanently any contributor for other behaviors that they deem inappropriate, threatening, offensive, or harmful.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#scope","title":"Scope","text":"

    This Code of Conduct applies within all project spaces, and it also applies when an individual is representing the project or its community in public spaces. Examples of representing a project or community include using an official project e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event. Representation of a project may be further defined and clarified by project maintainers.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#enforcement","title":"Enforcement","text":"

    Instances of abusive, harassing, or otherwise unacceptable behavior may be reported by contacting Chris White at chris@prefect.io. All complaints will be reviewed and investigated and will result in a response that is deemed necessary and appropriate to the circumstances. The project team is obligated to maintain confidentiality with regard to the reporter of an incident. Further details of specific enforcement policies may be posted separately.

    Project maintainers who do not follow or enforce the Code of Conduct in good faith may face temporary or permanent repercussions as determined by other members of the project's leadership.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#attribution","title":"Attribution","text":"

    This Code of Conduct is adapted from the Contributor Covenant, version 1.4, available at https://www.contributor-covenant.org/version/1/4/code-of-conduct.html

    For answers to common questions about this code of conduct, see https://www.contributor-covenant.org/faq

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#developer-tooling","title":"Developer tooling","text":"

    The Prefect CLI provides several helpful commands to aid development.

    Start all services with hot-reloading on code changes (requires UI dependencies to be installed):

    prefect dev start\n

    Start a Prefect API that reloads on code changes:

    prefect dev api\n

    Start a Prefect worker that reloads on code changes:

    prefect dev agent\n
    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#ui-development","title":"UI development","text":"

    Developing the Prefect UI requires that npm is installed.

    Start a development UI that reloads on code changes:

    prefect dev ui\n

    Build the static UI (the UI served by prefect server start):

    prefect dev build-ui\n
    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#docs-development","title":"Docs Development","text":"

    Prefect uses mkdocs for the docs website and the mkdocs-material theme. While we use mkdocs-material-insiders for production, builds can still happen without the extra plugins. Deploy previews are available on pull requests, so you'll be able to browse the final look of your changes before merging.

    To build the docs:

    mkdocs build\n

    To serve the docs locally at http://127.0.0.1:8000/:

    mkdocs serve\n

    For additional mkdocs help and options:

    mkdocs --help\n

    We use the mkdocs-material theme. To add additional JavaScript or CSS to the docs, please see the theme documentation here.

    Internal developers can install the production theme by running:

    pip install -e git+https://github.com/PrefectHQ/mkdocs-material-insiders.git#egg=mkdocs-material\nmkdocs build # or mkdocs build --config-file mkdocs.insiders.yml if needed\n
    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#kubernetes-development","title":"Kubernetes development","text":"

    Generate a manifest to deploy a development API to a local kubernetes cluster:

    prefect dev kubernetes-manifest\n

    To access the Prefect UI running in a Kubernetes cluster, use the kubectl port-forward command to forward a port on your local machine to an open port within the cluster. For example:

    kubectl port-forward deployment/prefect-dev 4200:4200\n

    This forwards port 4200 on the default internal loop IP for localhost to the Prefect server deployment.

    To tell the local prefect command how to communicate with the Prefect API running in Kubernetes, set the PREFECT_API_URL environment variable:

    export PREFECT_API_URL=http://localhost:4200/api\n

    Since you previously configured port forwarding for the localhost port to the Kubernetes environment, you\u2019ll be able to interact with the Prefect API running in Kubernetes when using local Prefect CLI commands.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/overview/#adding-database-migrations","title":"Adding Database Migrations","text":"

    To make changes to a table, first update the SQLAlchemy model in src/prefect/server/database/orm_models.py. For example, if you wanted to add a new column to the flow_run table, you would add a new column to the FlowRun model:

    # src/prefect/server/database/orm_models.py\n\n@declarative_mixin\nclass ORMFlowRun(ORMRun):\n    \"\"\"SQLAlchemy model of a flow run.\"\"\"\n    ...\n    new_column = Column(String, nullable=True) # <-- add this line\n

    Next, you will need to generate new migration files. You must generate a new migration file for each database type. Migrations will be generated for whatever database type PREFECT_API_DATABASE_CONNECTION_URL is set to. See here for how to set the database connection URL for each database type.

    To generate a new migration file, run the following command:

    prefect server database revision --autogenerate -m \"<migration name>\"\n

    Try to make your migration name brief but descriptive. For example:

    • add_flow_run_new_column
    • add_flow_run_new_column_idx
    • rename_flow_run_old_column_to_new_column

    The --autogenerate flag will automatically generate a migration file based on the changes to the models.

    Always inspect the output of --autogenerate

    --autogenerate will generate a migration file based on the changes to the models. However, it is not perfect. Be sure to check the file to make sure it only includes the changes you want to make. Additionally, you may need to remove extra statements that were included and not related to your change.

    The new migration can be found in the src/prefect/server/database/migrations/versions/ directory. Each database type has its own subdirectory. For example, the SQLite migrations are stored in src/prefect/server/database/migrations/versions/sqlite/.

    After you have inspected the migration file, you can apply the migration to your database by running the following command:

    prefect server database upgrade -y\n

    Once you have successfully created and applied migrations for all database types, make sure to update MIGRATION-NOTES.md to document your additions.

    ","tags":["open source","contributing","development","standards","migrations"],"boost":2},{"location":"contributing/style/","title":"Code style and practices","text":"

    Generally, we follow the Google Python Style Guide. This document covers sections where we differ or where additional clarification is necessary.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#imports","title":"Imports","text":"

    A brief collection of rules and guidelines for how imports should be handled in this repository.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#imports-in-__init__-files","title":"Imports in __init__ files","text":"

    Leave __init__ files empty unless exposing an interface. If you must expose objects to present a simpler API, please follow these rules.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#exposing-objects-from-submodules","title":"Exposing objects from submodules","text":"

    If importing objects from submodules, the __init__ file should use a relative import. This is required for type checkers to understand the exposed interface.

    # Correct\nfrom .flows import flow\n
    # Wrong\nfrom prefect.flows import flow\n
    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#exposing-submodules","title":"Exposing submodules","text":"

    Generally, submodules should not be imported in the __init__ file. Submodules should only be exposed when the module is designed to be imported and used as a namespaced object.

    For example, we do this for our schema and model modules because it is important to know if you are working with an API schema or database model, both of which may have similar names.

    import prefect.server.schemas as schemas\n\n# The full module is accessible now\nschemas.core.FlowRun\n

    If exposing a submodule, use a relative import as you would when exposing an object.

    # Correct\nfrom . import flows\n
    # Wrong\nimport prefect.flows\n
    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#importing-to-run-side-effects","title":"Importing to run side-effects","text":"

    Another use case for importing submodules is perform global side-effects that occur when they are imported.

    Often, global side-effects on import are a dangerous pattern. Avoid them if feasible.

    We have a couple acceptable use-cases for this currently:

    • To register dispatchable types, e.g. prefect.serializers.
    • To extend a CLI application e.g. prefect.cli.
    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#imports-in-modules","title":"Imports in modules","text":"","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#importing-other-modules","title":"Importing other modules","text":"

    The from syntax should be reserved for importing objects from modules. Modules should not be imported using the from syntax.

    # Correct\nimport prefect.server.schemas  # use with the full name\nimport prefect.server.schemas as schemas  # use the shorter name\n
    # Wrong\nfrom prefect.server import schemas\n

    Unless in an __init__.py file, relative imports should not be used.

    # Correct\nfrom prefect.utilities.foo import bar\n
    # Wrong\nfrom .utilities.foo import bar\n

    Imports dependent on file location should never be used without explicit indication it is relative. This avoids confusion about the source of a module.

    # Correct\nfrom . import test\n
    # Wrong\nimport test\n
    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#resolving-circular-dependencies","title":"Resolving circular dependencies","text":"

    Sometimes, we must defer an import and perform it within a function to avoid a circular dependency.

    ## This function in `settings.py` requires a method from the global `context` but the context\n## uses settings\ndef from_context():\n    from prefect.context import get_profile_context\n\n    ...\n

    Attempt to avoid circular dependencies. This often reveals overentanglement in the design.

    When performing deferred imports, they should all be placed at the top of the function.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#with-type-annotations","title":"With type annotations","text":"

    If you are just using the imported object for a type signature, you should use the TYPE_CHECKING flag.

    # Correct\nfrom typing import TYPE_CHECKING\n\nif TYPE_CHECKING:\n    from prefect.server.schemas.states import State\n\ndef foo(state: \"State\"):\n    pass\n

    Note that usage of the type within the module will need quotes e.g. \"State\" since it is not available at runtime.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#importing-optional-requirements","title":"Importing optional requirements","text":"

    We do not have a best practice for this yet. See the kubernetes, docker, and distributed implementations for now.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#delaying-expensive-imports","title":"Delaying expensive imports","text":"

    Sometimes, imports are slow. We'd like to keep the prefect module import times fast. In these cases, we can lazily import the slow module by deferring import to the relevant function body. For modules that are consumed by many functions, the pattern used for optional requirements may be used instead.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#command-line-interface-cli-output-messages","title":"Command line interface (CLI) output messages","text":"

    Upon executing a command that creates an object, the output message should offer: - A short description of what the command just did. - A bullet point list, rehashing user inputs, if possible. - Next steps, like the next command to run, if applicable. - Other relevant, pre-formatted commands that can be copied and pasted, if applicable. - A new line before the first line and after the last line.

    Output Example:

    $ prefect work-queue create testing\n\nCreated work queue with properties:\n    name - 'abcde'\n    uuid - 940f9828-c820-4148-9526-ea8107082bda\n    tags - None\n    deployment_ids - None\n\nStart an agent to pick up flows from the created work queue:\n    prefect agent start -q 'abcde'\n\nInspect the created work queue:\n    prefect work-queue inspect 'abcde'\n

    Additionally:

    • Wrap generated arguments in apostrophes (') to ensure validity by using suffixing formats with !r.
    • Indent example commands, instead of wrapping in backticks (`).
    • Use placeholders if the example cannot be pre-formatted completely.
    • Capitalize placeholder labels and wrap them in less than (<) and greater than (>) signs.
    • Utilize textwrap.dedent to remove extraneous spacing for strings that are written with triple quotes (\"\"\").

    Placeholder Example:

    Create a work queue with tags:\n    prefect work-queue create '<WORK QUEUE NAME>' -t '<OPTIONAL TAG 1>' -t '<OPTIONAL TAG 2>'\n

    Dedent Example:

    from textwrap import dedent\n...\noutput_msg = dedent(\n    f\"\"\"\n    Created work queue with properties:\n        name - {name!r}\n        uuid - {result}\n        tags - {tags or None}\n        deployment_ids - {deployment_ids or None}\n\n    Start an agent to pick up flows from the created work queue:\n        prefect agent start -q {name!r}\n\n    Inspect the created work queue:\n        prefect work-queue inspect {name!r}\n    \"\"\"\n)\n

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/style/#api-versioning","title":"API Versioning","text":"

    The Prefect client can be run separately from the Prefect orchestration server and communicate entirely via an API. Among other things, the Prefect client includes anything that runs task or flow code, (e.g. agents, and the Python client) or any consumer of Prefect metadata, (e.g. the Prefect UI, and CLI). The Prefect server stores this metadata and serves it via the REST API.

    Sometimes, we make breaking changes to the API (for good reasons). In order to check that a Prefect client is compatible with the API it's making requests to, every API call the client makes includes a three-component API_VERSION header with major, minor, and patch versions.

    For example, a request with the X-PREFECT-API-VERSION=3.2.1 header has a major version of 3, minor version 2, and patch version 1.

    This version header can be changed by modifying the API_VERSION constant in prefect.server.api.server.

    When making a breaking change to the API, we should consider if the change might be backwards compatible for clients, meaning that the previous version of the client would still be able to make calls against the updated version of the server code. This might happen if the changes are purely additive: such as adding a non-critical API route. In these cases, we should make sure to bump the patch version.

    In almost all other cases, we should bump the minor version, which denotes a non-backwards-compatible API change. We have reserved the major version chanes to denote also-backwards compatible change that might be significant in some way, such as a major release milestone.

    ","tags":["standards","code style","coding practices","contributing"],"boost":2},{"location":"contributing/versioning/","title":"Versioning","text":"","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#understanding-version-numbers","title":"Understanding version numbers","text":"

    Versions are composed of three parts: MAJOR.MINOR.PATCH. For example, the version 2.5.0 has a major version of 2, a minor version of 5, and patch version of 0.

    Occasionally, we will add a suffix to the version such as rc, a, or b. These indicate pre-release versions that users can opt-into installing to test functionality before it is ready for release.

    Each release will increase one of the version numbers. If we increase a number other than the patch version, the versions to the right of it will be reset to zero.

    ","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#prefects-versioning-scheme","title":"Prefect's versioning scheme","text":"

    Prefect will increase the major version when significant and widespread changes are made to the core product. It is very unlikely that the major version will change without extensive warning.

    Prefect will increase the minor version when:

    • Introducing a new concept that changes how Prefect can be used
    • Changing an existing concept in a way that fundamentally alters how it is used
    • Removing a deprecated feature

    Prefect will increase the patch version when:

    • Making enhancements to existing features
    • Fixing behavior in existing features
    • Adding new functionality to existing concepts
    • Updating dependencies
    ","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#breaking-changes-and-deprecation","title":"Breaking changes and deprecation","text":"

    A breaking change means that your code will need to change to use a new version of Prefect. We strive to avoid breaking changes in all releases.

    At times, Prefect will deprecate a feature. This means that a feature has been marked for removal in the future. When you use it, you may see warnings that it will be removed. A feature is deprecated when it will no longer be maintained. Frequently, a deprecated feature will have a new and improved alternative. Deprecated features will be retained for at least 3 minor version increases or 6 months, whichever is longer. We may retain deprecated features longer than this time period.

    Prefect will sometimes include changes to behavior to fix a bug. These changes are not categorized as breaking changes.

    ","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#client-compatibility-with-prefect","title":"Client compatibility with Prefect","text":"

    When running a Prefect server, you are in charge of ensuring the version is compatible with those of the clients that are using the server. Prefect aims to maintain backwards compatibility with old clients for each server release. In contrast, sometimes new clients cannot be used with an old server. The new client may expect the server to support functionality that it does not yet include. For this reason, we recommend that all clients are the same version as the server or older.

    For example, a client on 2.1.0 can be used with a server on 2.5.0. A client on 2.5.0 cannot be used with a server on 2.1.0.

    ","tags":["versioning","semver"],"boost":2},{"location":"contributing/versioning/#client-compatibility-with-cloud","title":"Client compatibility with Cloud","text":"

    Prefect Cloud targets compatibility with all versions of Prefect clients. If you encounter a compatibility issue, please file a bug report.

    ","tags":["versioning","semver"],"boost":2},{"location":"getting-started/installation/","title":"Installation","text":"

    Prefect requires Python 3.8 or newer.

    We recommend installing Prefect using a Python virtual environment manager such as pipenv, conda, or virtualenv/venv.

    Windows and Linux requirements

    See Windows installation notes and Linux installation notes for details on additional installation requirements and considerations.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#install-prefect","title":"Install Prefect","text":"

    The following sections describe how to install Prefect in your development or execution environment.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#installing-the-latest-version","title":"Installing the latest version","text":"

    Prefect is published as a Python package. To install the latest release or upgrade an existing Prefect install, run the following command in your terminal:

    pip install -U prefect\n

    To install a specific version, specify the version number like this:

    pip install -U \"prefect==2.10.4\"\n

    See available release versions in the Prefect Release Notes.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#installing-the-bleeding-edge","title":"Installing the bleeding edge","text":"

    If you'd like to test with the most up-to-date code, you can install directly off the main branch on GitHub:

    pip install -U git+https://github.com/PrefectHQ/prefect\n

    The main branch may not be stable

    Please be aware that this method installs unreleased code and may not be stable.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#installing-for-development","title":"Installing for development","text":"

    If you'd like to install a version of Prefect for development:

    1. Clone the Prefect repository.
    2. Install an editable version of the Python package with pip install -e.
    3. Install pre-commit hooks.
    $ git clone https://github.com/PrefectHQ/prefect.git\n$ cd prefect\n$ pip install -e \".[dev]\"\n$ pre-commit install\n

    See our Contributing guide for more details about standards and practices for contributing to Prefect.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#checking-your-installation","title":"Checking your installation","text":"

    To confirm that Prefect was installed correctly, run the command prefect version to print the version and environment details to your console.

    $ prefect version\n\nVersion:             2.10.21\nAPI version:         0.8.4\nPython version:      3.10.12\nGit commit:          da816542\nBuilt:               Thu, Jul 13, 2023 2:05 PM\nOS/Arch:             darwin/arm64\nProfile:              local\nServer type:         ephemeral\nServer:\n  Database:          sqlite\n  SQLite version:    3.42.0\n
    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#windows-installation-notes","title":"Windows installation notes","text":"

    You can install and run Prefect via Windows PowerShell, the Windows Command Prompt, or conda. After installation, you may need to manually add the Python local packages Scripts folder to your Path environment variable.

    The Scripts folder path looks something like this (the username and Python version may be different on your system):

    C:\\Users\\MyUserNameHere\\AppData\\Local\\Packages\\PythonSoftwareFoundation.Python.3.11_qbz5n2kfra8p0\\LocalCache\\local-packages\\Python311\\Scripts\n

    Watch the pip install output messages for the Scripts folder path on your system.

    If you're using Windows Subsystem for Linux (WSL), see Linux installation notes.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#linux-installation-notes","title":"Linux installation notes","text":"

    Linux is a popular operating system for running Prefect. You can use Prefect Cloud as your API server, or host your own Prefect server backed by PostgreSQL.

    For development, you can use SQLite 2.24 or newer as your database. Note that certain Linux versions of SQLite can be problematic. Compatible versions include Ubuntu 22.04 LTS and Ubuntu 20.04 LTS.

    Alternatively, you can install SQLite on Red Hat Enterprise Linux (RHEL) or use the conda virtual environment manager and configure a compatible SQLite version.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#using-a-self-signed-ssl-certificate","title":"Using a self-signed SSL certificate","text":"

    If you're using a self-signed SSL certificate, you need to configure your environment to trust the certificate. You can add the certificate to your system bundle and pointing your tools to use that bundle by configuring the SSL_CERT_FILE environment variable.

    If the certificate is not part of your system bundle, you can set the PREFECT_API_TLS_INSECURE_SKIP_VERIFY to True to disable certificate verification altogether.

    Note: Disabling certificate validation is insecure and only suggested as an option for testing!

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#proxies","title":"Proxies","text":"

    Prefect supports communicating via proxies through environment variables. Simply set HTTPS_PROXY and SSL_CERT_FILE in your environment, and the underlying network libraries will route Prefect\u2019s requests appropriately. Read more about using Prefect Cloud with proxies here.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#external-requirements","title":"External requirements","text":"","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#sqlite","title":"SQLite","text":"

    You can use Prefect Cloud as your API server, or host your own Prefect server backed by PostgreSQL.

    By default, a local Prefect server instance uses SQLite as the backing database. SQLite is not packaged with the Prefect installation. Most systems will already have SQLite installed, because it is typically bundled as a part of Python. Prefect requires SQLite version 3.24.0 or later.

    You can check your SQLite version by executing the following command in a terminal:

    $ sqlite3 --version\n

    Or use the Prefect CLI command prefect version, which prints version and environment details to your console, including the server database and version. For example:

    $ prefect version\nVersion:             2.10.21\nAPI version:         0.8.4\nPython version:      3.10.12\nGit commit:          a46cbebb\nBuilt:               Sat, Jul 15, 2023 7:59 AM\nOS/Arch:             darwin/arm64\nProfile:              default\nServer type:         cloud\n
    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#install-sqlite-on-rhel","title":"Install SQLite on RHEL","text":"

    The following steps are needed to install an appropriate version of SQLite on Red Hat Enterprise Linux (RHEL). Note that some RHEL instances have no C compiler, so you may need to check for and install gcc first:

    yum install gcc\n

    Download and extract the tarball for SQLite.

    wget https://www.sqlite.org/2022/sqlite-autoconf-3390200.tar.gz\ntar -xzf sqlite-autoconf-3390200.tar.gz\n

    Move to the extracted SQLite directory, then build and install SQLite.

    cd sqlite-autoconf-3390200/\n./configure\nmake\nmake install\n

    Add LD_LIBRARY_PATH to your profile.

    echo 'export LD_LIBRARY_PATH=\"/usr/local/lib\"' >> /etc/profile\n

    Restart your shell to register these changes.

    Now you can install Prefect using pip.

    pip3 install prefect\n
    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#using-prefect-in-an-environment-with-http-proxies","title":"Using Prefect in an environment with HTTP proxies","text":"

    If you are using Prefect Cloud or hosting your own Prefect server instance, the Prefect library will connect to the API via any proxies you have listed in the HTTP_PROXY, HTTPS_PROXY, or ALL_PROXY environment variables. You may also use the NO_PROXY environment variable to specify which hosts should not be sent through the proxy.

    For more information about these environment variables, see the cURL documentation.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/installation/#next-steps","title":"Next steps","text":"

    Now that you have Prefect installed and your environment configured, you may want to check out the Tutorial to get more familiar with Prefect.

    ","tags":["installation","pip install","development","Linux","Windows","SQLite","upgrading"],"boost":2},{"location":"getting-started/quickstart/","title":"Quickstart","text":"

    Prefect is an orchestration and observability platform that empowers developers to build and scale resilient code quickly, turning their python scripts into resilient, recurring workflows.

    In this quickstart, you'll see how you can schedule your code on remote infrastructure and observe the state of your workflows. With Prefect, you can go from a Python script to a production-ready workflow that runs remotely in minutes.

    Let's get started!

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#setup","title":"Setup","text":"

    Here's a basic script that fetches statistics about the main Prefect GitHub repository.

    import httpx\n\ndef get_repo_info():\n    url = \"https://api.github.com/repos/PrefectHQ/prefect\"\n    response = httpx.get(url)\n    repo = response.json()\n    print(\"PrefectHQ/prefect repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n\nif __name__ == \"__main__\":\n    get_repo_info()\n

    How can we make this script schedulable, observable, resilient, and capable of running anywhere?

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-1-install-prefect","title":"Step 1: Install Prefect","text":"
    pip install -U prefect\n

    See the install guide for more detailed installation instructions, if needed.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-2-connect-to-prefects-api","title":"Step 2: Connect to Prefect's API","text":"

    Much of Prefect's functionality is backed by an API. Sign up for a forever free Prefect Cloud account or accept your organization's invite to join their Prefect Cloud account.

    1. Create a new account or sign in at https://app.prefect.cloud/.
    2. Use the prefect cloud login CLI command to log in to Prefect Cloud from your environment.
    prefect cloud login\n

    Choose Log in with a web browser and click the Authorize button in the browser window that opens.

    Self-hosted Prefect server instance

    If you would like to host a Prefect server instance on your own infrastructure, see the tutorial and select the \"Self-hosted\" tab. Note that you will need to both host your own server and run your flows on your own infrastructure.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-3-turn-your-function-into-a-prefect-flow","title":"Step 3: Turn your function into a Prefect flow","text":"

    The fastest way to get started with Prefect is to add a @flow decorator to your Python function. Flows are the core observable, deployable units in Prefect and are the primary entrypoint to orchestrated work.

    my_gh_workflow.py
    import httpx\nfrom prefect import flow, task\n\n\n@task(retries=2)\ndef get_repo_info(repo_owner: str, repo_name: str):\n    \"\"\"Get info about a repo - will retry twice after failing\"\"\"\n    url = f\"https://api.github.com/repos/{repo_owner}/{repo_name}\"\n    api_response = httpx.get(url)\n    api_response.raise_for_status()\n    repo_info = api_response.json()\n    return repo_info\n\n\n@task\ndef get_contributors(repo_info: dict):\n    \"\"\"Get contributors for a repo\"\"\"\n    contributors_url = repo_info[\"contributors_url\"]\n    response = httpx.get(contributors_url)\n    response.raise_for_status()\n    contributors = response.json()\n    return contributors\n\n\n@flow(log_prints=True)\ndef repo_info(repo_owner: str = \"PrefectHQ\", repo_name: str = \"prefect\"):\n    \"\"\"\n    Given a GitHub repository, logs the number of stargazers\n    and contributors for that repo.\n    \"\"\"\n    repo_info = get_repo_info(repo_owner, repo_name)\n    print(f\"Stars \ud83c\udf20 : {repo_info['stargazers_count']}\")\n\n    contributors = get_contributors(repo_info)\n    print(f\"Number of contributors \ud83d\udc77: {len(contributors)}\")\n\n\nif __name__ == \"__main__\":\n    repo_info()\n

    Note that we added a log_prints=True argument to the @flow decorator so that print statements within the flow-decorated function will be logged. Also note that our flow calls two tasks, which are defined by the task decorator. Tasks are the smallest unit of observed and orchestrated work in Prefect.

    python my_gh_workflow.py\n

    Now when we run this script, Prefect will automatically track the state of the flow run and log the output where we can see it in the UI and CLI.

    14:28:31.099 | INFO    | prefect.engine - Created flow run 'energetic-panther' for flow 'repo-info'\n14:28:31.100 | INFO    | Flow run 'energetic-panther' - View at https://app.prefect.cloud/account/123/workspace/abc/flow-runs/flow-run/xyz\n14:28:32.178 | INFO    | Flow run 'energetic-panther' - Created task run 'get_repo_info-0' for task 'get_repo_info'\n14:28:32.179 | INFO    | Flow run 'energetic-panther' - Executing 'get_repo_info-0' immediately...\n14:28:32.584 | INFO    | Task run 'get_repo_info-0' - Finished in state Completed()\n14:28:32.599 | INFO    | Flow run 'energetic-panther' - Stars \ud83c\udf20 : 13609\n14:28:32.682 | INFO    | Flow run 'energetic-panther' - Created task run 'get_contributors-0' for task 'get_contributors'\n14:28:32.682 | INFO    | Flow run 'energetic-panther' - Executing 'get_contributors-0' immediately...\n14:28:33.118 | INFO    | Task run 'get_contributors-0' - Finished in state Completed()\n14:28:33.134 | INFO    | Flow run 'energetic-panther' - Number of contributors \ud83d\udc77: 30\n14:28:33.255 | INFO    | Flow run 'energetic-panther' - Finished in state Completed('All states completed.')\n

    You should see similar output in your terminal, with your own randomly generated flow run name and your own Prefect Cloud account URL.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-4-choose-a-remote-infrastructure-location","title":"Step 4: Choose a remote infrastructure location","text":"

    Let's get this workflow running on infrastructure other than your local machine! We can tell Prefect where we want to run our workflow by creating a work pool.

    We can have Prefect Cloud run our flow code for us with a Prefect Managed work pool.

    Let's create a Prefect Managed work pool so that Prefect can run our flows for us. We can create a work pool in the UI or from the CLI. Let's use the CLI:

    prefect work-pool create my-managed-pool --type prefect:managed\n

    You should see a message in the CLI that your work pool was created. Feel free to check out your new work pool on the Work Pools page in the UI.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#step-4-make-your-code-schedulable","title":"Step 4: Make your code schedulable","text":"

    We have a flow function and we have a work pool where we can run our flow remotely. Let's package both of these things, along with the location for where to find our flow code, into a deployment so that we can schedule our workflow to run remotely.

    Deployments elevate flows to remotely configurable entities that have their own API.

    Let's make a script to build a deployment with the name my-first-deployment and set it to run on a schedule.

    create_deployment.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=\"https://github.com/discdiver/demos.git\",\n        entrypoint=\"my_gh_workflow.py:repo_info\",\n    ).deploy(\n        name=\"my-first-deployment\",\n        work_pool_name=\"my-managed-pool\",\n        cron=\"0 1 * * *\",\n    )\n

    Run the script to create the deployment on the Prefect Cloud server. Note that the cron argument will schedule the deployment to run at 1am every day.

    python create_deployment.py\n

    You should see a message that your deployment was created, similar to the one below.

    Successfully created/updated all deployments!\n\n                     Deployments                     \n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                          \u2503 Status  \u2503 Details \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 repo-info/my-first-deployment \u2502 applied \u2502         \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n\nTo schedule a run for this deployment, use the following command:\n\n        $ prefect deployment run 'repo-info/my-first-deployment'\n\n\nYou can also run your flow via the Prefect UI: <https://app.prefect.cloud/account/abc/workspace/123/deployments/deployment/xyz>\n

    Head to the Deployments page of the UI to check it out.

    Code storage options

    You can store your flow code in nearly any location. You just need to tell Prefect where to find it. In this example, we use a GitHub repository, but you could bake your code into a Docker image or store it in cloud provider storage. Read more here.

    Push your code to GitHub

    In the example above, we use an existing GitHub repository. If you make changes to the flow code, you will need to push those changes to your own GitHub account and update the source argument to point to your repository.

    You can trigger a manual run of this deployment by either clicking the Run button in the top right of the deployment page in the UI, or by running the following CLI command in your terminal:

    prefect deployment run 'repo-info/my-first-deployment'\n

    The deployment is configured to run on a Prefect Managed work pool, so Prefect will automatically spin up the infrastructure to run this flow. It may take a minute to set up the Docker image in which the flow will run.

    After a minute or so, you should see the flow run graph and logs on the Flow Run page in the UI.

    Remove the schedule

    Click the Remove button in the top right of the Deployment page so that the workflow is no longer scheduled to run once per day.

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"getting-started/quickstart/#next-steps","title":"Next steps","text":"

    You've seen how to move from a Python script to a scheduled, observable, remotely orchestrated workflow with Prefect.

    To learn how to run flows on your own infrastructure, see how to customize the Docker image where your flow runs, and see how to gain lots of orchestration and observation benefits check out the tutorial.

    Need help?

    Get your questions answered by a Prefect Product Advocate! Book a Meeting

    Happy building!

    ","tags":["getting started","quickstart","overview"],"boost":2},{"location":"guides/","title":"How-to Guides","text":"

    This section of the documentation contains how-to guides for common workflows and use cases.

    ","tags":["guides","how to"],"boost":2},{"location":"guides/#development","title":"Development","text":"Title Description Hosting Host your own Prefect server instance. Profiles & Settings Configure Prefect and save your settings. Testing Easily test your workflows. Global Concurrency Limits Limit flow runs. Runtime Context Enable a flow to access metadata about itself and its context when it runs. Variables Store and retrieve configuration data. Prefect Client Use PrefectClient to interact with the API server. Human-in-the-Loop Workflows: Create human-in-the-loop workflows by pausing flow runs for input. Automations Configure actions that Prefect executes automatically based on trigger conditions. Webhooks Receive, observe, and react to events from other systems. Terraform Provider Use the Terraform Provider for Prefect Cloud for infrastructure as code. CI/CD Use CI/CD with Prefect. Prefect Recipes Common, extensible examples for setting up Prefect.","tags":["guides","how to"],"boost":2},{"location":"guides/#execution","title":"Execution","text":"Title Description Docker Deploy flows with Docker containers. State Change Hooks Execute code in response to state changes. Dask and Ray Scale your flows with parallel computing frameworks. Read and Write Data Read and write data to and from cloud provider storage. Big Data Handle large data with Prefect. Logging Configure Prefect's logger and aggregate logs from other tools. Troubleshooting Identify and resolve common issues with Prefect. Managed Execution Let prefect run your code.","tags":["guides","how to"],"boost":2},{"location":"guides/#work-pools","title":"Work Pools","text":"Title Description Deploying Flows to Work Pools and Workers Learn how to run you code with dynamic infrastructure. Upgrade from Agents to Workers Why and how to upgrade from agents to workers. Storage Store your code for deployed flows. Kubernetes Deploy flows on Kubernetes. Serverless Push Work Pools Run flows on serverless infrastructure without a worker. Serverless Work Pools with Workers Run flows on serverless infrastructure with a worker. Automatic infrastructure provisioning Automatically provision cloud infrastructure. Daemonize Processes Set up a systemd service to run a Prefect worker or .serve process. Custom Workers Develop your own worker type.

    Need help?

    Get your questions answered by a Prefect Product Advocate! Book a Meeting

    ","tags":["guides","how to"],"boost":2},{"location":"guides/automations/","title":"Using Automations for Dynamic Responses","text":"

    From the Automations concept page, we saw what an automation can do and how to configure one within the UI.

    In this guide, we will showcase the following common use cases:

    • Create a simple notification automation in just a few UI clicks
    • Build upon an event based automation
    • Combine into a multi-layered responsive deployment pattern

    Available only on Prefect Cloud

    Automations are a Prefect Cloud feature.\n
    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/automations/#prerequisites","title":"Prerequisites","text":"

    Please have the following before exploring the guide:

    • Python installed
    • Prefect installed (follow the installation guide)
    • Authenticated to a Prefect Cloud workspace
    • A work pool set up to handle the deployments
    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/automations/#creating-the-example-script","title":"Creating the example script","text":"

    Automations allow you to take actions in response to triggering events recorded by Prefect.

    For example, let's try to grab data from an API and send a notification based on the end state.

    We can start by pulling hypothetical user data from an endpoint and then performing data cleaning and transformations.

    Let's create a simple extract method, that pulls the data from a random user data generator endpoint.

    from prefect import flow, task, get_run_logger\nimport requests\nimport json\n\n@task\ndef fetch(url: str):\n    logger = get_run_logger()\n    response = requests.get(url)\n    raw_data = response.json()\n    logger.info(f\"Raw response: {raw_data}\")\n    return raw_data\n\n@task\ndef clean(raw_data: dict):\n    print(raw_data.get('results')[0])\n    results = raw_data.get('results')[0]\n    logger = get_run_logger()\n    logger.info(f\"Cleaned results: {results}\")\n    return results['name']\n\n@flow\ndef build_names(num: int = 10):\n    df = []\n    url = \"https://randomuser.me/api/\"\n    logger = get_run_logger()\n    copy = num\n    while num != 0:\n        raw_data = fetch(url)\n        df.append(clean(raw_data))\n        num -= 1\n    logger.info(f\"Built {copy} names: {df}\")\n    return df\n\nif __name__ == \"__main__\":\n    list_of_names = build_names()\n

    The data cleaning workflow has visibility into each step, and we are sending a list of names to our next step of our pipeline.

    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/automations/#create-notification-block-within-the-ui","title":"Create notification block within the UI","text":"

    Now let's try to send a notification based off a completed state outcome. We can configure a notification to be sent so that we know when to look into our workflow logic.

    1. Prior to creating the automation, let's confirm the notification location. We have to create a notification block to help define where the notification will be sent.

    2. Let's navigate to the blocks page on the UI, and click into creating an email notification block.

    3. Now that we created a notification block, we can go to the automations page to create our first automation.

    4. Next we try to find the trigger type, in this case let's use a flow completion.

    5. Finally, let's create the actions that will be done once the triggered is hit. In this case, let's create a notification to be sent out to showcase the completion.

    6. Now the automation is ready to be triggered from a flow run completion. Let's run the file locally and see that the notification is sent to our inbox after the completion. It may take a few minutes for the notification to arrive.

    No deployment created

    Keep in mind, we did not need to create a deployment to trigger our automation, where a state outcome of a local flow run helped trigger this notification block. We are not required to create a deployment to trigger a notification.

    Now that you've seen how to create an email notification from a flow run completion, let's see how we can kick off a deployment run in response to an event.

    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/automations/#event-based-deployment-automation","title":"Event-based deployment automation","text":"

    We can create an automation that can kick off a deployment instead of a notification. Let's explore how we can programmatically create this automation. We will take advantage of Prefect's REST API to help create this automation.

    See the REST API documentation as a reference for interacting with the Prefect Cloud automation endpoints.

    Let's create a deployment where we can kick off some work based on how long a flow is running. For example, if the build_names flow is taking too long to execute, we can kick off a deployment of the with the same build_names flow, but replace the count value with a lower number - to speed up completion. You can create a deployment with a prefect.yaml file or a Python file that uses flow.deploy.

    prefect.yaml.deploy

    Create a prefect.yaml file like this one for our flow build_names:

      # Welcome to your prefect.yaml file! You can use this file for storing and managing\n  # configuration for deploying your flows. We recommend committing this file to source\n  # control along with your flow code.\n\n  # Generic metadata about this project\n  name: automations-guide\n  prefect-version: 2.13.1\n\n  # build section allows you to manage and build docker images\n  build: null\n\n  # push section allows you to manage if and how this project is uploaded to remote locations\n  push: null\n\n  # pull section allows you to provide instructions for cloning this project in remote locations\n  pull:\n  - prefect.deployments.steps.set_working_directory:\n      directory: /Users/src/prefect/Playground/automations-guide\n\n  # the deployments section allows you to provide configuration for deploying flows\n  deployments:\n  - name: deploy-build-names\n    version: null\n    tags: []\n    description: null\n    entrypoint: test-automations.py:build_names\n    parameters: {}\n    work_pool:\n      name: tutorial-process-pool\n      work_queue_name: null\n      job_variables: {}\n    schedule: null\n

    To follow a more python based approach to create a deployment, you can use flow.deploy as in the example below.

    # .deploy only needs a name, valid work pool \n# and a reference to where the flow code exists\n\nif __name__ == \"__main__\":\nbuild_names.deploy(\n    name=\"deploy-build-names\",\n    work_pool_name=\"tutorial-process-pool\"\n    image=\"my_registry/my_image:my_image_tag\",\n)\n

    Now let's grab our deployment_id from this deployment, and embed it in our automation. There are many ways to obtain the deployment_id, but the CLI is a quick way to see all of your deployment ids.

    Find deployment_id from the CLI

    The quickest way to see the ID's associated with your deployment would be running prefect deployment ls in an authenticated command prompt, and you will be able to see the id's associated with all of your deployments

    prefect deployment ls\n                                          Deployments                                           \n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                                                  \u2503 ID                                   \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 Extract islands/island-schedule                       \u2502 d9d7289c-7a41-436d-8313-80a044e61532 \u2502\n\u2502 build-names/deploy-build-names                        \u2502 8b10a65e-89ef-4c19-9065-eec5c50242f4 \u2502\n\u2502 ride-duration-prediction-backfill/backfill-deployment \u2502 76dc6581-1773-45c5-a291-7f864d064c57 \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n
    We can create an automation via a POST call, where we can programmatically create the automation. Ensure you have your api_key, account_id, and workspace_id.

    def create_event_driven_automation():\n    api_url = f\"https://api.prefect.cloud/api/accounts/{account_id}/workspaces/{workspace_id}/automations/\"\n    data = {\n    \"name\": \"Event Driven Redeploy\",\n    \"description\": \"Programmatically created an automation to redeploy a flow based on an event\",\n    \"enabled\": \"true\",\n    \"trigger\": {\n    \"after\": [\n        \"string\"\n    ],\n    \"expect\": [\n        \"prefect.flow-run.Running\"\n    ],\n    \"for_each\": [\n        \"prefect.resource.id\"\n    ],\n    \"posture\": \"Proactive\",\n    \"threshold\": 30,\n    \"within\": 0\n    },\n    \"actions\": [\n    {\n        \"type\": \"run-deployment\",\n        \"source\": \"selected\",\n        \"deployment_id\": \"YOUR-DEPLOYMENT-ID\", \n        \"parameters\": \"10\"\n    }\n    ],\n    \"owner_resource\": \"string\"\n        }\n\n    headers = {\"Authorization\": f\"Bearer {PREFECT_API_KEY}\"}\n    response = requests.post(api_url, headers=headers, json=data)\n\n    print(response.json())\n    return response.json()\n

    After running this function, you will see within the UI the changes that came from the post request. Keep in mind, the context will be \"custom\" on UI.

    Let's run the underlying flow and see the deployment get kicked off after 30 seconds elapsed. This will result in a new flow run of build_names, and we are able to see this new deployment get initiated with the custom parameters we outlined above.

    In a few quick changes, we are able to programmatically create an automation that deploys workflows with custom parameters.

    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/automations/#using-an-underlying-yaml-file","title":"Using an underlying .yaml file","text":"

    We can extend this idea one step further by utilizing our own .yaml version of the automation, and registering that file with our UI. This simplifies the requirements of the automation by declaring it in its own .yaml file, and then registering that .yaml with the API.

    Let's first start with creating the .yaml file that will house the automation requirements. Here is how it would look like:

    name: Cancel long running flows\ndescription: Cancel any flow run after an hour of execution\ntrigger:\n  match:\n    \"prefect.resource.id\": \"prefect.flow-run.*\"\n  match_related: {}\n  after:\n    - \"prefect.flow-run.Failed\"\n  expect:\n    - \"prefect.flow-run.*\"\n  for_each:\n    - \"prefect.resource.id\"\n  posture: \"Proactive\"\n  threshold: 1\n  within: 30\nactions:\n  - type: \"cancel-flow-run\"\n

    We can then have a helper function that applies this YAML file with the REST API function.

    import yaml\n\nfrom utils import post, put\n\ndef create_or_update_automation(path: str = \"automation.yaml\"):\n    \"\"\"Create or update an automation from a local YAML file\"\"\"\n    # Load the definition\n    with open(path, \"r\") as fh:\n        payload = yaml.safe_load(fh)\n\n    # Find existing automations by name\n    automations = post(\"/automations/filter\")\n    existing_automation = [a[\"id\"] for a in automations if a[\"name\"] == payload[\"name\"]]\n    automation_exists = len(existing_automation) > 0\n\n    # Create or update the automation\n    if automation_exists:\n        print(f\"Automation '{payload['name']}' already exists and will be updated\")\n        put(f\"/automations/{existing_automation[0]}\", payload=payload)\n    else:\n        print(f\"Creating automation '{payload['name']}'\")\n        post(\"/automations/\", payload=payload)\n\nif __name__ == \"__main__\":\n    create_or_update_automation()\n

    You can find a complete repo with these APIs examples in this GitHub repository.

    In this example, we managed to create the automation by registering the .yaml file with a helper function. This offers another experience when trying to create an automation.

    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/automations/#custom-webhook-kicking-off-an-automation","title":"Custom webhook kicking off an automation","text":"

    We can use webhooks to expose the events API which allows us to extend the functionality of deployments and ways to respond to changes in our workflow through a few easy steps.

    By exposing a webhook endpoint, we can kick off workflows that can trigger deployments - all from a simple event created from an HTTP request.

    Lets create a webhook within the UI. Here is the webhook we can use to create these dynamic events.

    {\n    \"event\": \"model-update\",\n    \"resource\": {\n        \"prefect.resource.id\": \"product.models.{{ body.model_id}}\",\n        \"prefect.resource.name\": \"{{ body.friendly_name }}\",\n        \"run_count\": \"{{body.run_count}}\"\n    }\n}\n
    From a simple input, we can easily create an exposed webhook endpoint.

    Each webhook will correspond to a custom event created, where you can react to it downstream with a separate deployment or automation.

    For example, we can create a curl request that sends the endpoint information such as a run count for our deployment.

    curl -X POST https://api.prefect.cloud/hooks/34iV2SFke3mVa6y5Y-YUoA -d \"model_id=adhoc\" -d \"run_count=10\" -d \"friendly_name=test-user-input\"\n
    From here, we can make a webhook that is connected to pulling in parameters on the curl command, and then it kicks off a deployment that uses these pulled parameters.

    Let us go into the event feed, and we can automate straight from this event.

    This allows us to create automations that respond to these webhook events. From a few clicks in the UI, we are able to associate an external process with the Prefect events API, that can enable us to trigger downstream deployments.

    In the next section, we will explore event triggers that automate the kickoff of a deployment run.

    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/automations/#using-triggers","title":"Using triggers","text":"

    Let's take this idea one step further, by creating a deployment that will be triggered when a flow run takes longer than expected. We can take advantage of Prefect's Marvin library that will use an LLM to classify our data. Marvin is great at embedding data science and data analysis applications within your pre-existing data engineering workflows. In this case, we can use Marvin'd AI functions to help make our dataset more information rich.

    Install Marvin with pip install marvin and set you OpenAI API key as shown here

    We can add a trigger to run a deployment in response to a specific event.

    Let's create an example with Marvin's AI functions. We will take in a pandas DataFrame and use the AI function to analyze it.

    Here is an example of pulling in that data and classifying using Marvin AI. We can help create dummy data based on classifications we have already created.

    from marvin import ai_classifier\nfrom enum import Enum\nimport pandas as pd\n\n@ai_fn\ndef generate_synthetic_user_data(build_of_names: list[dict]) -> list:\n    \"\"\"\n    Generate additional data for userID (numerical values with 6 digits), location, and timestamp as separate columns and append the data onto 'build_of_names'. Make userID the first column\n    \"\"\"\n\n@flow\ndef create_fake_user_dataset(df):\n  artifact_df = generate_synthetic_user_data(df)\n  print(artifact_df)\n\n  create_table_artifact(\n      key=\"fake-user-data\",\n      table=artifact_df,\n      description= \"Dataset that is comprised of a mix of autogenerated data based on user data\"\n  )\n\nif __name__ == \"__main__\":\n    create_fake_artifact()  \n

    Let's kick off a deployment with a trigger defined in a prefect.yaml file. Let's specify what we want to trigger when the event stays in a running state for longer than 30 seconds.

    # Welcome to your prefect.yaml file! You can use this file for storing and managing\n# configuration for deploying your flows. We recommend committing this file to source\n# control along with your flow code.\n\n# Generic metadata about this project\nname: automations-guide\nprefect-version: 2.13.1\n\n# build section allows you to manage and build docker images\nbuild: null\n\n# push section allows you to manage if and how this project is uploaded to remote locations\npush: null\n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect.deployments.steps.set_working_directory:\n    directory: /Users/src/prefect/Playground/marvin-extension\n\n# the deployments section allows you to provide configuration for deploying flows\ndeployments:\n- name: create-fake-user-dataset\n  triggers:\n    - enabled: true\n      match:\n        prefect.resource.id: \"prefect.flow-run.*\"\n      after: \"prefect.flow-run.Running\",\n      expect: [],\n      for_each: [\"prefect.resource.id\"],\n      parameters:\n        param_1: 10\n      posture: \"Proactive\"\n  version: null\n  tags: []\n  description: null\n  entrypoint: marvin-extension.py:create_fake_user_dataset\n  parameters: {}\n  work_pool:\n    name: tutorial-process-pool\n    work_queue_name: null\n    job_variables: {}\n  schedule: null\n

    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/automations/#next-steps","title":"Next steps","text":"

    You've seen how to create automations via the UI, REST API, and a triggers defined in a prefect.yaml deployment definition.

    To learn more about events that can act as automation triggers, see the events docs. To learn more about event webhooks in particular, see the webhooks guide

    ","tags":["automations","event-driven","trigger"],"boost":2},{"location":"guides/big-data/","title":"Big data with Prefect","text":"

    In this guide you'll learn tips for working with large amounts of data in Prefect.

    Big data doesn't have a widely accepted, precise definition. In this guide, we'll discuss methods to reduce the processing time or memory utilization of Prefect workflows, without editing your Python code.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#optimizing-your-python-code-with-prefect-for-big-data","title":"Optimizing your Python code with Prefect for big data","text":"

    Depending upon your needs, you may want to optimize your Python code for speed, memory, compute, or disk space.

    Prefect provides several options that we'll explore in this guide:

    1. Remove task introspection with quote to save time running your code.
    2. Write task results to cloud storage such as S3 using a block to save memory.
    3. Save data to disk within a flow rather than using results.
    4. Cache task results to save time and compute.
    5. Compress results written to disk to save space.
    6. Use a task runner for parallelizable operations to save time.
    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#remove-task-introspection","title":"Remove task introspection","text":"

    When a task is called from a flow, each argument is introspected by Prefect, by default. To speed up your flow runs, you can disable this behavior for a task by wrapping the argument using quote.

    To demonstrate, let's use a basic example that extracts and transforms some New York taxi data.

    et_quote.py
    from prefect import task, flow\nfrom prefect.utilities.annotations import quote\nimport pandas as pd\n\n\n@task\ndef extract(url: str):\n    \"\"\"Extract data\"\"\"\n    df_raw = pd.read_parquet(url)\n    print(df_raw.info())\n    return df_raw\n\n\n@task\ndef transform(df: pd.DataFrame):\n    \"\"\"Basic transformation\"\"\"\n    df[\"tip_fraction\"] = df[\"tip_amount\"] / df[\"total_amount\"]\n    print(df.info())\n    return df\n\n\n@flow(log_prints=True)\ndef et(url: str):\n    \"\"\"ET pipeline\"\"\"\n    df_raw = extract(url)\n    df = transform(quote(df_raw))\n\n\nif __name__ == \"__main__\":\n    url = \"https://d37ci6vzurychx.cloudfront.net/trip-data/yellow_tripdata_2023-09.parquet\"\n    et(url)\n

    Introspection can take significant time when the object being passed is a large collection, such as dictionary or DataFrame, where each element needs to be visited. Note that using quote reduces execution time at the expense of disabling task dependency tracking for the wrapped object.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#write-task-results-to-cloud-storage","title":"Write task results to cloud storage","text":"

    By default, the results of task runs are stored in memory in your execution environment. This behavior makes flow runs fast for small data, but can be problematic for large data. You can save memory by writing results to disk. In production, you'll generally want to write results to a cloud provider storage such as AWS S3. Prefect lets you to use a storage block from a Prefect cloud integration library such as prefect-aws to save your configuration information. Learn more about blocks here.

    Install the relevant library, register the block with the server, and create your storage block. Then you can reference the block in your flow like this:

    ...\nfrom prefect_aws.s3 import S3Bucket\n\nmy_s3_block = S3Bucket.load(\"MY_BLOCK_NAME\")\n\n...\n@task(result_storage=my_s3_block)\n

    Now the result of the task will be written to S3, rather than stored in memory.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#save-data-to-disk-within-a-flow","title":"Save data to disk within a flow","text":"

    To save memory and time with big data, you don't need to pass results between tasks at all. Instead, you can write and read data to disk directly in your flow code. Prefect has integration libraries for each of the major cloud providers. Each library contains blocks with methods that make it convenient to read and write data to and from cloud object storage. The moving data guide has step-by-step examples for each cloud provider.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#cache-task-results","title":"Cache task results","text":"

    Caching allows you to avoid re-running tasks when doing so is unnecessary. Caching can save you time and compute. Note that caching requires task result persistence. Caching is discussed in detail in the tasks concept page.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#compress-results-written-to-disk","title":"Compress results written to disk","text":"

    If you're using Prefect's task result persistence, you can save disk space by compressing the results. You just need to specify the result type with compressed/ prefixed like this:

    @task(result_serializer=\"compressed/json\")\n

    Read about compressing results with Prefect for more details. The tradeoff of using compression is that it takes time to compress and decompress the data.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/big-data/#use-a-task-runner-for-parallelizable-operations","title":"Use a task runner for parallelizable operations","text":"

    Prefect's task runners allow you to use the Dask and Ray Python libraries to run tasks in parallel and distributed across multiple machines. This can save you time and compute when operating on large data structures. See the guide to working with Dask and Ray Task Runners for details.

    ","tags":["big data","flow configuration","parallel execution","distributed execution","caching"],"boost":2},{"location":"guides/ci-cd/","title":"CI/CD with Prefect","text":"

    Many organizations deploy Prefect workflows via their CI/CD process. Each organization has their own unique CI/CD setup, but a common pattern is to use CI/CD to manage Prefect deployments. Combining Prefect's deployment features with CI/CD tools enables efficient management of flow code updates, scheduling changes, and container builds. This guide uses GitHub Actions to implement a CI/CD process, but these concepts are generally applicable across many CI/CD tools.

    Note that Prefect's primary ways for creating deployments, a .deploy flow method or a prefect.yaml configuration file, are both designed with building and pushing images to a Docker registry in mind.

    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#getting-started-with-github-actions-and-prefect","title":"Getting started with GitHub Actions and Prefect","text":"

    In this example, you'll write a GitHub Actions workflow that will run each time you push to your repository's main branch. This workflow will build and push a Docker image containing your flow code to Docker Hub, then deploy the flow to Prefect Cloud.

    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#repository-secrets","title":"Repository secrets","text":"

    Your CI/CD process must be able to authenticate with Prefect in order to deploy flows.

    Deploying flows securely and non-interactively in your CI/CD process can be accomplished by saving your PREFECT_API_URL and PREFECT_API_KEY as secrets in your repository's settings so they can be accessed in your CI/CD runner's environment without exposing them in any scripts or configuration files.

    In this scenario, deploying flows involves building and pushing Docker images, so add DOCKER_USERNAME and DOCKER_PASSWORD as secrets to your repository as well.

    You can create secrets for GitHub Actions in your repository under Settings -> Secrets and variables -> Actions -> New repository secret:

    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#writing-a-github-workflow","title":"Writing a GitHub workflow","text":"

    To deploy your flow via GitHub Actions, you'll need a workflow YAML file. GitHub will look for workflow YAML files in the .github/workflows/ directory in the root of your repository. In their simplest form, GitHub workflow files are made up of triggers and jobs.

    The on: trigger is set to run the workflow each time a push occurs on the main branch of the repository.

    The deploy job is comprised of four steps:

    • Checkout clones your repository into the GitHub Actions runner so you can reference files or run scripts from your repository in later steps.
    • Log in to Docker Hub authenticates to DockerHub so your image can be pushed to the Docker registry in your DockerHub account. docker/login-action is an existing GitHub action maintained by Docker. with: passes values into the Action, similar to passing parameters to a function.
    • Setup Python installs your selected version of Python.
    • Prefect Deploy installs the dependencies used in your flow, then deploys your flow. env: makes the PREFECT_API_KEY and PREFECT_API_URL secrets from your repository available as environment variables during this step's execution.

    For reference, the examples below can be found on their respective branches of this repository.

    .deployprefect.yaml
    .\n\u251c\u2500\u2500 .github/\n\u2502   \u2514\u2500\u2500 workflows/\n\u2502       \u2514\u2500\u2500 deploy-prefect-flow.yaml\n\u251c\u2500\u2500 flow.py\n\u2514\u2500\u2500 requirements.txt\n

    flow.py

    from prefect import flow\n\n@flow(log_prints=True)\ndef hello():\n  print(\"Hello!\")\n\nif __name__ == \"__main__\":\n    hello.deploy(\n        name=\"my-deployment\",\n        work_pool_name=\"my-work-pool\",\n        image=\"my_registry/my_image:my_image_tag\",\n    )\n

    .github/workflows/deploy-prefect-flow.yaml

    name: Deploy Prefect flow\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    name: Deploy\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout\n        uses: actions/checkout@v4\n\n      - name: Log in to Docker Hub\n        uses: docker/login-action@v3\n        with:\n          username: ${{ secrets.DOCKER_USERNAME }}\n          password: ${{ secrets.DOCKER_PASSWORD }}\n\n      - name: Setup Python\n        uses: actions/setup-python@v5\n        with:\n          python-version: '3.11'\n\n      - name: Prefect Deploy\n        env:\n          PREFECT_API_KEY: ${{ secrets.PREFECT_API_KEY }}\n          PREFECT_API_URL: ${{ secrets.PREFECT_API_URL }}\n        run: |\n          pip install -r requirements.txt\n          python flow.py\n
    .\n\u251c\u2500\u2500 .github/\n\u2502   \u2514\u2500\u2500 workflows/\n\u2502       \u2514\u2500\u2500 deploy-prefect-flow.yaml\n\u251c\u2500\u2500 flow.py\n\u251c\u2500\u2500 prefect.yaml\n\u2514\u2500\u2500 requirements.txt\n

    flow.py

    from prefect import flow\n\n@flow(log_prints=True)\ndef hello():\n  print(\"Hello!\")\n

    prefect.yaml

    name: cicd-example\nprefect-version: 2.14.11\n\nbuild:\n  - prefect_docker.deployments.steps.build_docker_image:\n      id: build_image\n      requires: prefect-docker>=0.3.1\n      image_name: my_registry/my_image\n      tag: my_image_tag\n      dockerfile: auto\n\npush:\n  - prefect_docker.deployments.steps.push_docker_image:\n      requires: prefect-docker>=0.3.1\n      image_name: \"{{ build_image.image_name }}\"\n      tag: \"{{ build_image.tag }}\"\n\npull: null\n\ndeployments:\n  - name: my-deployment\n    entrypoint: flow.py:hello\n    work_pool:\n      name: my-work-pool\n      work_queue_name: default\n      job_variables:\n        image: \"{{ build-image.image }}\"\n

    .github/workflows/deploy-prefect-flow.yaml

    name: Deploy Prefect flow\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    name: Deploy\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout\n        uses: actions/checkout@v4\n\n      - name: Log in to Docker Hub\n        uses: docker/login-action@v3\n        with:\n          username: ${{ secrets.DOCKER_USERNAME }}\n          password: ${{ secrets.DOCKER_PASSWORD }}\n\n      - name: Setup Python\n        uses: actions/setup-python@v5\n        with:\n          python-version: '3.11'\n\n      - name: Prefect Deploy\n        env:\n          PREFECT_API_KEY: ${{ secrets.PREFECT_API_KEY }}\n          PREFECT_API_URL: ${{ secrets.PREFECT_API_URL }}\n        run: |\n          pip install -r requirements.txt\n          prefect deploy -n my-deployment\n
    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#running-a-github-workflow","title":"Running a GitHub workflow","text":"

    After pushing commits to your repository, GitHub will automatically trigger a run of your workflow. The status of running and completed workflows can be monitored from the Actions tab of your repository.

    You can view the logs from each workflow step as they run. The Prefect Deploy step will include output about your image build and push, and the creation/update of your deployment.

    Successfully built image '***/cicd-example:latest'\n\nSuccessfully pushed image '***/cicd-example:latest'\n\nSuccessfully created/updated all deployments!\n\n                Deployments\n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                \u2503 Status  \u2503 Details \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 hello/my-deployment \u2502 applied \u2502         \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n
    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#prefect-github-actions","title":"Prefect GitHub Actions","text":"

    Prefect provides its own GitHub Actions for authentication and deployment creation. These actions can simplify deploying with CI/CD when using prefect.yaml, especially in cases where a repository contains flows that are used in multiple deployments across multiple Prefect Cloud workspaces.

    Here's an example of integrating these actions into the workflow we created above:

    name: Deploy Prefect flow\n\non:\n  push:\n    branches:\n      - main\n\njobs:\n  deploy:\n    name: Deploy\n    runs-on: ubuntu-latest\n\n    steps:\n      - name: Checkout\n        uses: actions/checkout@v4\n\n      - name: Log in to Docker Hub\n        uses: docker/login-action@v3\n        with:\n          username: ${{ secrets.DOCKER_USERNAME }}\n          password: ${{ secrets.DOCKER_PASSWORD }}\n\n      - name: Setup Python\n        uses: actions/setup-python@v5\n        with:\n          python-version: \"3.11\"\n\n      - name: Prefect Auth\n        uses: PrefectHQ/actions-prefect-auth@v1\n        with:\n          prefect-api-key: ${{ secrets.PREFECT_API_KEY }}\n          prefect-workspace: ${{ secrets.PREFECT_WORKSPACE }}\n\n      - name: Run Prefect Deploy\n        uses: PrefectHQ/actions-prefect-deploy@v3\n        with:\n          deployment-names: my-deployment\n          requirements-file-paths: requirements.txt\n
    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#authenticating-to-other-docker-image-registries","title":"Authenticating to other Docker image registries","text":"

    The docker/login-action GitHub Action supports pushing images to a wide variety of image registries.

    For example, if you are storing Docker images in AWS Elastic Container Registry, you can add your ECR registry URL to the registry key in the with: part of the action and use an AWS_ACCESS_KEY_ID and AWS_SECRET_ACCESS_KEY as your username and password.

    - name: Login to ECR\n  uses: docker/login-action@v3\n  with:\n    registry: <aws-account-number>.dkr.ecr.<region>.amazonaws.com\n    username: ${{ secrets.AWS_ACCESS_KEY_ID }}\n    password: ${{ secrets.AWS_SECRET_ACCESS_KEY }}\n
    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/ci-cd/#other-resources","title":"Other resources","text":"

    Check out the Prefect Cloud Terraform provider if you're using Terraform to manage your infrastructure.

    ","tags":["CI/CD","continuous integration","continuous delivery"],"boost":2},{"location":"guides/creating-human-in-the-loop-workflows/","title":"Creating Human-in-the-Loop Workflows","text":"

    Experimental

    The wait_for_input parameter used in the pause_flow_run or suspend_flow_run functions is an experimental feature. The interface or behavior of this feature may change without warning in future releases.

    If you encounter any issues, please let us know in Slack or with a Github issue.

    When a flow run is paused or suspended, you can receive input from the user. This is useful when you need to ask the user for additional information or feedback before resuming the flow run.

    ","tags":["flow run","pause","suspend","input"],"boost":2},{"location":"guides/creating-human-in-the-loop-workflows/#waiting-for-input","title":"Waiting for input","text":"

    To receive input you must use the wait_for_input parameter in the pause_flow_run or suspend_flow_run functions. This parameter accepts a subclass of prefect.input.RunInput. RunInput is a subclass of pydantic.BaseModel and can be used to define the input that you want to receive:

    from prefect.input import RunInput\n\nclass UserNameInput(RunInput):\n    name: str\n

    In this case we are defining a UserNameInput class that will receive a name string from the user. You can then use this class in the wait_for_input parameter:

    @flow\nasync def greet_user():\n    logger = get_run_logger()\n\n    user_input = await pause_flow_run(\n        wait_for_input=UserNameInput\n    )\n\n    logger.info(f\"Hello, {user_input.name}!\")\n

    When the flow run is paused, the user will be prompted to enter a name. If the user does not enter a name, the flow run will not resume. If the user enters a name, the flow run will resume and the user_input variable will contain the name that the user entered.

    ","tags":["flow run","pause","suspend","input"],"boost":2},{"location":"guides/creating-human-in-the-loop-workflows/#providing-initial-data","title":"Providing initial data","text":"

    You can set default values for fields in your model by using the with_initial_data method. This is useful when you want to provide default values for the fields in your own RunInput subclasses.

    Expanding on the example above, you could default the name field to something anonymous.

    @flow\nasync def greet_user():\n    logger = get_run_logger()\n\n    user_input = await pause_flow_run(\n        wait_for_input=UserNameInput.with_initial_data(name=\"anonymous\")\n    )\n\n    if user_input.name == \"anonymous\":\n        logger.info(\"Hello, stranger!\")\n    else:\n        logger.info(f\"Hello, {user_input.name}!\")\n
    ","tags":["flow run","pause","suspend","input"],"boost":2},{"location":"guides/creating-human-in-the-loop-workflows/#handling-custom-validation","title":"Handling custom validation","text":"

    Prefect uses the fields and type hints on your RunInput subclass to validate the general structure of input your flow run receives, but you might require more complex validation. If you do, you can use Pydantic validators.

    Custom validation runs after the flow run resumes

    Prefect transforms the type annotations in your RunInput class to a JSON schema and use that schema in the UI to do client-side validation. However, custom validation requires running logic defined in your RunInput class. This happens after the flow resumes, so you'll probably want to handle it explicitly in your flow. Continue reading for an example best practice.

    The following is an example RunInput class that uses a custom field validator:

    import pydantic\nfrom prefect.input import RunInput\n\n\nclass ShirtOrder(RunInput):\n    size: Literal[\"small\", \"medium\", \"large\", \"xlarge\"]\n    color: Literal[\"red\", \"green\", \"black\"]\n\n    @pydantic.validator(\"color\")\n    def validate_age(cls, value, values, **kwargs):\n        if value == \"green\" and values[\"size\"] == \"small\":\n            raise ValueError(\"Green is only in-stock for medium, large, and XL sizes.\")\n\n        return value\n

    In the example, we use Pydantic's validator decorator to define a custom validation method for the color field. We can use it in a flow like this:

    import pydantic\nfrom prefect import flow\nfrom prefect.input import RunInput\n\n\nclass ShirtOrder(RunInput):\n    size: Literal[\"small\", \"medium\", \"large\", \"xlarge\"]\n    color: Literal[\"red\", \"green\", \"black\"]\n\n    @pydantic.validator(\"color\")\n    def validate_age(cls, value, values, **kwargs):\n        if value == \"green\" and values[\"size\"] == \"small\":\n            raise ValueError(\"Green is only in-stock for medium, large, and XL sizes.\")\n\n        return value\n\n\n@flow\ndef get_shirt_order():\n    shirt_order = pause_flow_run(wait_for_input=ShirtOrder)\n

    If a user chooses any size and color combination other than small and green, the flow run will resume successfully. However, if the user chooses size small and color green, the flow run will resume, and pause_flow_run will raise a ValidationError exception. This will cause the flow run to fail and log the error.

    However, what if you don't want the flow run to fail? One way to handle this case is to use a while loop and pause again if the ValidationError exception is raised:

    import pydantic\nfrom prefect import flow, get_run_logger\nfrom prefect.input import RunInput\n\n\nclass ShirtOrder(RunInput):\n    size: Literal[\"small\", \"medium\", \"large\", \"xlarge\"]\n    color: Literal[\"red\", \"green\", \"black\"]\n\n    @pydantic.validator(\"color\")\n    def validate_age(cls, value, values, **kwargs):\n        if value == \"green\" and values[\"size\"] == \"small\":\n            raise ValueError(\"Green is only in-stock for medium, large, and XL sizes.\")\n\n        return value\n\n\n@flow\ndef get_shirt_order():\n    logger = get_run_logger()\n    shirt_order = None\n\n    while shirt_order is None:\n        try:\n            shirt_order = pause_flow_run(wait_for_input=ShirtOrder)\n        except pydantic.ValidationError as exc:\n            logger.error(f\"Invalid size and color combination: {exc}\")\n\n    logger.info(f\"Shirt order: {shirt_order.size}, {shirt_order.color}\")\n

    This code will cause the flow run to continually pause until the user enters a valid age.

    As an additional step, you may want to use an automation or notification to alert the user to the error.

    ","tags":["flow run","pause","suspend","input"],"boost":2},{"location":"guides/dask-ray-task-runners/","title":"Dask and Ray Task Runners","text":"

    Task runners provide an execution environment for tasks. In a flow decorator, you can specify a task runner to run the tasks called in that flow.

    The default task runner is the ConcurrentTaskRunner.

    Use .submit to run your tasks asynchronously

    To run tasks asynchronously use the .submit method when you call them. If you call a task as you would normally in Python code it will run synchronously, even if you are calling the task within a flow that uses the ConcurrentTaskRunner, DaskTaskRunner, or RayTaskRunner.

    Many real-world data workflows benefit from true parallel, distributed task execution. For these use cases, the following Prefect-developed task runners for parallel task execution may be installed as Prefect Integrations.

    • DaskTaskRunner runs tasks requiring parallel execution using dask.distributed.
    • RayTaskRunner runs tasks requiring parallel execution using Ray.

    These task runners can spin up a local Dask cluster or Ray instance on the fly, or let you connect with a Dask or Ray environment you've set up separately. Then you can take advantage of massively parallel computing environments.

    Use Dask or Ray in your flows to choose the execution environment that fits your particular needs.

    To show you how they work, let's start small.

    Remote storage

    We recommend configuring remote file storage for task execution with DaskTaskRunner or RayTaskRunner. This ensures tasks executing in Dask or Ray have access to task result storage, particularly when accessing a Dask or Ray instance outside of your execution environment.

    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/dask-ray-task-runners/#configuring-a-task-runner","title":"Configuring a task runner","text":"

    You may have seen this briefly in a previous tutorial, but let's look a bit more closely at how you can configure a specific task runner for a flow.

    Let's start with the SequentialTaskRunner. This task runner runs all tasks synchronously and may be useful when used as a debugging tool in conjunction with async code.

    Let's start with this simple flow. We import the SequentialTaskRunner, specify a task_runner on the flow, and call the tasks with .submit().

    from prefect import flow, task\nfrom prefect.task_runners import SequentialTaskRunner\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n@flow(task_runner=SequentialTaskRunner())\ndef greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n\ngreetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n

    Save this as sequential_flow.py and run it in a terminal. You'll see output similar to the following:

    $ python sequential_flow.py\n16:51:17.967 | INFO    | prefect.engine - Created flow run 'humongous-mink' for flow 'greetings'\n16:51:17.967 | INFO    | Flow run 'humongous-mink' - Starting 'SequentialTaskRunner'; submitted tasks will be run sequentially...\n16:51:18.038 | INFO    | Flow run 'humongous-mink' - Created task run 'say_hello-811087cd-0' for task 'say_hello'\n16:51:18.038 | INFO    | Flow run 'humongous-mink' - Executing 'say_hello-811087cd-0' immediately...\nhello arthur\n16:51:18.060 | INFO    | Task run 'say_hello-811087cd-0' - Finished in state Completed()\n16:51:18.107 | INFO    | Flow run 'humongous-mink' - Created task run 'say_goodbye-261e56a8-0' for task 'say_goodbye'\n16:51:18.107 | INFO    | Flow run 'humongous-mink' - Executing 'say_goodbye-261e56a8-0' immediately...\ngoodbye arthur\n16:51:18.123 | INFO    | Task run 'say_goodbye-261e56a8-0' - Finished in state Completed()\n16:51:18.134 | INFO    | Flow run 'humongous-mink' - Created task run 'say_hello-811087cd-1' for task 'say_hello'\n16:51:18.134 | INFO    | Flow run 'humongous-mink' - Executing 'say_hello-811087cd-1' immediately...\nhello trillian\n16:51:18.150 | INFO    | Task run 'say_hello-811087cd-1' - Finished in state Completed()\n16:51:18.159 | INFO    | Flow run 'humongous-mink' - Created task run 'say_goodbye-261e56a8-1' for task 'say_goodbye'\n16:51:18.159 | INFO    | Flow run 'humongous-mink' - Executing 'say_goodbye-261e56a8-1' immediately...\ngoodbye trillian\n16:51:18.181 | INFO    | Task run 'say_goodbye-261e56a8-1' - Finished in state Completed()\n16:51:18.190 | INFO    | Flow run 'humongous-mink' - Created task run 'say_hello-811087cd-2' for task 'say_hello'\n16:51:18.190 | INFO    | Flow run 'humongous-mink' - Executing 'say_hello-811087cd-2' immediately...\nhello ford\n16:51:18.210 | INFO    | Task run 'say_hello-811087cd-2' - Finished in state Completed()\n16:51:18.219 | INFO    | Flow run 'humongous-mink' - Created task run 'say_goodbye-261e56a8-2' for task 'say_goodbye'\n16:51:18.219 | INFO    | Flow run 'humongous-mink' - Executing 'say_goodbye-261e56a8-2' immediately...\ngoodbye ford\n16:51:18.237 | INFO    | Task run 'say_goodbye-261e56a8-2' - Finished in state Completed()\n16:51:18.246 | INFO    | Flow run 'humongous-mink' - Created task run 'say_hello-811087cd-3' for task 'say_hello'\n16:51:18.246 | INFO    | Flow run 'humongous-mink' - Executing 'say_hello-811087cd-3' immediately...\nhello marvin\n16:51:18.264 | INFO    | Task run 'say_hello-811087cd-3' - Finished in state Completed()\n16:51:18.273 | INFO    | Flow run 'humongous-mink' - Created task run 'say_goodbye-261e56a8-3' for task 'say_goodbye'\n16:51:18.273 | INFO    | Flow run 'humongous-mink' - Executing 'say_goodbye-261e56a8-3' immediately...\ngoodbye marvin\n16:51:18.290 | INFO    | Task run 'say_goodbye-261e56a8-3' - Finished in state Completed()\n16:51:18.321 | INFO    | Flow run 'humongous-mink' - Finished in state Completed('All states completed.')\n

    If we take out the log messages and just look at the printed output of the tasks, you see they're executed in sequential order:

    $ python sequential_flow.py\nhello arthur\ngoodbye arthur\nhello trillian\ngoodbye trillian\nhello ford\ngoodbye ford\nhello marvin\ngoodbye marvin\n
    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/dask-ray-task-runners/#running-parallel-tasks-with-dask","title":"Running parallel tasks with Dask","text":"

    You could argue that this simple flow gains nothing from parallel execution, but let's roll with it so you can see just how simple it is to take advantage of the DaskTaskRunner.

    To configure your flow to use the DaskTaskRunner:

    1. Make sure the prefect-dask collection is installed by running pip install prefect-dask.
    2. In your flow code, import DaskTaskRunner from prefect_dask.task_runners.
    3. Assign it as the task runner when the flow is defined using the task_runner=DaskTaskRunner argument.
    4. Use the .submit method when calling functions.

    This is the same flow as above, with a few minor changes to use DaskTaskRunner where we previously configured SequentialTaskRunner. Install prefect-dask, made these changes, then save the updated code as dask_flow.py.

    from prefect import flow, task\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n@flow(task_runner=DaskTaskRunner())\ndef greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n\nif __name__ == \"__main__\":\n    greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n

    Note that, because you're using DaskTaskRunner in a script, you must use if __name__ == \"__main__\": or you'll see warnings and errors.

    Now run dask_flow.py. If you get a warning about accepting incoming network connections, that's okay - everything is local in this example.

    $ python dask_flow.py\n19:29:03.798 | INFO    | prefect.engine - Created flow run 'fine-bison' for flow 'greetings'\n\n19:29:03.798 | INFO    | Flow run 'fine-bison' - Using task runner 'DaskTaskRunner'\n\n19:29:04.080 | INFO    | prefect.task_runner.dask - Creating a new Dask cluster with `distributed.deploy.local.LocalCluster`\n16:54:18.465 | INFO    | prefect.engine - Created flow run 'radical-finch' for flow 'greetings'\n16:54:18.465 | INFO    | Flow run 'radical-finch' - Starting 'DaskTaskRunner'; submitted tasks will be run concurrently...\n16:54:18.465 | INFO    | prefect.task_runner.dask - Creating a new Dask cluster with `distributed.deploy.local.LocalCluster`\n16:54:19.811 | INFO    | prefect.task_runner.dask - The Dask dashboard is available at <http://127.0.0.1:8787/status>\n16:54:19.881 | INFO    | Flow run 'radical-finch' - Created task run 'say_hello-811087cd-0' for task 'say_hello'\n16:54:20.364 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_hello-811087cd-0' for execution.\n16:54:20.379 | INFO    | Flow run 'radical-finch' - Created task run 'say_goodbye-261e56a8-0' for task 'say_goodbye'\n16:54:20.386 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_goodbye-261e56a8-0' for execution.\n16:54:20.397 | INFO    | Flow run 'radical-finch' - Created task run 'say_hello-811087cd-1' for task 'say_hello'\n16:54:20.401 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_hello-811087cd-1' for execution.\n16:54:20.417 | INFO    | Flow run 'radical-finch' - Created task run 'say_goodbye-261e56a8-1' for task 'say_goodbye'\n16:54:20.423 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_goodbye-261e56a8-1' for execution.\n16:54:20.443 | INFO    | Flow run 'radical-finch' - Created task run 'say_hello-811087cd-2' for task 'say_hello'\n16:54:20.449 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_hello-811087cd-2' for execution.\n16:54:20.462 | INFO    | Flow run 'radical-finch' - Created task run 'say_goodbye-261e56a8-2' for task 'say_goodbye'\n16:54:20.474 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_goodbye-261e56a8-2' for execution.\n16:54:20.500 | INFO    | Flow run 'radical-finch' - Created task run 'say_hello-811087cd-3' for task 'say_hello'\n16:54:20.511 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_hello-811087cd-3' for execution.\n16:54:20.544 | INFO    | Flow run 'radical-finch' - Created task run 'say_goodbye-261e56a8-3' for task 'say_goodbye'\n16:54:20.555 | INFO    | Flow run 'radical-finch' - Submitted task run 'say_goodbye-261e56a8-3' for execution.\nhello arthur\ngoodbye ford\ngoodbye arthur\nhello ford\ngoodbye marvin\ngoodbye trillian\nhello trillian\nhello marvin\n

    DaskTaskRunner automatically creates a local Dask cluster, then starts executing all of the tasks in parallel. The results do not return in the same order as the sequential code above.

    Notice what happens if you do not use the submit method when calling tasks:

    from prefect import flow, task\nfrom prefect_dask.task_runners import DaskTaskRunner\n\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n\n@flow(task_runner=DaskTaskRunner())\ndef greetings(names):\n    for name in names:\n        say_hello(name)\n        say_goodbye(name)\n\n\nif __name__ == \"__main__\":\n    greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n
    $ python dask_flow.py\n\n16:57:34.534 | INFO    | prefect.engine - Created flow run 'papaya-honeybee' for flow 'greetings'\n16:57:34.534 | INFO    | Flow run 'papaya-honeybee' - Starting 'DaskTaskRunner'; submitted tasks will be run concurrently...\n16:57:34.535 | INFO    | prefect.task_runner.dask - Creating a new Dask cluster with `distributed.deploy.local.LocalCluster`\n16:57:35.715 | INFO    | prefect.task_runner.dask - The Dask dashboard is available at <http://127.0.0.1:8787/status>\n16:57:35.787 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_hello-811087cd-0' for task 'say_hello'\n16:57:35.788 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_hello-811087cd-0' immediately...\nhello arthur\n16:57:35.810 | INFO    | Task run 'say_hello-811087cd-0' - Finished in state Completed()\n16:57:35.820 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_goodbye-261e56a8-0' for task 'say_goodbye'\n16:57:35.820 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_goodbye-261e56a8-0' immediately...\ngoodbye arthur\n16:57:35.840 | INFO    | Task run 'say_goodbye-261e56a8-0' - Finished in state Completed()\n16:57:35.849 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_hello-811087cd-1' for task 'say_hello'\n16:57:35.849 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_hello-811087cd-1' immediately...\nhello trillian\n16:57:35.869 | INFO    | Task run 'say_hello-811087cd-1' - Finished in state Completed()\n16:57:35.878 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_goodbye-261e56a8-1' for task 'say_goodbye'\n16:57:35.878 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_goodbye-261e56a8-1' immediately...\ngoodbye trillian\n16:57:35.894 | INFO    | Task run 'say_goodbye-261e56a8-1' - Finished in state Completed()\n16:57:35.907 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_hello-811087cd-2' for task 'say_hello'\n16:57:35.907 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_hello-811087cd-2' immediately...\nhello ford\n16:57:35.924 | INFO    | Task run 'say_hello-811087cd-2' - Finished in state Completed()\n16:57:35.933 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_goodbye-261e56a8-2' for task 'say_goodbye'\n16:57:35.933 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_goodbye-261e56a8-2' immediately...\ngoodbye ford\n16:57:35.951 | INFO    | Task run 'say_goodbye-261e56a8-2' - Finished in state Completed()\n16:57:35.959 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_hello-811087cd-3' for task 'say_hello'\n16:57:35.959 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_hello-811087cd-3' immediately...\nhello marvin\n16:57:35.976 | INFO    | Task run 'say_hello-811087cd-3' - Finished in state Completed()\n16:57:35.985 | INFO    | Flow run 'papaya-honeybee' - Created task run 'say_goodbye-261e56a8-3' for task 'say_goodbye'\n16:57:35.985 | INFO    | Flow run 'papaya-honeybee' - Executing 'say_goodbye-261e56a8-3' immediately...\ngoodbye marvin\n16:57:36.004 | INFO    | Task run 'say_goodbye-261e56a8-3' - Finished in state Completed()\n16:57:36.289 | INFO    | Flow run 'papaya-honeybee' - Finished in state Completed('All states completed.')\n

    The tasks are not submitted to the DaskTaskRunner and are run sequentially.

    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/dask-ray-task-runners/#running-parallel-tasks-with-ray","title":"Running parallel tasks with Ray","text":"

    To demonstrate the ability to flexibly apply the task runner appropriate for your workflow, use the same flow as above, with a few minor changes to use the RayTaskRunner where we previously configured DaskTaskRunner.

    To configure your flow to use the RayTaskRunner:

    1. Make sure the prefect-ray collection is installed by running pip install prefect-ray.
    2. In your flow code, import RayTaskRunner from prefect_ray.task_runners.
    3. Assign it as the task runner when the flow is defined using the task_runner=RayTaskRunner argument.

    Ray environment limitations

    While we're excited about parallel task execution via Ray to Prefect, there are some inherent limitations with Ray you should be aware of:

    • Support for Python 3.11 is experimental.
    • Ray support for non-x86/64 architectures such as ARM/M1 processors with installation from pip alone and will be skipped during installation of Prefect. It is possible to manually install the blocking component with conda. See the Ray documentation for instructions.
    • Ray's Windows support is currently in beta.

    See the Ray installation documentation for further compatibility information.

    Save this code in ray_flow.py.

    from prefect import flow, task\nfrom prefect_ray.task_runners import RayTaskRunner\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n@flow(task_runner=RayTaskRunner())\ndef greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n\nif __name__ == \"__main__\":\n    greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n

    Now run ray_flow.py RayTaskRunner automatically creates a local Ray instance, then immediately starts executing all of the tasks in parallel. If you have an existing Ray instance, you can provide the address as a parameter to run tasks in the instance. See Running tasks on Ray for details.

    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/dask-ray-task-runners/#using-multiple-task-runners","title":"Using multiple task runners","text":"

    Many workflows include a variety of tasks, and not all of them benefit from parallel execution. You'll most likely want to use the Dask or Ray task runners and spin up their respective resources only for those tasks that need them.

    Because task runners are specified on flows, you can assign different task runners to tasks by using subflows to organize those tasks.

    This example uses the same tasks as the previous examples, but on the parent flow greetings() we use the default ConcurrentTaskRunner. Then we call a ray_greetings() subflow that uses the RayTaskRunner to execute the same tasks in a Ray instance.

    from prefect import flow, task\nfrom prefect_ray.task_runners import RayTaskRunner\n\n@task\ndef say_hello(name):\n    print(f\"hello {name}\")\n\n@task\ndef say_goodbye(name):\n    print(f\"goodbye {name}\")\n\n@flow(task_runner=RayTaskRunner())\ndef ray_greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n\n@flow()\ndef greetings(names):\n    for name in names:\n        say_hello.submit(name)\n        say_goodbye.submit(name)\n    ray_greetings(names)\n\nif __name__ == \"__main__\":\n    greetings([\"arthur\", \"trillian\", \"ford\", \"marvin\"])\n

    If you save this as ray_subflow.py and run it, you'll see that the flow greetings runs as you'd expect for a concurrent flow, then flow ray-greetings spins up a Ray instance to run the tasks again.

    ","tags":["tasks","task runners","flow configuration","parallel execution","distributed execution","Dask","Ray"],"boost":2},{"location":"guides/docker/","title":"Running Flows with Docker","text":"

    In the Deployments tutorial, we looked at serving a flow that enables scheduling or creating flow runs via the Prefect API.

    With our Python script in hand, we can build a Docker image for our script, allowing us to serve our flow in various remote environments. We'll use Kubernetes in this guide, but you can use any Docker-compatible infrastructure.

    In this guide we'll:

    • Write a Dockerfile to build an image that stores our Prefect flow code.
    • Build a Docker image for our flow.
    • Deploy and run our Docker image on a Kubernetes cluster.
    • Look at the Prefect-maintained Docker images and discuss options for use

    Note that in this guide we'll create a Dockerfile from scratch. Alternatively, Prefect makes it convenient to build a Docker image as part of deployment creation. You can even include environment variables and specify additional Python packages to install at runtime.

    If creating a deployment with a prefect.yaml file, the build step makes it easy to customize your Docker image and push it to the registry of your choice. See an example here.

    Deployment creation with a Python script that includes flow.deploy similarly allows you to customize your Docker image with keyword arguments as shown below.

    ...\n\nif __name__ == \"__main__\":\n    hello_world.deploy(\n        name=\"my-first-deployment\",\n        work_pool_name=\"above-ground\",\n        image='my_registry/hello_world:demo',\n        job_variables={\"env\": { \"EXTRA_PIP_PACKAGES\": \"boto3\" } }\n    )\n
    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#prerequisites","title":"Prerequisites","text":"

    To complete this guide, you'll need the following:

    • A Python script that defines and serves a flow.
    • We'll use the flow script and deployment from the Deployments tutorial.
    • Access to a running Prefect API server.
    • You can sign up for a forever free Prefect Cloud account or run a Prefect API server locally with prefect server start.
    • Docker Desktop installed on your machine.
    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#writing-a-dockerfile","title":"Writing a Dockerfile","text":"

    First let's make a clean directory to work from, prefect-docker-guide.

    mkdir prefect-docker-guide\ncd prefect-docker-guide\n

    In this directory, we'll create a sub-directory named flows and put our flow script from the Deployments tutorial in it.

    mkdir flows\ncd flows\ntouch prefect-docker-guide-flow.py\n

    Here's the flow code for reference:

    prefect-docker-guide-flow.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.serve(name=\"prefect-docker-guide\")\n

    The next file we'll add to the prefect-docker-guide directory is a requirements.txt. We'll include all dependencies required for our prefect-docker-guide-flow.py script in the Docker image we'll build.

    touch requirements.txt\n

    Here's what we'll put in our requirements.txt file:

    requirements.txt
    prefect>=2.12.0\nhttpx\n

    Next, we'll create a Dockerfile that we'll use to create a Docker image that will also store the flow code.

    touch Dockerfile\n

    We'll add the following content to our Dockerfile:

    Dockerfile
    # We're using the latest version of Prefect with Python 3.10\nFROM prefecthq/prefect:2-python3.10\n\n# Add our requirements.txt file to the image and install dependencies\nCOPY requirements.txt .\nRUN pip install -r requirements.txt --trusted-host pypi.python.org --no-cache-dir\n\n# Add our flow code to the image\nCOPY flows /opt/prefect/flows\n\n# Run our flow script when the container starts\nCMD [\"python\", \"flows/prefect-docker-guide-flow.py\"]\n
    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#building-a-docker-image","title":"Building a Docker image","text":"

    Now that we have a Dockerfile we can build our image by running:

    docker build -t prefect-docker-guide-image .\n

    We can check that our build worked by running a container from our new image.

    CloudSelf-hosted

    Our container will need an API URL and and API key to communicate with Prefect Cloud.

    • You can get an API key from the API Keys section of the user settings in the Prefect UI.

    • You can get your API URL by running prefect config view and copying the PREFECT_API_URL value.

    We'll provide both these values to our container by passing them as environment variables with the -e flag.

    docker run -e PREFECT_API_URL=YOUR_PREFECT_API_URL -e PREFECT_API_KEY=YOUR_API_KEY prefect-docker-guide-image\n

    After running the above command, the container should start up and serve the flow within the container!

    Our container will need an API URL and network access to communicate with the Prefect API.

    For this guide, we'll assume the Prefect API is running on the same machine that we'll run our container on and the Prefect API was started with prefect server start. If you're running a different setup, check out the Hosting a Prefect server guide for information on how to connect to your Prefect API instance.

    To ensure that our flow container can communicate with the Prefect API, we'll set our PREFECT_API_URL to http://host.docker.internal:4200/api. If you're running Linux, you'll need to set your PREFECT_API_URL to http://localhost:4200/api and use the --network=\"host\" option instead.

    docker run --network=\"host\" -e PREFECT_API_URL=http://host.docker.internal:4200/api prefect-docker-guide-image\n

    After running the above command, the container should start up and serve the flow within the container!

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#deploying-to-a-remote-environment","title":"Deploying to a remote environment","text":"

    Now that we have a Docker image with our flow code embedded, we can deploy it to a remote environment!

    For this guide, we'll simulate a remote environment by using Kubernetes locally with Docker Desktop. You can use the instructions provided by Docker to set up Kubernetes locally.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#creating-a-kubernetes-deployment-manifest","title":"Creating a Kubernetes deployment manifest","text":"

    To ensure the process serving our flow is always running, we'll create a Kubernetes deployment. If our flow's container ever crashes, Kubernetes will automatically restart it, ensuring that we won't miss any scheduled runs.

    First, we'll create a deployment-manifest.yaml file in our prefect-docker-guide directory:

    touch deployment-manifest.yaml\n

    And we'll add the following content to our deployment-manifest.yaml file:

    CloudSelf-hosted deployment-manifest.yaml
    apiVersion: apps/v1\nkind: Deployment\nmetadata:\n  name: prefect-docker-guide\nspec:\n  replicas: 1\n  selector:\n    matchLabels:\n      flow: get-repo-info\n  template:\n    metadata:\n      labels:\n        flow: get-repo-info\n    spec:\n      containers:\n      - name: flow-container\n        image: prefect-docker-guide-image:latest\n        env:\n        - name: PREFECT_API_URL\n          value: YOUR_PREFECT_API_URL\n        - name: PREFECT_API_KEY\n          value: YOUR_API_KEY\n        # Never pull the image because we're using a local image\n        imagePullPolicy: Never\n

    Keep your API key secret

    In the above manifest we are passing in the Prefect API URL and API key as environment variables. This approach is simple, but it is not secure. If you are deploying your flow to a remote cluster, you should use a Kubernetes secret to store your API key.

    deployment-manifest.yaml
    apiVersion: apps/v1\nkind: Deployment\nmetadata:\n  name: prefect-docker-guide\nspec:\n  replicas: 1\n  selector:\n    matchLabels:\n      flow: get-repo-info\n  template:\n    metadata:\n      labels:\n        flow: get-repo-info\n    spec:\n      containers:\n      - name: flow-container\n        image: prefect-docker-guide-image:latest\n        env:\n        - name: PREFECT_API_URL\n          value: <http://host.docker.internal:4200/api>\n        # Never pull the image because we're using a local image\n        imagePullPolicy: Never\n

    Linux users

    If you're running Linux, you'll need to set your PREFECT_API_URL to use the IP address of your machine instead of host.docker.internal.

    This manifest defines how our image will run when deployed in our Kubernetes cluster. Note that we will be running a single replica of our flow container. If you want to run multiple replicas of your flow container to keep up with an active schedule, or because our flow is resource-intensive, you can increase the replicas value.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#deploying-our-flow-to-the-cluster","title":"Deploying our flow to the cluster","text":"

    Now that we have a deployment manifest, we can deploy our flow to the cluster by running:

    kubectl apply -f deployment-manifest.yaml\n

    We can monitor the status of our Kubernetes deployment by running:

    kubectl get deployments\n

    Once the deployment has successfully started, we can check the logs of our flow container by running the following:

    kubectl logs -l flow=get-repo-info\n

    Now that we're serving our flow in our cluster, we can trigger a flow run by running:

    prefect deployment run get-repo-info/prefect-docker-guide\n

    If we navigate to the URL provided by the prefect deployment run command, we can follow the flow run via the logs in the Prefect UI!

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#prefect-maintained-docker-images","title":"Prefect-maintained Docker images","text":"

    Every release of Prefect results in several new Docker images. These images are all named prefecthq/prefect and their tags identify their differences.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#image-tags","title":"Image tags","text":"

    When a release is published, images are built for all of Prefect's supported Python versions. These images are tagged to identify the combination of Prefect and Python versions contained. Additionally, we have \"convenience\" tags which are updated with each release to facilitate automatic updates.

    For example, when release 2.11.5 is published:

    1. Images with the release packaged are built for each supported Python version (3.8, 3.9, 3.10, 3.11) with both standard Python and Conda.
    2. These images are tagged with the full description, e.g. prefect:2.1.1-python3.10 and prefect:2.1.1-python3.10-conda.
    3. For users that want more specific pins, these images are also tagged with the SHA of the git commit of the release, e.g. sha-88a7ff17a3435ec33c95c0323b8f05d7b9f3f6d2-python3.10
    4. For users that want to be on the latest 2.1.x release, receiving patch updates, we update a tag without the patch version to this release, e.g. prefect.2.1-python3.10.
    5. For users that want to be on the latest 2.x.y release, receiving minor version updates, we update a tag without the minor or patch version to this release, e.g. prefect.2-python3.10
    6. Finally, for users who want the latest 2.x.y release without specifying a Python version, we update 2-latest to the image for our highest supported Python version, which in this case would be equivalent to prefect:2.1.1-python3.10.

    Choose image versions carefully

    It's a good practice to use Docker images with specific Prefect versions in production.

    Use care when employing images that automatically update to new versions (such as prefecthq/prefect:2-python3.11 or prefecthq/prefect:2-latest).

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#standard-python","title":"Standard Python","text":"

    Standard Python images are based on the official Python slim images, e.g. python:3.10-slim.

    Tag Prefect Version Python Version 2-latest most recent v2 PyPi version 3.10 2-python3.11 most recent v2 PyPi version 3.11 2-python3.10 most recent v2 PyPi version 3.10 2-python3.9 most recent v2 PyPi version 3.9 2-python3.8 most recent v2 PyPi version 3.8 2.X-python3.11 2.X 3.11 2.X-python3.10 2.X 3.10 2.X-python3.9 2.X 3.9 2.X-python3.8 2.X 3.8 sha-<hash>-python3.11 <hash> 3.11 sha-<hash>-python3.10 <hash> 3.10 sha-<hash>-python3.9 <hash> 3.9 sha-<hash>-python3.8 <hash> 3.8","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#conda-flavored-python","title":"Conda-flavored Python","text":"

    Conda flavored images are based on continuumio/miniconda3. Prefect is installed into a conda environment named prefect.

    Tag Prefect Version Python Version 2-latest-conda most recent v2 PyPi version 3.10 2-python3.11-conda most recent v2 PyPi version 3.11 2-python3.10-conda most recent v2 PyPi version 3.10 2-python3.9-conda most recent v2 PyPi version 3.9 2-python3.8-conda most recent v2 PyPi version 3.8 2.X-python3.11-conda 2.X 3.11 2.X-python3.10-conda 2.X 3.10 2.X-python3.9-conda 2.X 3.9 2.X-python3.8-conda 2.X 3.8 sha-<hash>-python3.11-conda <hash> 3.11 sha-<hash>-python3.10-conda <hash> 3.10 sha-<hash>-python3.9-conda <hash> 3.9 sha-<hash>-python3.8-conda <hash> 3.8","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#building-your-own-image","title":"Building your own image","text":"

    If your flow relies on dependencies not found in the default prefecthq/prefect images, you may want to build your own image. You can either base it off of one of the provided prefecthq/prefect images, or build your own image. See the Work pool deployment guide for discussion of how Prefect can help you build custom images with dependencies specifiied in a requirements.txt file.

    By default, Prefect work pools that use containers refer to the 2-latest image. You can specify another image at work pool creation. The work pool image choice can be overridden in individual deployments.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#extending-the-prefecthqprefect-image-manually","title":"Extending the prefecthq/prefect image manually","text":"

    Here we provide an example Dockerfile for building an image based on prefecthq/prefect:2-latest, but with scikit-learn installed.

    FROM prefecthq/prefect:2-latest\n\nRUN pip install scikit-learn\n
    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#choosing-an-image-strategy","title":"Choosing an image strategy","text":"

    The options described above have different complexity (and performance) characteristics. For choosing a strategy, we provide the following recommendations:

    • If your flow only makes use of tasks defined in the same file as the flow, or tasks that are part of prefect itself, then you can rely on the default provided prefecthq/prefect image.

    • If your flow requires a few extra dependencies found on PyPI, you can use the default prefecthq/prefect image and set prefect.deployments.steps.pip_install_requirements: in the pullstep to install these dependencies at runtime.

    • If the installation process requires compiling code or other expensive operations, you may be better off building a custom image instead.

    • If your flow (or flows) require extra dependencies or shared libraries, we recommend building a shared custom image with all the extra dependencies and shared task definitions you need. Your flows can then all rely on the same image, but have their source stored externally. This option can ease development, as the shared image only needs to be rebuilt when dependencies change, not when the flow source changes.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/docker/#next-steps","title":"Next steps","text":"

    We only served a single flow in this guide, but you can extend this setup to serve multiple flows in a single Docker image by updating your Python script to using flow.to_deployment and serve to serve multiple flows or the same flow with different configuration.

    To learn more about deploying flows, check out the Deployments concept doc!

    For advanced infrastructure requirements, such as executing each flow run within its own dedicated Docker container, learn more in the Work pool deployment guide.

    ","tags":["Docker","containers","orchestration","infrastructure","deployments","images","Kubernetes"],"boost":2},{"location":"guides/global-concurrency-limits/","title":"Global concurrency limits and rate limits","text":"

    Global concurrency limits allow you to manage execution efficiently, controlling how many tasks, flows, or other operations can run simultaneously. They are ideal when optimizing resource usage, preventing bottlenecks, and customizing task execution are priorities.

    Clarification on use of the term 'tasks'

    In the context of global concurrency and rate limits, \"tasks\" refers not specifically to Prefect tasks, but to concurrent units of work in general, such as those managed by an event loop or TaskGroup in asynchronous programming. These general \"tasks\" could include Prefect tasks when they are part of an asynchronous execution environment.

    Rate Limits ensure system stability by governing the frequency of requests or operations. They are suitable for preventing overuse, ensuring fairness, and handling errors gracefully.

    When selecting between Concurrency and Rate Limits, consider your primary goal. Choose Concurrency Limits for resource optimization and task management. Choose Rate Limits to maintain system stability and fair access to services.

    The core difference between a rate limit and a concurrency limit is the way in which slots are released. With a rate limit, slots are released at a controlled rate, controlled by slot_decay_per_second whereas with a concurrency limit, slots are released when the concurrency manager is exited.

    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#managing-global-concurrency-limits-and-rate-limits","title":"Managing Global concurrency limits and rate limits","text":"

    You can create, read, edit, and delete concurrency limits via the Prefect UI.

    When creating a concurrency limit, you can specify the following parameters:

    • Name: The name of the concurrency limit. This name is also how you'll reference the concurrency limit in your code. Special characters, such as /, %, &, >, <, are not allowed.
    • Concurrency Limit: The maximum number of slots that can be occupied on this concurrency limit.
    • Slot Decay Per Second: Controls the rate at which slots are released when the concurrency limit is used as a rate limit. This value must be configured when using the rate_limit function.
    • Active: Whether or not the concurrency limit is in an active state.
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#active-vs-inactive-limits","title":"Active vs inactive limits","text":"

    Global concurrency limits can be in either an active or inactive state.

    • Active: In this state, slots can be occupied, and code execution will be blocked when slots are unable to be acquired.
    • Inactive: In this state, slots will not be occupied, and code execution will not be blocked. Concurrency enforcement occurs only when you activate the limit.
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#slot-decay","title":"Slot decay","text":"

    Global concurrency limits can be configured with slot decay. This is used when the concurrency limit is used as a rate limit, and it governs the pace at which slots are released or become available for reuse after being occupied. These slots effectively represent the concurrency capacity within a specific concurrency limit. The concept is best understood as the rate at which these slots \"decay\" or refresh.

    To configure slot decay, you can set the slot_decay_per_second parameter when defining or adjusting a concurrency limit.

    For practical use, consider the following:

    • Higher values: Setting slot_decay_per_second to a higher value, such as 5.0, results in slots becoming available relatively quickly. In this scenario, a slot that was occupied by a task will free up after just 0.2 (1.0 / 5.0) seconds.

    • Lower values: Conversely, setting slot_decay_per_second to a lower value, like 0.1, causes slots to become available more slowly. In this scenario it would take 10 (1.0 / 0.1) seconds for a slot to become available again after occupancy

    Slot decay provides fine-grained control over the availability of slots, enabling you to optimize the rate of your workflow based on your specific requirements.

    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#using-the-concurrency-context-manager","title":"Using the concurrency context manager","text":"

    The concurrencycontext manager allows control over the maximum number of concurrent operations. You can select either the synchronous (sync) or asynchronous (async) version, depending on your use case. Here's how to use it:

    Concurrency limits are implicitly created

    When using the concurrency context manager, the concurrency limit you use will be created, in an inactive state, if it does not already exist.

    Sync

    from prefect import flow, task\nfrom prefect.concurrency.sync import concurrency\n\n\n@task\ndef process_data(x, y):\n    with concurrency(\"database\", occupy=1):\n        return x + y\n\n\n@flow\ndef my_flow():\n    for x, y in [(1, 2), (2, 3), (3, 4), (4, 5)]:\n        process_data.submit(x, y)\n\n\nif __name__ == \"__main__\":\n    my_flow()\n

    Async

    import asyncio\nfrom prefect import flow, task\nfrom prefect.concurrency.asyncio import concurrency\n\n\n@task\nasync def process_data(x, y):\n    async with concurrency(\"database\", occupy=1):\n        return x + y\n\n\n@flow\nasync def my_flow():\n    for x, y in [(1, 2), (2, 3), (3, 4), (4, 5)]:\n        await process_data.submit(x, y)\n\n\nif __name__ == \"__main__\":\n    asyncio.run(my_flow())\n
    1. The code imports the necessary modules and the concurrency context manager. Use the prefect.concurrency.sync module for sync usage and the prefect.concurrency.asyncio module for async usage.
    2. It defines a process_data task, taking x and y as input arguments. Inside this task, the concurrency context manager controls concurrency, using the database concurrency limit and occupying one slot. If another task attempts to run with the same limit and no slots are available, that task will be blocked until a slot becomes available.
    3. A flow named my_flow is defined. Within this flow, it iterates through a list of tuples, each containing pairs of x and y values. For each pair, the process_data task is submitted with the corresponding x and y values for processing.
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#using-rate_limit","title":"Using rate_limit","text":"

    The Rate Limit feature provides control over the frequency of requests or operations, ensuring responsible usage and system stability. Depending on your requirements, you can utilize rate_limit to govern both synchronous (sync) and asynchronous (async) operations. Here's how to make the most of it:

    Slot decay

    When using the rate_limit function the concurrency limit you use must have a slot decay configured.

    Sync

    from prefect import flow, task\nfrom prefect.concurrency.sync import rate_limit\n\n\n@task\ndef make_http_request():\n    rate_limit(\"rate-limited-api\")\n    print(\"Making an HTTP request...\")\n\n\n@flow\ndef my_flow():\n    for _ in range(10):\n        make_http_request.submit()\n\n\nif __name__ == \"__main__\":\n    my_flow()\n

    Async

    import asyncio\n\nfrom prefect import flow, task\nfrom prefect.concurrency.asyncio import rate_limit\n\n\n@task\nasync def make_http_request():\n    await rate_limit(\"rate-limited-api\")\n    print(\"Making an HTTP request...\")\n\n\n@flow\nasync def my_flow():\n    for _ in range(10):\n        await make_http_request.submit()\n\n\nif __name__ == \"__main__\":\n    asyncio.run(my_flow())\n
    1. The code imports the necessary modules and the rate_limit function. Use the prefect.concurrency.sync module for sync usage and the prefect.concurrency.asyncio module for async usage.
    2. It defines a make_http_request task. Inside this task, the rate_limit function is used to ensure that the requests are made at a controlled pace.
    3. A flow named my_flow is defined. Within this flow the make_http_request task is submitted 10 times.
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#using-concurrency-and-rate_limit-outside-of-a-flow","title":"Using concurrency and rate_limit outside of a flow","text":"

    concurreny and rate_limit can be used outside of a flow to control concurrency and rate limits for any operation.

    import asyncio\n\nfrom prefect.concurrency.asyncio import rate_limit\n\n\nasync def main():\n    for _ in range(10):\n        await rate_limit(\"rate-limited-api\")\n        print(\"Making an HTTP request...\")\n\n\n\nif __name__ == \"__main__\":\n    asyncio.run(main())\n
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#use-cases","title":"Use cases","text":"","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#throttling-task-submission","title":"Throttling task submission","text":"

    Throttling task submission to avoid overloading resources, to comply with external rate limits, or ensure a steady, controlled flow of work.

    In this scenario the rate_limit function is used to throttle the submission of tasks. The rate limit acts as a bottleneck, ensuring that tasks are submitted at a controlled rate, governed by the slot_decay_per_second setting on the associated concurrency limit.

    from prefect import flow, task\nfrom prefect.concurrency.sync import rate_limit\n\n\n@task\ndef my_task(i):\n    return i\n\n\n@flow\ndef my_flow():\n    for _ in range(100):\n        rate_limit(\"slow-my-flow\", occupy=1)\n        my_task.submit(1)\n\n\nif __name__ == \"__main__\":\n    my_flow()\n
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#managing-database-connections","title":"Managing database connections","text":"

    Managing the maximum number of concurrent database connections to avoid exhausting database resources.

    In this scenario we've setup a concurrency limit named database and given it a maximum concurrency limit that matches the maximum number of database connections we want to allow. We then use the concurrency context manager to control the number of database connections allowed at any one time.

    from prefect import flow, task, concurrency\nimport psycopg2\n\n@task\ndef database_query(query):\n    # Here we request a single slot on the 'database' concurrency limit. This\n    # will block in the case that all of the database connections are in use\n    # ensuring that we never exceed the maximum number of database connections.\n    with concurrency(\"database\", occupy=1):\n        connection = psycopg2.connect(\"<connection_string>\")\n        cursor = connection.cursor()\n        cursor.execute(query)\n        result = cursor.fetchall()\n        connection.close()\n        return result\n\n@flow\ndef my_flow():\n    queries = [\"SELECT * FROM table1\", \"SELECT * FROM table2\", \"SELECT * FROM table3\"]\n\n    for query in queries:\n        database_query.submit(query)\n\nif __name__ == \"__main__\":\n    my_flow()\n
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/global-concurrency-limits/#parallel-data-processing","title":"Parallel data processing","text":"

    Limiting the maximum number of parallel processing tasks.

    In this scenario we want to limit the number of process_data tasks to five at any one time. We do this by using the concurrency context manager to request five slots on the data-processing concurrency limit. This will block until five slots are free and then submit five more tasks, ensuring that we never exceed the maximum number of parallel processing tasks.

    import asyncio\nfrom prefect.concurrency.sync import concurrency\n\n\nasync def process_data(data):\n    print(f\"Processing: {data}\")\n    await asyncio.sleep(1)\n    return f\"Processed: {data}\"\n\n\nasync def main():\n    data_items = list(range(100))\n    processed_data = []\n\n    while data_items:\n        with concurrency(\"data-processing\", occupy=5):\n            chunk = [data_items.pop() for _ in range(5)]\n            processed_data += await asyncio.gather(\n                *[process_data(item) for item in chunk]\n            )\n\n    print(processed_data)\n\n\nif __name__ == \"__main__\":\n    asyncio.run(main())\n
    ","tags":["concurrency","rate limits"],"boost":2},{"location":"guides/host/","title":"Hosting a Prefect server","text":"

    After you install Prefect you have a Python SDK client that can communicate with Prefect Cloud, the platform hosted by Prefect. You also have an API server backed by a database and a UI.

    In this section you'll learn how to host your own Prefect server.

    Spin up a local Prefect server UI with the prefect server start CLI command in the terminal:

    prefect server start\n

    Open the URL for the Prefect server UI (http://127.0.0.1:4200 by default) in a browser.

    Shut down the Prefect server with ctrl + c in the terminal.","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#differences-between-a-self-hosted-prefect-server-and-prefect-cloud","title":"Differences between a self-hosted Prefect server and Prefect Cloud","text":"

    A self-hosted Prefect server and Prefect Cloud share a common set of features. Prefect Cloud includes the following additional features:

    • Workspaces \u2014 isolated environments to organize your flows, deployments, and flow runs.
    • Automations \u2014 configure triggers, actions, and notifications in response to real-time monitoring events.
    • Email notifications \u2014 send email alerts from Prefect's servers based on automation triggers.
    • Service accounts \u2014 configure API access for running workers or executing flow runs on remote infrastructure.
    • Custom role-based access controls (RBAC) \u2014 assign users granular permissions to perform activities within an account or workspace.
    • Single Sign-on (SSO) \u2014 authentication using your identity provider.
    • Audit Log \u2014 a record of user activities to monitor security and compliance.

    You can read more about Prefect Cloud in the Cloud section.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#configuring-a-prefect-server-instance","title":"Configuring a Prefect server instance","text":"

    Go to your terminal session and run this command to set the API URL to point to a Prefect server instance:

    prefect config set PREFECT_API_URL=\"http://127.0.0.1:4200/api\"\n

    PREFECT_API_URL required when running Prefect inside a container

    You must set the API server address to use Prefect within a container, such as a Docker container.

    You can save the API server address in a Prefect profile. Whenever that profile is active, the API endpoint will be be at that address.

    See Profiles & Configuration for more information on profiles and configurable Prefect settings.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#prefect-database","title":"Prefect database","text":"

    The Prefect database persists data to track the state of your flow runs and related Prefect concepts, including:

    • Flow run and task run state
    • Run history
    • Logs
    • Deployments
    • Flow and task run concurrency limits
    • Storage blocks for flow and task results
    • Variables
    • Artifacts
    • Work pool status

    Currently Prefect supports the following databases:

    • SQLite: The default in Prefect, and our recommendation for lightweight, single-server deployments. SQLite requires essentially no setup.
    • PostgreSQL: Best for connecting to external databases, but does require additional setup (such as Docker). Prefect uses the pg_trgm extension, so it must be installed and enabled.
    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#using-the-database","title":"Using the database","text":"

    A local SQLite database is the default database and is configured upon Prefect installation. The database is located at ~/.prefect/prefect.db by default.

    To reset your database, run the CLI command:

    prefect server database reset -y\n

    This command will clear all data and reapply the schema.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#database-settings","title":"Database settings","text":"

    Prefect provides several settings for configuring the database. Here are the default settings:

    PREFECT_API_DATABASE_CONNECTION_URL='sqlite+aiosqlite:///${PREFECT_HOME}/prefect.db'\nPREFECT_API_DATABASE_ECHO='False'\nPREFECT_API_DATABASE_MIGRATE_ON_START='True'\nPREFECT_API_DATABASE_PASSWORD='None'\n

    You can save a setting to your active Prefect profile with prefect config set.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#configuring-a-postgresql-database","title":"Configuring a PostgreSQL database","text":"

    To connect Prefect to a PostgreSQL database, you can set the following environment variable:

    prefect config set PREFECT_API_DATABASE_CONNECTION_URL=\"postgresql+asyncpg://postgres:yourTopSecretPassword@localhost:5432/prefect\"\n

    The above environment variable assumes that:

    • You have a username called postgres
    • Your password is set to yourTopSecretPassword
    • Your database runs on the same host as the Prefect server instance, localhost
    • You use the default PostgreSQL port 5432
    • Your PostgreSQL instance has a database called prefect

    If you want to quickly start a PostgreSQL instance that can be used as your Prefect database, you can use the following command that will start a Docker container running PostgreSQL:

    docker run -d --name prefect-postgres -v prefectdb:/var/lib/postgresql/data -p 5432:5432 -e POSTGRES_USER=postgres -e POSTGRES_PASSWORD=yourTopSecretPassword -e POSTGRES_DB=prefect postgres:latest\n

    The above command:

    • Pulls the latest version of the official postgres Docker image, which is compatible with Prefect.
    • Starts a container with the name prefect-postgres.
    • Creates a database prefect with a user postgres and yourTopSecretPassword password.
    • Mounts the PostgreSQL data to a Docker volume called prefectdb to provide persistence if you ever have to restart or rebuild that container.

    You can inspect your profile to be sure that the environment variable has been set properly:

    prefect config view --show-sources\n

    Start the Prefect server and it should from now on use your PostgreSQL database instance:

    prefect server start\n
    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#in-memory-database","title":"In-memory database","text":"

    One of the benefits of SQLite is in-memory database support.

    To use an in-memory SQLite database, set the following environment variable:

    prefect config set PREFECT_API_DATABASE_CONNECTION_URL=\"sqlite+aiosqlite:///file::memory:?cache=shared&uri=true&check_same_thread=false\"\n

    Use SQLite database for testing only

    SQLite is only supported by Prefect for testing purposes and is not compatible with multiprocessing.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#database-versions","title":"Database versions","text":"

    The following database versions are required for use with Prefect:

    • SQLite 3.24 or newer
    • PostgreSQL 13.0 or newer
    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#migrations","title":"Migrations","text":"

    Prefect uses Alembic to manage database migrations. Alembic is a database migration tool for usage with the SQLAlchemy Database Toolkit for Python. Alembic provides a framework for generating and applying schema changes to a database.

    To apply migrations to your database you can run the following commands:

    To upgrade:

    prefect server database upgrade -y\n

    To downgrade:

    prefect server database downgrade -y\n

    You can use the -r flag to specify a specific migration version to upgrade or downgrade to. For example, to downgrade to the previous migration version you can run:

    prefect server database downgrade -y -r -1\n

    or to downgrade to a specific revision:

    prefect server database downgrade -y -r d20618ce678e\n

    See the contributing docs for information on how to create new database migrations.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#notifications","title":"Notifications","text":"

    When you use Prefect Cloud you gain access to a hosted platform with Workspace & User controls, Events, and Automations. Prefect Cloud has an option for automation notifications. The more limited Notifications option is provided for the self-hosted Prefect server.

    Notifications enable you to set up alerts that are sent when a flow enters any state you specify. When your flow and task runs changes state, Prefect notes the state change and checks whether the new state matches any notification policies. If it does, a new notification is queued.

    Prefect supports sending notifications via:

    • Slack message to a channel
    • Microsoft Teams message to a channel
    • Opsgenie to alerts
    • PagerDuty to alerts
    • Twilio to phone numbers
    • Email (requires your own server)

    Notifications in Prefect Cloud

    Prefect Cloud uses the robust Automations interface to enable notifications related to flow run state changes and work pool status.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/host/#configure-notifications","title":"Configure notifications","text":"

    To configure a notification in a Prefect server, go to the Notifications page and select Create Notification or the + button.

    Notifications are structured just as you would describe them to someone. You can choose:

    • Which run states should trigger a notification.
    • Tags to filter which flow runs are covered by the notification.
    • Whether to send an email, a Slack message, Microsoft Teams message, or other services.

    For email notifications (supported on Prefect Cloud only), the configuration requires email addresses to which the message is sent.

    For Slack notifications, the configuration requires webhook credentials for your Slack and the channel to which the message is sent.

    For example, to get a Slack message if a flow with a daily-etl tag fails, the notification will read:

    If a run of any flow with daily-etl tag enters a failed state, send a notification to my-slack-webhook

    When the conditions of the notification are triggered, you\u2019ll receive a message:

    The fuzzy-leopard run of the daily-etl flow entered a failed state at 22-06-27 16:21:37 EST.

    On the Notifications page you can pause, edit, or delete any configured notification.

    ","tags":["UI","dashboard","Prefect Server","Observability","Events","Serve","Database","SQLite"],"boost":2},{"location":"guides/logs/","title":"Logging","text":"

    Prefect enables you to log a variety of useful information about your flow and task runs, capturing information about your workflows for purposes such as monitoring, troubleshooting, and auditing.

    Prefect captures logs for your flow and task runs by default, even if you have not started a Prefect server with prefect server start.

    You can view and filter logs in the Prefect UI or Prefect Cloud, or access log records via the API.

    Prefect enables fine-grained customization of log levels for flows and tasks, including configuration for default levels and log message formatting.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-overview","title":"Logging overview","text":"

    Whenever you run a flow, Prefect automatically logs events for flow runs and task runs, along with any custom log handlers you have configured. No configuration is needed to enable Prefect logging.

    For example, say you created a simple flow in a file flow.py. If you create a local flow run with python flow.py, you'll see an example of the log messages created automatically by Prefect:

    16:45:44.534 | INFO    | prefect.engine - Created flow run 'gray-dingo' for flow\n'hello-flow'\n16:45:44.534 | INFO    | Flow run 'gray-dingo' - Using task runner 'SequentialTaskRunner'\n16:45:44.598 | INFO    | Flow run 'gray-dingo' - Created task run 'hello-task-54135dc1-0'\nfor task 'hello-task'\nHello world!\n16:45:44.650 | INFO    | Task run 'hello-task-54135dc1-0' - Finished in state\nCompleted(None)\n16:45:44.672 | INFO    | Flow run 'gray-dingo' - Finished in state\nCompleted('All states completed.')\n

    You can see logs for a flow run in the Prefect UI by navigating to the Flow Runs page and selecting a specific flow run to inspect.

    These log messages reflect the logging configuration for log levels and message formatters. You may customize the log levels captured and the default message format through configuration, and you can capture custom logging events by explicitly emitting log messages during flow and task runs.

    Prefect supports the standard Python logging levels CRITICAL, ERROR, WARNING, INFO, and DEBUG. By default, Prefect displays INFO-level and above events. You can configure the root logging level as well as specific logging levels for flow and task runs.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-configuration","title":"Logging configuration","text":"","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-settings","title":"Logging settings","text":"

    Prefect provides several settings for configuring logging level and loggers.

    By default, Prefect displays INFO-level and above logging records. You may change this level to DEBUG and DEBUG-level logs created by Prefect will be shown as well. You may need to change the log level used by loggers from other libraries to see their log records.

    You can override any logging configuration by setting an environment variable or Prefect Profile setting using the syntax PREFECT_LOGGING_[PATH]_[TO]_[KEY], with [PATH]_[TO]_[KEY] corresponding to the nested address of any setting.

    For example, to change the default logging levels for Prefect to DEBUG, you can set the environment variable PREFECT_LOGGING_LEVEL=\"DEBUG\".

    You may also configure the \"root\" Python logger. The root logger receives logs from all loggers unless they explicitly opt out by disabling propagation. By default, the root logger is configured to output WARNING level logs to the console. As with other logging settings, you can override this from the environment or in the logging configuration file. For example, you can change the level with the variable PREFECT_LOGGING_ROOT_LEVEL.

    You may adjust the log level used by specific handlers. For example, you could set PREFECT_LOGGING_HANDLERS_API_LEVEL=ERROR to have only ERROR logs reported to the Prefect API. The console handlers will still default to level INFO.

    There is a logging.yml file packaged with Prefect that defines the default logging configuration.

    You can customize logging configuration by creating your own version of logging.yml with custom settings, by either creating the file at the default location (/.prefect/logging.yml) or by specifying the path to the file with PREFECT_LOGGING_SETTINGS_PATH. (If the file does not exist at the specified location, Prefect ignores the setting and uses the default configuration.)

    See the Python Logging configuration documentation for more information about the configuration options and syntax used by logging.yml.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#prefect-loggers","title":"Prefect loggers","text":"

    To access the Prefect logger, import from prefect import get_run_logger. You can send messages to the logger in both flows and tasks.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-in-flows","title":"Logging in flows","text":"

    To log from a flow, retrieve a logger instance with get_run_logger(), then call the standard Python logging methods.

    from prefect import flow, get_run_logger\n\n@flow(name=\"log-example-flow\")\ndef logger_flow():\n    logger = get_run_logger()\n    logger.info(\"INFO level log message.\")\n

    Prefect automatically uses the flow run logger based on the flow context. If you run the above code, Prefect captures the following as a log event.

    15:35:17.304 | INFO    | Flow run 'mottled-marten' - INFO level log message.\n

    The default flow run log formatter uses the flow run name for log messages.

    Note

    Starting in 2.7.11, if you use a logger that sends logs to the API outside of a flow or task run, a warning will be displayed instead of an error. You can silence this warning by setting `PREFECT_LOGGING_TO_API_WHEN_MISSING_FLOW=ignore` or have the logger raise an error by setting the value to `error`.\n
    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-in-tasks","title":"Logging in tasks","text":"

    Logging in tasks works much as logging in flows: retrieve a logger instance with get_run_logger(), then call the standard Python logging methods.

    from prefect import flow, task, get_run_logger\n\n@task(name=\"log-example-task\")\ndef logger_task():\n    logger = get_run_logger()\n    logger.info(\"INFO level log message from a task.\")\n\n@flow(name=\"log-example-flow\")\ndef logger_flow():\n    logger_task()\n

    Prefect automatically uses the task run logger based on the task context. The default task run log formatter uses the task run name for log messages.

    15:33:47.179 | INFO   | Task run 'logger_task-80a1ffd1-0' - INFO level log message from a task.\n

    The underlying log model for task runs captures the task name, task run ID, and parent flow run ID, which are persisted to the database for reporting and may also be used in custom message formatting.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#logging-print-statements","title":"Logging print statements","text":"

    Prefect provides the log_prints option to enable the logging of print statements at the task or flow level. When log_prints=True for a given task or flow, the Python builtin print will be patched to redirect to the Prefect logger for the scope of that task or flow.

    By default, tasks and subflows will inherit the log_prints setting from their parent flow, unless opted out with their own explicit log_prints setting.

    from prefect import task, flow\n\n@task\ndef my_task():\n    print(\"we're logging print statements from a task\")\n\n@flow(log_prints=True)\ndef my_flow():\n    print(\"we're logging print statements from a flow\")\n    my_task()\n

    Will output:

    15:52:11.244 | INFO    | prefect.engine - Created flow run 'emerald-gharial' for flow 'my-flow'\n15:52:11.812 | INFO    | Flow run 'emerald-gharial' - we're logging print statements from a flow\n15:52:11.926 | INFO    | Flow run 'emerald-gharial' - Created task run 'my_task-20c6ece6-0' for task 'my_task'\n15:52:11.927 | INFO    | Flow run 'emerald-gharial' - Executing 'my_task-20c6ece6-0' immediately...\n15:52:12.217 | INFO    | Task run 'my_task-20c6ece6-0' - we're logging print statements from a task\n
    from prefect import task, flow\n\n@task\ndef my_task(log_prints=False):\n    print(\"not logging print statements in this task\")\n\n@flow(log_prints=True)\ndef my_flow():\n    print(\"we're logging print statements from a flow\")\n    my_task()\n

    Using log_prints=False at the task level will output:

    15:52:11.244 | INFO    | prefect.engine - Created flow run 'emerald-gharial' for flow 'my-flow'\n15:52:11.812 | INFO    | Flow run 'emerald-gharial' - we're logging print statements from a flow\n15:52:11.926 | INFO    | Flow run 'emerald-gharial' - Created task run 'my_task-20c6ece6-0' for task 'my_task'\n15:52:11.927 | INFO    | Flow run 'emerald-gharial' - Executing 'my_task-20c6ece6-0' immediately...\nnot logging print statements in this task\n

    You can also configure this behavior globally for all Prefect flows, tasks, and subflows.

    prefect config set PREFECT_LOGGING_LOG_PRINTS=True\n
    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#formatters","title":"Formatters","text":"

    Prefect log formatters specify the format of log messages. You can see details of message formatting for different loggers in logging.yml. For example, the default formatting for task run log records is:

    \"%(asctime)s.%(msecs)03d | %(levelname)-7s | Task run %(task_run_name)r - %(message)s\"\n

    The variables available to interpolate in log messages varies by logger. In addition to the run context, message string, and any keyword arguments, flow and task run loggers have access to additional variables.

    The flow run logger has the following:

    • flow_run_name
    • flow_run_id
    • flow_name

    The task run logger has the following:

    • task_run_id
    • flow_run_id
    • task_run_name
    • task_name
    • flow_run_name
    • flow_name

    You can specify custom formatting by setting an environment variable or by modifying the formatter in a logging.yml file as described earlier. For example, to change the formatting for the flow runs formatter:

    PREFECT_LOGGING_FORMATTERS_STANDARD_FLOW_RUN_FMT=\"%(asctime)s.%(msecs)03d | %(levelname)-7s | %(flow_run_id)s - %(message)s\"\n

    The resulting messages, using the flow run ID instead of name, would look like this:

    10:40:01.211 | INFO    | e43a5a80-417a-41c4-a39e-2ef7421ee1fc - Created task run\n'othertask-1c085beb-3' for task 'othertask'\n
    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#styles","title":"Styles","text":"

    By default, Prefect highlights specific keywords in the console logs with a variety of colors.

    Highlighting can be toggled on/off with the PREFECT_LOGGING_COLORS setting, e.g.

    PREFECT_LOGGING_COLORS=False\n

    You can change what gets highlighted and also adjust the colors by updating the styles in a logging.yml file. Below lists the specific keys built-in to the PrefectConsoleHighlighter.

    URLs:

    • log.web_url
    • log.local_url

    Log levels:

    • log.info_level
    • log.warning_level
    • log.error_level
    • log.critical_level

    State types:

    • log.pending_state
    • log.running_state
    • log.scheduled_state
    • log.completed_state
    • log.cancelled_state
    • log.failed_state
    • log.crashed_state

    Flow (run) names:

    • log.flow_run_name
    • log.flow_name

    Task (run) names:

    • log.task_run_name
    • log.task_name

    You can also build your own handler with a custom highlighter. For example, to additionally highlight emails:

    1. Copy and paste the following into my_package_or_module.py (rename as needed) in the same directory as the flow run script, or ideally part of a Python package so it's available in site-packages to be accessed anywhere within your environment.
    import logging\nfrom typing import Dict, Union\n\nfrom rich.highlighter import Highlighter\n\nfrom prefect.logging.handlers import PrefectConsoleHandler\nfrom prefect.logging.highlighters import PrefectConsoleHighlighter\n\nclass CustomConsoleHighlighter(PrefectConsoleHighlighter):\n    base_style = \"log.\"\n    highlights = PrefectConsoleHighlighter.highlights + [\n        # ?P<email> is naming this expression as `email`\n        r\"(?P<email>[\\w-]+@([\\w-]+\\.)+[\\w-]+)\",\n    ]\n\nclass CustomConsoleHandler(PrefectConsoleHandler):\n    def __init__(\n        self,\n        highlighter: Highlighter = CustomConsoleHighlighter,\n        styles: Dict[str, str] = None,\n        level: Union[int, str] = logging.NOTSET,\n   ):\n        super().__init__(highlighter=highlighter, styles=styles, level=level)\n
    1. Update /.prefect/logging.yml to use my_package_or_module.CustomConsoleHandler and additionally reference the base_style and named expression: log.email.
        console_flow_runs:\n        level: 0\n        class: my_package_or_module.CustomConsoleHandler\n        formatter: flow_runs\n        styles:\n            log.email: magenta\n            # other styles can be appended here, e.g.\n            # log.completed_state: green\n
    1. Then on your next flow run, text that looks like an email will be highlighted--e.g. my@email.com is colored in magenta here.
    from prefect import flow, get_run_logger\n\n@flow\ndef log_email_flow():\n    logger = get_run_logger()\n    logger.info(\"my@email.com\")\n\nlog_email_flow()\n
    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#applying-markup-in-logs","title":"Applying markup in logs","text":"

    To use Rich's markup in Prefect logs, first configure PREFECT_LOGGING_MARKUP.

    PREFECT_LOGGING_MARKUP=True\n

    Then, the following will highlight \"fancy\" in red.

    from prefect import flow, get_run_logger\n\n@flow\ndef my_flow():\n    logger = get_run_logger()\n    logger.info(\"This is [bold red]fancy[/]\")\n\nmy_flow()\n

    Inaccurate logs could result

    Although this can be convenient, the downside is, if enabled, strings that contain square brackets may be inaccurately interpreted and lead to incomplete output, e.g. DROP TABLE [dbo].[SomeTable];\" outputs DROP TABLE .[SomeTable];.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#log-database-schema","title":"Log database schema","text":"

    Logged events are also persisted to the Prefect database. A log record includes the following data:

    Column Description id Primary key ID of the log record. created Timestamp specifying when the record was created. updated Timestamp specifying when the record was updated. name String specifying the name of the logger. level Integer representation of the logging level. flow_run_id ID of the flow run associated with the log record. If the log record is for a task run, this is the parent flow of the task. task_run_id ID of the task run associated with the log record. Null if logging a flow run event. message Log message. timestamp The client-side timestamp of this logged statement.

    For more information, see Log schema in the API documentation.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/logs/#including-logs-from-other-libraries","title":"Including logs from other libraries","text":"

    By default, Prefect won't capture log statements from libraries that your flows and tasks use. You can tell Prefect to include logs from these libraries with the PREFECT_LOGGING_EXTRA_LOGGERS setting.

    To use this setting, specify one or more Python library names to include, separated by commas. For example, if you want to make sure Prefect captures Dask and SciPy logging statements with your flow and task run logs:

    PREFECT_LOGGING_EXTRA_LOGGERS=dask,scipy\n

    You can set this setting as an environment variable or in a profile. See Settings for more details about how to use settings.

    ","tags":["UI","dashboard","Prefect Cloud","flows","tasks","logging","log formatters","configuration","debug"],"boost":2},{"location":"guides/managed-execution/","title":"Managed Execution","text":"

    Prefect Cloud can run your flows on your behalf with prefect:managed work pools. Flows run with this work pool do not require a worker or cloud provider account. Prefect handles the infrastructure and code execution for you.

    Managed execution is a great option for users who want to get started quickly, with no infrastructure setup.

    Managed Execution is in beta

    Managed Execution is currently in beta. Features are likely to change without warning.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#usage-guide","title":"Usage guide","text":"

    Run a flow with managed infrastructure in three steps.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#step-1","title":"Step 1","text":"

    Create a new work pool of type prefect:managed. you can do this via the UI wizard, or via the CLI

    prefect work-pool create my-managed-pool --type prefect:managed\n
    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#step-2","title":"Step 2","text":"

    Create a deployment using the flow deploy method or prefect.yaml.

    Specify the name of your managed work pool, as shown in this example that uses the deploy method:

    managed-execution.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n    source=\"https://github.com/desertaxle/demo.git\",\n    entrypoint=\"flow.py:my_flow\",\n    ).deploy(\n        name=\"test-managed-flow\",\n        work_pool_name=\"my-managed-pool\",\n    )\n

    With your CLI authenticated to your Prefect Cloud workspace, run the script to create your deployment:

    python managed-execution.py\n

    Note that this deployment uses flow code stored in a GitHub repository.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#step-3","title":"Step 3","text":"

    Run the deployment from the UI or from the CLI.

    That's it! You ran a flow on remote infrastructure without any infrastructure setup, worker, or cloud provider account.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#adding-dependencies","title":"Adding dependencies","text":"

    You can install Python package dependencies at runtime by configuring job_variables={\"pip_packages\": [\"pandas\", \"prefect-aws\"]} like this:

    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n    source=\"https://github.com/desertaxle/demo.git\",\n    entrypoint=\"flow.py:my_flow\",\n    ).deploy(\n        name=\"test-managed-flow\",\n        work_pool_name=\"my-managed-pool\",\n        job_variables={\"pip_packages\": [\"pandas\", \"prefect-aws\"]}\n    )\n

    Alternatively, you can specify a requirements.txt file and reference it in your prefect.yaml pull_step.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#limitations","title":"Limitations","text":"

    Managed execution requires Prefect 2.14.4 or newer.

    All limitations listed below may change without warning during the beta period. We will update this page as we make changes.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#concurrency-work-pools","title":"Concurrency & work pools","text":"

    Free tier accounts are limited to:

    • Maximum of 1 concurrent flow run per workspace across all prefect:managed pools.
    • Maximum of 1 managed execution work pool per workspace.

    Pro tier and above accounts are limited to:

    • Maximum of 10 concurrent flow runs per workspace across all prefect:managed pools.
    • Maximum of 5 managed execution work pools per workspace.
    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#images","title":"Images","text":"

    At this time, managed execution requires that you run the official Prefect Docker image: prefecthq/prefect:2-latest. However, as noted above, you can install Python package dependencies at runtime. If you need to use your own image, we recommend using another type of work pool.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#code-storage","title":"Code storage","text":"

    Flow code must be stored in an accessible remote location. This means git-based cloud providers such as GitHub, Bitbucket, or GitLab are supported. Remote block-based storage is also supported, so S3, GCS, and Azure Blob are additional code storage options.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#resources","title":"Resources","text":"

    Memory is limited to 2GB of RAM, which includes all operations such as dependency installation. Maximum job run time is 24 hours.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#usage-limits","title":"Usage limits","text":"

    Free tier accounts are limited to ten compute hours per workspace per month. Pro tier and above accounts are limited to 250 hours per workspace per month. you can view your compute hours quota usage on the work pools page.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/managed-execution/#next-steps","title":"Next steps","text":"

    Read more about creating deployments in the deployment guide.

    If you find that you need more control over your infrastructure, such as the ability to run custom Docker images, serverless push work pools might be a good option. Read more here.

    ","tags":["managed infrastructure","infrastructure"],"boost":2},{"location":"guides/migration-guide/","title":"Migrating from Prefect 1 to Prefect 2","text":"

    This guide is designed to help you migrate your workflows from Prefect 1 to Prefect 2.

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#what-stayed-the-same","title":"What stayed the same","text":"

    Prefect 2 still:

    • Has tasks and flows.
    • Orchestrates your flow runs and provides observability into their execution states.
    • Runs and inspects flow runs locally.
    • Provides a coordination plane for your dataflows based on the same principles.
    • Employs the same hybrid execution model, where Prefect doesn't store your flow code or data.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#what-changed","title":"What changed","text":"

    Prefect 2 requires modifications to your existing tasks, flows, and deployment patterns. We've organized this section into the following categories:

    • Simplified patterns \u2014 abstractions from Prefect 1 that are no longer necessary in the dynamic, DAG-free Prefect workflows that support running native Python code in your flows.
    • Conceptual and syntax changes that often clarify names and simplify familiar abstractions such as retries and caching.
    • New features enabled by the dynamic and flexible Prefect API.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#simplified-patterns","title":"Simplified patterns","text":"

    Since Prefect 2 allows running native Python code within the flow function, some abstractions are no longer necessary:

    • Parameter tasks: in Prefect 2, inputs to your flow function are automatically treated as parameters of your flow. You can define the parameter values in your flow code when you create your Deployment, or when you schedule an ad-hoc flow run. One benefit of Prefect parametrization is built-in type validation with pydantic.
    • Task-level state_handlers: in Prefect 2, you can build custom logic that reacts to task-run states within your flow function without the need for state_handlers. The page \" How to take action on a state change of a task run\" provides a further explanation and code examples.
    • Instead of using signals, Prefect 2 allows you to raise an arbitrary exception in your task or flow and return a custom state. For more details and examples, see How can I stop the task run based on a custom logic.
    • Conditional tasks such as case are no longer required. Use Python native if...else statements to build a conditional logic. The Discourse tag \"conditional-logic\" provides more resources.
    • Since you can use any context manager directly in your flow, a resource_manager is no longer necessary. As long as you point to your flow script in your Deployment, you can share database connections and any other resources between tasks in your flow. The Discourse page How to clean up resources used in a flow provides a full example.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#conceptual-and-syntax-changes","title":"Conceptual and syntax changes","text":"

    The changes listed below require you to modify your workflow code. The following table shows how Prefect 1 concepts have been implemented in Prefect 2. The last column contains references to additional resources that provide more details and examples.

    Concept Prefect 1 Prefect 2 Reference links Flow definition. with Flow(\"flow_name\") as flow: @flow(name=\"flow_name\") How can I define a flow? Flow executor that determines how to execute your task runs. Executor such as LocalExecutor. Task runner such as ConcurrentTaskRunner. What is the default TaskRunner (executor)? Configuration that determines how and where to execute your flow runs. Run configuration such as flow.run_config = DockerRun(). Create an infrastructure block such as a Docker Container and specify it as the infrastructure when creating a deployment. How can I run my flow in a Docker container? Assignment of schedules and default parameter values. Schedules are attached to the flow object and default parameter values are defined within the Parameter tasks. Schedules and default parameters are assigned to a flow\u2019s Deployment, rather than to a Flow object. How can I attach a schedule to a flow? Retries @task(max_retries=2, retry_delay=timedelta(seconds=5)) @task(retries=2, retry_delay_seconds=5) How can I specify the retry behavior for a specific task? Logger syntax. Logger is retrieved from prefect.context and can only be used within tasks. In Prefect 2, you can log not only from tasks, but also within flows. To get the logger object, use: prefect.get_run_logger(). How can I add logs to my flow? The syntax and contents of Prefect context. Context is a thread-safe way of accessing variables related to the flow run and task run. The syntax to retrieve it: prefect.context. Context is still available, but its content is much richer, allowing you to retrieve even more information about your flow runs and task runs. The syntax to retrieve it: prefect.context.get_run_context(). How to access Prefect context values? Task library. Included in the main Prefect Core repository. Separated into individual repositories per system, cloud provider, or technology. How to migrate Prefect 1 tasks to Prefect 2 integrations.","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#what-changed-in-dataflow-orchestration","title":"What changed in dataflow orchestration?","text":"

    Let\u2019s look at the differences in how Prefect 2 transitions your flow and task runs between various execution states.

    • In Prefect 2, the final state of a flow run that finished without errors is Completed, while in Prefect 1, this flow run has a Success state. You can find more about that topic here.
    • The decision about whether a flow run should be considered successful or not is no longer based on special reference tasks. Instead, your flow\u2019s return value determines the final state of a flow run. This link provides a more detailed explanation with code examples.
    • In Prefect 1, concurrency limits were only available to Prefect Cloud users. Prefect 2 provides customizable concurrency limits with the open-source Prefect server and Prefect Cloud. In Prefect 2, flow run concurrency limits are set on work pools.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#what-changed-in-flow-deployment-patterns","title":"What changed in flow deployment patterns?","text":"

    To deploy your Prefect 1 flows, you have to send flow metadata to the backend in a step called registration. Prefect 2 no longer requires flow pre-registration. Instead, you create a Deployment that specifies the entry point to your flow code and optionally specifies:

    • Where to run your flow (your Infrastructure, such as a DockerContainer, KubernetesJob, or ECSTask).
    • When to run your flow (an Interval, Cron, or RRule schedule).
    • How to run your flow (execution details such as parameters, flow deployment name, and more).
    • The work pool for your deployment. If no work pool is specified, a default work pool named default is used.

    The API is now implemented as a REST API rather than GraphQL. This page illustrates how you can interact with the API.

    In Prefect 1, the logical grouping of flows was based on projects. Prefect 2 provides a much more flexible way of organizing your flows, tasks, and deployments through customizable filters and\u00a0tags. This page provides more details on how to assign tags to various Prefect 2 objects.

    The role of agents has changed:

    • In Prefect 2, there is only one generic agent type. The agent polls a work pool looking for flow runs.
    • See this Discourse page for a more detailed discussion.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#new-features-introduced-in-prefect-2","title":"New features introduced in Prefect 2","text":"

    The following new components and capabilities are enabled by Prefect 2.

    • More flexibility thanks to the elimination of flow pre-registration.
    • More flexibility for flow deployments, including easier promotion of a flow through development, staging, and production environments.
    • Native async support.
    • Out-of-the-box pydantic validation.
    • Blocks allowing you to securely store UI-editable, type-checked configuration to external systems and an easy-to-use Key-Value Store. All those components are configurable in one place and provided as part of the open-source Prefect 2 product. In contrast, the concept of Secrets in Prefect 1 was much more narrow and only available in Prefect Cloud.
    • Notifications available in the open-source Prefect 2 version, as opposed to Cloud-only Automations in Prefect 1.
    • A first-class subflows concept: Prefect 1 only allowed the flow-of-flows orchestrator pattern. With Prefect 2 subflows, you gain a natural and intuitive way of organizing your flows into modular sub-components. For more details, see the following list of resources about subflows.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#orchestration-behind-the-api","title":"Orchestration behind the API","text":"

    Apart from new features, Prefect 2 simplifies many usage patterns and provides a much more seamless onboarding experience.

    Every time you run a flow, whether it is tracked by the API server or ad-hoc through a Python script, it is on the same UI page for easier debugging and observability.

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#code-as-workflows","title":"Code as workflows","text":"

    With Prefect 2, your functions\u00a0are\u00a0your flows and tasks. Prefect 2 automatically detects your flows and tasks without the need to define a rigid DAG structure. While use of tasks is encouraged to provide you the maximum visibility into your workflows, they are no longer required. You can add a single @flow decorator to your main function to transform any Python script into a Prefect workflow.

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#incremental-adoption","title":"Incremental adoption","text":"

    The built-in SQLite database automatically tracks all your locally executed flow runs. As soon as you start a Prefect server and open the Prefect UI in your browser (or authenticate your CLI with your Prefect Cloud workspace), you can see all your locally executed flow runs in the UI. You don't even need to start an agent.

    Then, when you want to move toward scheduled, repeatable workflows, you can build a deployment and send it to the server by running a CLI command or a Python script.

    • You can create a deployment to on remote infrastructure, where the run environment is defined by a reusable infrastructure block.
    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#fewer-ambiguities","title":"Fewer ambiguities","text":"

    Prefect 2 eliminates ambiguities in many ways. For example. there is no more confusion between Prefect Core and Prefect Server \u2014 Prefect 2 unifies those into a single open source product. This product is also much easier to deploy with no requirement for Docker or docker-compose.

    If you want to switch your backend to use Prefect Cloud for an easier production-level managed experience, Prefect profiles let you quickly connect to your workspace.

    In Prefect 1, there are several confusing ways you could implement caching. Prefect 2 resolves those ambiguities by providing a single cache_key_fn function paired with cache_expiration, allowing you to define arbitrary caching mechanisms \u2014 no more confusion about whether you need to use cache_for, cache_validator, or file-based caching using targets.

    For more details on how to configure caching, check out the following resources:

    • Caching docs
    • Time-based caching
    • Input-based caching

    A similarly confusing concept in Prefect 1 was distinguishing between the functional and imperative APIs. This distinction caused ambiguities with respect to how to define state dependencies between tasks. Prefect 1 users were often unsure whether they should use the functional upstream_tasks keyword argument or the imperative methods such as task.set_upstream(), task.set_downstream(), or flow.set_dependencies(). In Prefect 2, there is only the functional API.

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/migration-guide/#next-steps","title":"Next steps","text":"

    We know migrations can be tough. We encourage you to take it step-by-step and experiment with the new features.

    To make the migration process easier for you:

    • We provided a detailed FAQ section allowing you to find the right information you need to move your workflows to Prefect 2. If you still have some open questions, feel free to create a new topic describing your migration issue.
    • We have dedicated resources in the Customer Success team to help you along your migration journey. Reach out to cs@prefect.io to discuss how we can help.
    • You can ask questions in our 20,000+ member Community Slack.

    Happy Engineering!

    ","tags":["migration","upgrading","best practices"],"boost":2},{"location":"guides/moving-data/","title":"Read and Write Data to and From Cloud Provider Storage","text":"

    Writing data to cloud-based storage and reading data from that storage is a common task in data engineering. In this guide we'll learn how to use Prefect to move data to and from AWS, Azure, and GCP blob storage.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#prerequisites","title":"Prerequisites","text":"
    • Prefect installed
    • Authenticated with Prefect Cloud (or self-hosted Prefect server instance)
    • A cloud provider account (e.g. AWS)
    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#install-relevant-prefect-integration-library","title":"Install relevant Prefect integration library","text":"

    In the CLI, install the Prefect integration library for your cloud provider:

    AWSAzureGCP

    prefect-aws provides blocks for interacting with AWS services.

    pip install -U prefect-aws\n

    prefect-azure provides blocks for interacting with Azure services.

     pip install -U prefect-azure\n

    prefect-gcp provides blocks for interacting with GCP services.

     pip install -U prefect-gcp\n
    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#register-the-block-types","title":"Register the block types","text":"

    Register the new block types with Prefect Cloud (or with your self-hosted Prefect server instance):

    AWSAzureGCP
    prefect block register -m prefect_aws  \n
    prefect block register -m prefect_azure \n
    prefect block register -m prefect_gcp\n

    We should see a message in the CLI that several block types were registered. If we check the UI, we should see the new block types listed.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#create-a-storage-bucket","title":"Create a storage bucket","text":"

    Create a storage bucket in the cloud provider account. Ensure the bucket is publicly accessible or create a user or service account with the appropriate permissions to fetch and write data to the bucket.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#create-a-credentials-block","title":"Create a credentials block","text":"

    If the bucket is private, there are several options to authenticate:

    1. At deployment runtime, ensure the runtime environment is authenticated.
    2. Create a block with configuration details and reference it when creating the storage block.

    If saving credential details in a block we can use a credentials block specific to the cloud provider or use a more generic secret block. We can create blocks via the UI or Python code. Below we'll use Python code to create a credentials block for our cloud provider.

    Credentials safety

    Reminder, don't store credential values in public locations such as public git platform repositories. In the examples below we use environment variables to store credential values.

    AWSAzureGCP
    import os\nfrom prefect_aws import AwsCredentials\n\nmy_aws_creds = AwsCredentials(\n    aws_access_key_id=\"123abc\",\n    aws_secret_access_key=os.environ.get(\"MY_AWS_SECRET_ACCESS_KEY\"),\n)\nmy_aws_creds.save(name=\"my-aws-creds-block\", overwrite=True)\n
    import os\nfrom prefect_azure import AzureBlobStorageCredentials\n\nmy_azure_creds = AzureBlobStorageCredentials(\n    connection_string=os.environ.get(\"MY_AZURE_CONNECTION_STRING\"),\n)\nmy_azure_creds.save(name=\"my-azure-creds-block\", overwrite=True)\n

    We recommend specifying the service account key file contents as a string, rather than the path to the file, because that file might not be available in your production environments.

    import os\nfrom prefect_gcp import GCPCredentials\n\nmy_gcp_creds = GCPCredentials(\n    service_account_info=os.environ.get(\"GCP_SERVICE_ACCOUNT_KEY_FILE_CONTENTS\"), \n)\nmy_gcp_creds.save(name=\"my-gcp-creds-block\", overwrite=True)\n

    Run the code to create the block. We should see a message that the block was created.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#create-a-storage-block","title":"Create a storage block","text":"

    Let's create a block for the chosen cloud provider using Python code or the UI. In this example we'll use Python code.

    AWSAzureGCP

    Note that the S3Bucket block is not the same as the S3 block that ships with Prefect. The S3Bucket block we use in this example is part of the prefect-aws library and provides additional functionality.

    We'll reference the credentials block created above.

    from prefect_aws import S3Bucket\n\ns3bucket = S3Bucket.create(\n    bucket=\"my-bucket-name\",\n    credentials=\"my-aws-creds-block\"\n    )\ns3bucket.save(name=\"my-s3-bucket-block\", overwrite=True)\n

    Note that the AzureBlobStorageCredentials block is not the same as the Azure block that ships with Prefect. The AzureBlobStorageCredentials block we use in this example is part of the prefect-azure library and provides additional functionality.

    Azure blob storage doesn't require a separate block, the connection string used in the AzureBlobStorageCredentials block can encode the information needed.

    Note that the GcsBucket block is not the same as the GCS block that ships with Prefect. The GcsBucket block is part of the prefect-gcp library and provides additional functionality. We'll use it here.

    We'll reference the credentials block created above.

    from prefect_gcp.cloud_storage import GcsBucket\n\ngcsbucket = GcsBucket(\n    bucket=\"my-bucket-name\", \n    credentials=\"my-gcp-creds-block\"\n    )\ngcsbucket.save(name=\"my-gcs-bucket-block\", overwrite=True)\n

    Run the code to create the block. We should see a message that the block was created.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#write-data","title":"Write data","text":"

    Use your new block inside a flow to write data to your cloud provider.

    AWSAzureGCP
    from pathlib import Path\nfrom prefect import flow\nfrom prefect_aws.s3 import S3Bucket\n\n@flow()\ndef upload_to_s3():\n    \"\"\"Flow function to upload data\"\"\"\n    path = Path(\"my_path_to/my_file.parquet\")\n    aws_block = S3Bucket.load(\"my-s3-bucket-block\")\n    aws_block.upload_from_path(from_path=path, to_path=path)\n\nif __name__ == \"__main__\":\n    upload_to_s3()\n
    from prefect import flow\nfrom prefect_azure import AzureBlobStorageCredentials\nfrom prefect_azure.blob_storage import blob_storage_upload\n\n@flow\ndef upload_to_azure():\n    \"\"\"Flow function to upload data\"\"\"\n    blob_storage_credentials = AzureBlobStorageCredentials.load(\n        name=\"my-azure-creds-block\"\n    )\n\n    with open(\"my_path_to/my_file.parquet\", \"rb\") as f:\n        blob_storage_upload(\n            data=f.read(),\n            container=\"my_container\",\n            blob=\"my_path_to/my_file.parquet\",\n            blob_storage_credentials=blob_storage_credentials,\n        )\n\nif __name__ == \"__main__\":\n    upload_to_azure()\n
    from pathlib import Path\nfrom prefect import flow\nfrom prefect_gcp.cloud_storage import GcsBucket\n\n@flow()\ndef upload_to_gcs():\n    \"\"\"Flow function to upload data\"\"\"\n    path = Path(\"my_path_to/my_file.parquet\")\n    gcs_block = GcsBucket.load(\"my-gcs-bucket-block\")\n    gcs_block.upload_from_path(from_path=path, to_path=path)\n\nif __name__ == \"__main__\":\n    upload_to_gcs()\n
    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#read-data","title":"Read data","text":"

    Use your block to read data from your cloud provider inside a flow.

    AWSAzureGCP
    from prefect import flow\nfrom prefect_aws import S3Bucket\n\n@flow\ndef download_from_s3():\n    \"\"\"Flow function to download data\"\"\"\n    s3_block = S3Bucket.load(\"my-s3-bucket-block\")\n    s3_block.get_directory(\n        from_path=\"my_path_to/my_file.parquet\", \n        local_path=\"my_path_to/my_file.parquet\"\n    )\n\nif __name__ == \"__main__\":\n    download_from_s3()\n
    from prefect import flow\nfrom prefect_azure import AzureBlobStorageCredentials\nfrom prefect_azure.blob_storage import blob_storage_download\n\n@flow\ndef download_from_azure():\n    \"\"\"Flow function to download data\"\"\"\n    blob_storage_credentials = AzureBlobStorageCredentials.load(\n        name=\"my-azure-creds-block\"\n    )\n    blob_storage_download(\n        blob=\"my_path_to/my_file.parquet\",\n        container=\"my_container\",\n        blob_storage_credentials=blob_storage_credentials,\n    )\n\nif __name__ == \"__main__\":\n    download_from_azure()\n
    from prefect import flow\nfrom prefect_gcp.cloud_storage import GcsBucket\n\n@flow\ndef download_from_gcs():\n    gcs_block = GcsBucket.load(\"my-gcs-bucket-block\")\n    gcs_block.get_directory(\n        from_path=\"my_path_to/my_file.parquet\", \n        local_path=\"my_path_to/my_file.parquet\"\n    )\n\nif __name__ == \"__main__\":\n    download_from_gcs()\n

    In this guide we've seen how to use Prefect to read data from and write data to cloud providers!

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/moving-data/#next-steps","title":"Next steps","text":"

    Check out the prefect-aws, prefect-azure, and prefect-gcp docs to see additional methods for interacting with cloud storage providers. Each library also contains blocks for interacting with other cloud-provider services.

    ","tags":["data","storage","read data","write data","cloud providers","AWS","S3","Azure Storage","Azure Blob Storage","Azure","GCP","Google Cloud Storage","GCS","moving data"],"boost":2},{"location":"guides/prefect-deploy/","title":"Deploying Flows to Work Pools and Workers","text":"

    In this guide, we will configure a deployment that uses a work pool for dynamically provisioned infrastructure.

    All Prefect flow runs are tracked by the API. The API does not require prior registration of flows. With Prefect, you can call a flow locally or on a remote environment and it will be tracked.

    A deployment turns your workflow into an application that can be interacted with and managed via the Prefect API. A deployment enables you to:

    • Schedule flow runs.
    • Specify event triggers for flow runs.
    • Assign one or more tags to organize your deployments and flow runs. You can use those tags as filters in the Prefect UI.
    • Assign custom parameter values for flow runs based on the deployment.
    • Create ad-hoc flow runs from the API or Prefect UI.
    • Upload flow files to a defined storage location for retrieval at run time.

    Deployments created with .serve

    A deployment created with the Python flow.serve method or the serve function runs flows in a subprocess on the same machine where the deployment is created. It does not use a work pool or worker.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#work-pool-based-deployments","title":"Work pool-based deployments","text":"

    A work pool-based deployment is useful when you want to dynamically scale the infrastructure where your flow code runs. Work pool-based deployments contain information about the infrastructure type and configuration for your workflow execution.

    Work pool-based deployment infrastructure options include the following:

    • Process - runs flow in a subprocess. In most cases, you're better off using .serve.
    • Docker - runs flows in an ephemeral Docker container.
    • Kubernetes - runs flows as a Kubernetes Job.
    • Serverless Cloud Provider options - runs flows in a Docker container in a serverless cloud provider environment, such as AWS ECS, Azure Container Instance, Google Cloud Run, or Vertex AI.

    The following diagram provides a high-level overview of the conceptual elements involved in defining a work-pool based deployment that is polled by a worker and executes a flow run based on that deployment.

    %%{\n  init: {\n    'theme': 'base',\n    'themeVariables': {\n      'fontSize': '19px'\n    }\n  }\n}%%\n\nflowchart LR\n    F(\"<div style='margin: 5px 10px 5px 5px;'>Flow Code</div>\"):::yellow -.-> A(\"<div style='margin: 5px 10px 5px 5px;'>Deployment Definition</div>\"):::gold\n    subgraph Server [\"<div style='width: 150px; text-align: center; margin-top: 5px;'>Prefect API</div>\"]\n        D(\"<div style='margin: 5px 10px 5px 5px;'>Deployment</div>\"):::green\n    end\n    subgraph Remote Storage [\"<div style='width: 160px; text-align: center; margin-top: 5px;'>Remote Storage</div>\"]\n        B(\"<div style='margin: 5px 6px 5px 5px;'>Flow</div>\"):::yellow\n    end\n    subgraph Infrastructure [\"<div style='width: 150px; text-align: center; margin-top: 5px;'>Infrastructure</div>\"]\n        G(\"<div style='margin: 5px 10px 5px 5px;'>Flow Run</div>\"):::blue\n    end\n\n    A --> D\n    D --> E(\"<div style='margin: 5px 10px 5px 5px;'>Worker</div>\"):::red\n    B -.-> E\n    A -.-> B\n    E -.-> G\n\n    classDef gold fill:goldenrod,stroke:goldenrod,stroke-width:4px,color:black\n    classDef yellow fill:gold,stroke:gold,stroke-width:4px,color:black\n    classDef gray fill:lightgray,stroke:lightgray,stroke-width:4px\n    classDef blue fill:blue,stroke:blue,stroke-width:4px,color:white\n    classDef green fill:green,stroke:green,stroke-width:4px,color:white\n    classDef red fill:red,stroke:red,stroke-width:4px,color:white\n    classDef dkgray fill:darkgray,stroke:darkgray,stroke-width:4px,color:white

    The work pool types above require a worker to be running on your infrastructure to poll a work pool for scheduled flow runs.

    Additional work pool options available with Prefect Cloud

    Prefect Cloud offers other flavors of work pools that don't require a worker:

    • Push Work Pools - serverless cloud options that don't require a worker because Prefect Cloud submits them to your serverless cloud infrastructure on your behalf. Prefect can auto-provision your cloud infrastructure for you and set it up to use your work pool.

    • Managed Execution Prefect Cloud submits and runs your deployment on serverless infrastructure. No cloud provider account required.

    In this guide, we focus on deployments that require a worker.

    Work pool-based deployments that use a worker also allow you to assign a work queue name to prioritize work and allow you to limit concurrent runs at the work pool level.

    When creating a deployment that uses a work pool and worker, we must answer two basic questions:

    • What instructions does a worker need to set up an execution environment for our flow? For example, a flow may have Python package requirements, unique Kubernetes settings, or Docker networking configuration.
    • How should the flow code be accessed?
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#creating-work-pool-based-deployments","title":"Creating work pool-based deployments","text":"

    The tutorial shows how you can create a deployment with a long-running process using .serve and how to move to a work-pool-based deployment setup with .deploy. See the discussion of when you might want to move to work-pool-based deployments there.

    In this guide, we show how to use .deploy in more depth and discuss prefect.yaml, a YAML-based alternative for managing deployments.

    Use the tabs below to explore these two deployment creation options.

    .deployprefect.yaml

    The prefect.yaml file is a YAML file describing base settings for your deployments, procedural steps for preparing deployments, and instructions for preparing the execution environment for a deployment run.

    You can initialize your deployment configuration, which creates the prefect.yaml file, by running the CLI command prefect init in any directory or repository that stores your flow code.

    Deployment configuration recipes

    Prefect ships with many off-the-shelf \"recipes\" that allow you to get started with more structure within your prefect.yaml file; run prefect init to be prompted with available recipes in your installation. You can provide a recipe name in your initialization command with the --recipe flag, otherwise Prefect will attempt to guess an appropriate recipe based on the structure of your working directory (for example if you initialize within a git repository, Prefect will use the git recipe).

    The prefect.yaml file contains deployment configuration for deployments created from this file, default instructions for how to build and push any necessary code artifacts (such as Docker images), and default instructions for pulling a deployment in remote execution environments (e.g., cloning a GitHub repository).

    Any deployment configuration can be overridden via options available on the prefect deploy CLI command when creating a deployment.

    The base structure for prefect.yaml is as follows:

    # generic metadata\nprefect-version: null\nname: null\n\n# preparation steps\nbuild: null\npush: null\n\n# runtime steps\npull: null\n\n# deployment configurations\ndeployments:\n- # base metadata\n    name: null\n    version: null\n    tags: []\n    description: null\n    schedule: null\n\n    # flow-specific fields\n    entrypoint: null\n    parameters: {}\n\n    # infra-specific fields\n    work_pool:\n    name: null\n    work_queue_name: null\n    job_variables: {}\n

    The metadata fields are always pre-populated for you. These fields are for bookkeeping purposes only. The other sections are pre-populated based on recipe; if no recipe is provided, Prefect will attempt to guess an appropriate one based on local configuration.

    You can create deployments via the CLI command prefect deploy without ever needing to alter the deployments section of your prefect.yaml file \u2014 the prefect deploy command will help in deployment creation via interactive prompts. The prefect.yaml file facilitates version-controlling your deployment configuration and managing multiple deployments.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#automatically-bake-your-code-into-a-docker-image","title":"Automatically bake your code into a Docker image","text":"

    You can create a deployment from Python code by calling the .deploy method on a flow.

    buy.py
    from prefect import flow\n\n\n@flow(log_prints=True)\ndef buy():\n    print(\"Buying securities\")\n\n\nif __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-code-baked-into-an-image-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"my_registry/my_image:my_image_tag\"\n    )\n

    Make sure you have the work pool created in the Prefect Cloud workspace you are authenticated to or on your running self-hosted server instance. Then run the script to create a deployment (in future examples this step will be omitted for brevity):

    python buy.py\n

    You should see messages in your terminal that Docker is building your image. When the deployment build succeeds you will see helpful information in your terminal showing you how to start a worker for your deployment and how to run your deployment. Your deployment will be visible on the Deployments page in the UI.

    By default, .deploy will build a Docker image with your flow code baked into it and push the image to the Docker Hub registry specified in the image argument`.

    Authentication to Docker Hub

    You need your environment to be authenticated to your Docker registry to push an image to it.

    You can specify a registry other than Docker Hub by providing the full registry path in the image argument.

    Warning

    If building a Docker image, the environment in which you are creating the deployment needs to have Docker installed and running.

    To avoid pushing to a registry, set push=False in the .deploy method.

    if __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-code-baked-into-an-image-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"my_registry/my_image:my_image_tag\",\n        push=False\n    )\n

    To avoid building an image, set build=False in the .deploy method.

    if __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-code-baked-into-an-image-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"discdiver/no-build-image:1.0\",\n        build=False\n    )\n

    The specified image will need to be available in your deployment's execution environment for your flow code to be accessible.

    Prefect generates a Dockerfile for you that will build an image based off of one of Prefect's published images. The generated Dockerfile will copy the current directory into the Docker image and install any dependencies listed in a requirements.txt file.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#automatically-build-a-custom-docker-image-with-a-local-dockerfile","title":"Automatically build a custom Docker image with a local Dockerfile","text":"

    If you want to use a custom Dockerfile, you can specify the path to the Dockerfile with the DeploymentImage class:

    custom_dockerfile.py
    from prefect import flow\nfrom prefect.deployments import DeploymentImage\n\n\n@flow(log_prints=True)\ndef buy():\n    print(\"Selling securities\")\n\n\nif __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-custom-dockerfile-deployment\",\", \n        work_pool_name=\"my-docker-pool\", \n        image=DeploymentImage(\n            name=\"my_image\",\n            tag=\"deploy-guide\",\n            dockerfile=\"Dockerfile\"\n    ),\n    push=False\n)\n

    The DeploymentImage object allows for a great deal of image customization.

    For example, you can install a private Python package from GCP's artifact registry like this:

    Create a custom base Dockerfile.

    FROM python:3.10\n\nARG AUTHED_ARTIFACT_REG_URL\nCOPY ./requirements.txt /requirements.txt\n\nRUN pip install --extra-index-url ${AUTHED_ARTIFACT_REG_URL} -r /requirements.txt\n

    Create our deployment by leveraging the DeploymentImage class.

    private-package.py
    from prefect import flow\nfrom prefect.deployments.runner import DeploymentImage\nfrom prefect.blocks.system import Secret\nfrom my_private_package import do_something_cool\n\n\n@flow(log_prints=True)\ndef my_flow():\n    do_something_cool()\n\n\nif __name__ == \"__main__\":\n    artifact_reg_url: Secret = Secret.load(\"artifact-reg-url\")\n\n    my_flow.deploy(\n        name=\"my-deployment\",\n        work_pool_name=\"k8s-demo\",\n        image=DeploymentImage(\n            name=\"my-image\",\n            tag=\"test\",\n            dockerfile=\"Dockerfile\",\n            buildargs={\"AUTHED_ARTIFACT_REG_URL\": artifact_reg_url.get()},\n        ),\n    )\n

    Note that we used a Prefect Secret block to load the URL configuration for the artifact registry above.

    See all the optional keyword arguments for the DeploymentImage class here.

    Default Docker namespace

    You can set the PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE setting to append a default Docker namespace to all images you build with .deploy. This is great if you use a private registry to store your images.

    To set a default Docker namespace for your current profile run:

    prefect config set PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE=<docker-registry-url>/<organization-or-username>\n

    Once set, you can omit the namespace from your image name when creating a deployment:

    with_default_docker_namespace.py
    if __name__ == \"__main__\":\n    buy.deploy(\n        name=\"my-code-baked-into-an-image-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"my_image:my_image_tag\"\n    )\n

    The above code will build an image with the format <docker-registry-url>/<organization-or-username>/my_image:my_image_tag when PREFECT_DEFAULT_DOCKER_BUILD_NAMESPACE is set.

    While baking code into Docker images is a popular deployment option, many teams decide to store their workflow code in git-based storage, such as GitHub, Bitbucket, or Gitlab. Let's see how to do that next.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#store-your-code-in-git-based-cloud-storage","title":"Store your code in git-based cloud storage","text":"

    If you don't specify an image argument for .deploy, then you need to specify where to pull the flow code from at runtime with the from_source method.

    Here's how we can pull our flow code from a GitHub repository.

    git_storage.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        \"https://github.com/my_github_account/my_repo/my_file.git\",\n        entrypoint=\"flows/no-image.py:hello_world\",\n    ).deploy(\n        name=\"no-image-deployment\",\n        work_pool_name=\"my_pool\",\n        build=False\n    )\n

    The entrypoint is the path to the file the flow is located in and the function name, separated by a colon.

    Alternatively, you could specify a git-based cloud storage URL for a Bitbucket or Gitlab repository.

    Note

    If you don't specify an image as part of your deployment creation, the image specified in the work pool will be used to run your flow.

    After creating a deployment you might change your flow code. Generally, you can just push your code to GitHub, without rebuilding your deployment. The exception is if something that the server needs to know about changes, such as the flow entrypoint parameters. Rerunning the Python script with .deploy will update your deployment on the server with the new flow code.

    If you want to pull your flow code from private git-based storage,

    If you need to provide additional configuration, such as specifying a private repository, you can provide a GitRepository object instead of a URL:

    private_git_storage.py
    from prefect import flow\nfrom prefect.runner.storage import GitRepository\nfrom prefect.blocks.system import Secret\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=GitRepository(\n        url=\"https://github.com/org/private-repo.git\",\n        branch=\"dev\",\n        credentials={\n            \"access_token\": Secret.load(\"github-access-token\")\n        }\n    ),\n    entrypoint=\"flows/no-image.py:hello_world\",\n    ).deploy(\n        name=\"private-git-storage-deployment\",\n        work_pool_name=\"my_pool\",\n        build=False\n    )\n

    Note the use of the Secret block to load the GitHub access token. Alternatively, you could provide a username and password to the username and password fields of the credentials argument.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#store-your-code-in-cloud-provider-storage","title":"Store your code in cloud provider storage","text":"

    Another option for flow code storage is any fsspec-supported storage location, such as AWS S3, GCP GCS, or Azure Blob Storage.

    For example, you can pass the S3 bucket path to source.

    s3_storage.py
    from prefect import flow\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=\"s3://my-bucket/my-folder\",\n        entrypoint=\"flows.py:my_flow\",\n    ).deploy(\n        name=\"deployment-from-aws-flow\",\n        work_pool=\"my_pool\",\n    )\n

    In the example above your credentials will be auto-discovered from your deployment creation environment and credentials will need to be available in your runtime environment.

    If you need additional configuration for your cloud-based storage - for example, with a private S3 Bucket - we recommend using a storage block. A storage block also ensures your credentials will be available in both your deployment creation environment and your execution environment.

    Here's an example that uses an S3Bucket block from the prefect-aws library.

    s3_storage_auth.py
    from prefect import flow\nfrom prefect_aws.s3 import S3Bucket\n\nif __name__ == \"__main__\":\n    flow.from_source(\n        source=S3Bucket.load(\"my-code-storage\"), entrypoint=\"my_file.py:my_flow\"\n    ).deploy(name=\"test-s3\", work_pool=\"my_pool\")\n

    If you are familiar with the deployment creation mechanics with .serve, you will notice that .deploy is very similar. .deploy just requires a work pool name and has a number of parameters dealing with flow-code storage for Docker images.

    Unlike .serve, if you don't specify an image to use for your flow, you must to specify where to pull the flow code from at runtime with the from_source method, whereas from_source is optional with .serve.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#additional-configuration-with-deploy","title":"Additional configuration with .deploy","text":"

    Our examples thus far have explored options for where to store flow code. Let's turn our attention to other deployment configuration options.

    To pass parameters to your flow, you can use the parameters argument in the .deploy method. Just pass in a dictionary of key-value pairs.

    pass_params.py
    from prefect import flow\n\n@flow\ndef hello_world(name: str):\n    print(f\"Hello, {name}!\")\n\nif __name__ == \"__main__\":\n    hello_world.deploy(\n        name=\"pass-params-deployment\",\n        work_pool_name=\"my_pool\",\n        parameters=dict(name=\"Prefect\"),\n        image=\"my_registry/my_image:my_image_tag\",\n    )\n

    The job_variables parameter allows you to fine-tune the infrastructure settings for a deployment. The values passed in override default values in the specified work pool's base job template.

    You can override environment variables, such as image_pull_policy and image, for a specific deployment with the job_variables argument.

    job_var_image_pull.py
    if __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-deployment-never-pull\", \n        work_pool_name=\"my-docker-pool\", \n        job_variables={\"image_pull_policy\": \"Never\"},\n        image=\"my-image:my-tag\"\",\n        push=False\n    )\n

    Similarly, you can override the environment variables specified in a work pool through the job_variables parameter:

    job_var_env_vars.py
    if __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-deployment-never-pull\", \n        work_pool_name=\"my-docker-pool\", \n        job_variables={\"env\": {\"EXTRA_PIP_PACKAGES\": \"boto3\"} },\n        image=\"my-image:my-tag\"\",\n        push=False\n    )\n

    The dictionary key \"EXTRA_PIP_PACKAGES\" denotes a special environment variable that Prefect will use to install additional Python packages at runtime. This approach is an alternative to building an image with a custom requirements.txt copied into it.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-actions","title":"Deployment actions","text":"

    Deployment actions defined in your prefect.yaml file control the lifecycle of the creation and execution of your deployments. The three actions available are build, push, and pull. pull is the only required deployment action \u2014 it is used to define how Prefect will pull your deployment in remote execution environments.

    Each action is defined as a list of steps that are executing in sequence.

    Each step has the following format:

    section:\n- prefect_package.path.to.importable.step:\n    id: \"step-id\" # optional\n    requires: \"pip-installable-package-spec\" # optional\n    kwarg1: value\n    kwarg2: more-values\n

    Every step can optionally provide a requires field that Prefect will use to auto-install in the event that the step cannot be found in the current environment. Each step can also specify an id for the step which is used when referencing step outputs in later steps. The additional fields map directly onto Python keyword arguments to the step function. Within a given section, steps always run in the order that they are provided within the prefect.yaml file.

    Deployment Instruction Overrides

    build, push, and pull sections can all be overridden on a per-deployment basis by defining build, push, and pull fields within a deployment definition in the prefect.yaml file.

    The prefect deploy command will use any build, push, or pull instructions provided in a deployment's definition in the prefect.yaml file.

    This capability is useful with multiple deployments that require different deployment instructions.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#the-build-action","title":"The build action","text":"

    The build section of prefect.yaml is where any necessary side effects for running your deployments are built - the most common type of side effect produced here is a Docker image. If you initialize with the docker recipe, you will be prompted to provide required information, such as image name and tag:

    prefect init --recipe docker\n>> image_name: < insert image name here >\n>> tag: < insert image tag here >\n

    Use --field to avoid the interactive experience

    We recommend that you only initialize a recipe when you are first creating your deployment structure, and afterwards store your configuration files within version control. However, sometimes you may need to initialize programmatically and avoid the interactive prompts. To do so, provide all required fields for your recipe using the --field flag:

    prefect init --recipe docker \\\n    --field image_name=my-repo/my-image \\\n    --field tag=my-tag\n

    build:\n- prefect_docker.deployments.steps.build_docker_image:\n    requires: prefect-docker>=0.3.0\n    image_name: my-repo/my-image\n    tag: my-tag\n    dockerfile: auto\n    push: true\n

    Once you've confirmed that these fields are set to their desired values, this step will automatically build a Docker image with the provided name and tag and push it to the repository referenced by the image name. As the prefect-docker package documentation notes, this step produces a few fields that can optionally be used in future steps or within prefect.yaml as template values. It is best practice to use {{ image }} within prefect.yaml (specifically the work pool's job variables section) so that you don't risk having your build step and deployment specification get out of sync with hardcoded values.

    Note

    Note that in the build step example above, we relied on the prefect-docker package; in cases that deal with external services, additional packages are often required and will be auto-installed for you.

    Pass output to downstream steps

    Each deployment action can be composed of multiple steps. For example, if you wanted to build a Docker image tagged with the current commit hash, you could use the run_shell_script step and feed the output into the build_docker_image step:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        id: get-commit-hash\n        script: git rev-parse --short HEAD\n        stream_output: false\n    - prefect_docker.deployments.steps.build_docker_image:\n        requires: prefect-docker\n        image_name: my-image\n        image_tag: \"{{ get-commit-hash.stdout }}\"\n        dockerfile: auto\n

    Note that the id field is used in the run_shell_script step so that its output can be referenced in the next step.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#the-push-action","title":"The push action","text":"

    The push section is most critical for situations in which code is not stored on persistent filesystems or in version control. In this scenario, code is often pushed and pulled from a Cloud storage bucket of some kind (e.g., S3, GCS, Azure Blobs, etc.). The push section allows users to specify and customize the logic for pushing this code repository to arbitrary remote locations.

    For example, a user wishing to store their code in an S3 bucket and rely on default worker settings for its runtime environment could use the s3 recipe:

    prefect init --recipe s3\n>> bucket: < insert bucket name here >\n

    Inspecting our newly created prefect.yaml file we find that the push and pull sections have been templated out for us as follows:

    push:\n- prefect_aws.deployments.steps.push_to_s3:\n    id: push-code\n    requires: prefect-aws>=0.3.0\n    bucket: my-bucket\n    folder: project-name\n    credentials: null\n\npull:\n- prefect_aws.deployments.steps.pull_from_s3:\n    requires: prefect-aws>=0.3.0\n    bucket: my-bucket\n    folder: \"{{ push-code.folder }}\"\n    credentials: null\n

    The bucket has been populated with our provided value (which also could have been provided with the --field flag); note that the folder property of the push step is a template - the pull_from_s3 step outputs both a bucket value as well as a folder value that can be used to template downstream steps. Doing this helps you keep your steps consistent across edits.

    As discussed above, if you are using blocks, the credentials section can be templated with a block reference for secure and dynamic credentials access:

    push:\n- prefect_aws.deployments.steps.push_to_s3:\n    requires: prefect-aws>=0.3.0\n    bucket: my-bucket\n    folder: project-name\n    credentials: \"{{ prefect.blocks.aws-credentials.dev-credentials }}\"\n

    Anytime you run prefect deploy, this push section will be executed upon successful completion of your build section. For more information on the mechanics of steps, see below.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#the-pull-action","title":"The pull action","text":"

    The pull section is the most important section within the prefect.yaml file. It contains instructions for preparing your flows for a deployment run. These instructions will be executed each time a deployment created within this folder is run via a worker.

    There are three main types of steps that typically show up in a pull section:

    • set_working_directory: this step simply sets the working directory for the process prior to importing your flow
    • git_clone: this step clones the provided repository on the provided branch
    • pull_from_{cloud}: this step pulls the working directory from a Cloud storage location (e.g., S3)

    Use block and variable references

    All block and variable references within your pull step will remain unresolved until runtime and will be pulled each time your deployment is run. This allows you to avoid storing sensitive information insecurely; it also allows you to manage certain types of configuration from the API and UI without having to rebuild your deployment every time.

    Below is an example of how to use an existing GitHubCredentials block to clone a private GitHub repository:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/org/repo.git\n        credentials: \"{{ prefect.blocks.github-credentials.my-credentials }}\"\n

    Alternatively, you can specify a BitBucketCredentials or GitLabCredentials block to clone from Bitbucket or GitLab. In lieu of a credentials block, you can also provide a GitHub, GitLab, or Bitbucket token directly to the 'access_token` field. You can use a Secret block to do this securely:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://bitbucket.org/org/repo.git\n        access_token: \"{{ prefect.blocks.secret.bitbucket-token }}\"\n
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#utility-steps","title":"Utility steps","text":"

    Utility steps can be used within a build, push, or pull action to assist in managing the deployment lifecycle:

    • run_shell_script allows for the execution of one or more shell commands in a subprocess, and returns the standard output and standard error of the script. This step is useful for scripts that require execution in a specific environment, or those which have specific input and output requirements.

    Here is an example of retrieving the short Git commit hash of the current repository to use as a Docker image tag:

    build:\n    - prefect.deployments.steps.run_shell_script:\n        id: get-commit-hash\n        script: git rev-parse --short HEAD\n        stream_output: false\n    - prefect_docker.deployments.steps.build_docker_image:\n        requires: prefect-docker>=0.3.0\n        image_name: my-image\n        tag: \"{{ get-commit-hash.stdout }}\"\n        dockerfile: auto\n

    Provided environment variables are not expanded by default

    To expand environment variables in your shell script, set expand_env_vars: true in your run_shell_script step. For example:

    - prefect.deployments.steps.run_shell_script:\n    id: get-user\n    script: echo $USER\n    stream_output: true\n    expand_env_vars: true\n

    Without expand_env_vars: true, the above step would return a literal string $USER instead of the current user.

    • pip_install_requirements installs dependencies from a requirements.txt file within a specified directory.

    Below is an example of installing dependencies from a requirements.txt file after cloning:

    pull:\n    - prefect.deployments.steps.git_clone:\n        id: clone-step  # needed in order to be referenced in subsequent steps\n        repository: https://github.com/org/repo.git\n    - prefect.deployments.steps.pip_install_requirements:\n        directory: {{ clone-step.directory }}  # `clone-step` is a user-provided `id` field\n        requirements_file: requirements.txt\n

    Below is an example that retrieves an access token from a 3rd party Key Vault and uses it in a private clone step:

    pull:\n- prefect.deployments.steps.run_shell_script:\n    id: get-access-token\n    script: az keyvault secret show --name <secret name> --vault-name <secret vault> --query \"value\" --output tsv\n    stream_output: false\n- prefect.deployments.steps.git_clone:\n    repository: https://bitbucket.org/samples/deployments.git\n    branch: master\n    access_token: \"{{ get-access-token.stdout }}\"\n

    You can also run custom steps by packaging them. In the example below, retrieve_secrets is a custom python module that has been packaged into the default working directory of a Docker image (which is /opt/prefect by default). main is the function entry point, which returns an access token (e.g. return {\"access_token\": access_token}) like the preceding example, but utilizing the Azure Python SDK for retrieval.

    - retrieve_secrets.main:\n    id: get-access-token\n- prefect.deployments.steps.git_clone:\n    repository: https://bitbucket.org/samples/deployments.git\n    branch: master\n    access_token: '{{ get-access-token.access_token }}'\n
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#templating-options","title":"Templating options","text":"

    Values that you place within your prefect.yaml file can reference dynamic values in several different ways:

    • step outputs: every step of both build and push produce named fields such as image_name; you can reference these fields within prefect.yaml and prefect deploy will populate them with each call. References must be enclosed in double brackets and be of the form \"{{ field_name }}\"
    • blocks: Prefect blocks can also be referenced with the special syntax {{ prefect.blocks.block_type.block_slug }}. It is highly recommended that you use block references for any sensitive information (such as a GitHub access token or any credentials) to avoid hardcoding these values in plaintext
    • variables: Prefect variables can also be referenced with the special syntax {{ prefect.variables.variable_name }}. Variables can be used to reference non-sensitive, reusable pieces of information such as a default image name or a default work pool name.
    • environment variables: you can also reference environment variables with the special syntax {{ $MY_ENV_VAR }}. This is especially useful for referencing environment variables that are set at runtime.

    As an example, consider the following prefect.yaml file:

    build:\n- prefect_docker.deployments.steps.build_docker_image:\n    id: build-image\n    requires: prefect-docker>=0.3.0\n    image_name: my-repo/my-image\n    tag: my-tag\n    dockerfile: auto\n    push: true\n\ndeployments:\n- # base metadata\n    name: null\n    version: \"{{ build-image.tag }}\"\n    tags:\n        - \"{{ $my_deployment_tag }}\"\n        - \"{{ prefect.variables.some_common_tag }}\"\n    description: null\n    schedule: null\n\n    # flow-specific fields\n    entrypoint: null\n    parameters: {}\n\n    # infra-specific fields\n    work_pool:\n        name: \"my-k8s-work-pool\"\n        work_queue_name: null\n        job_variables:\n            image: \"{{ build-image.image }}\"\n            cluster_config: \"{{ prefect.blocks.kubernetes-cluster-config.my-favorite-config }}\"\n

    So long as our build steps produce fields called image_name and tag, every time we deploy a new version of our deployment, the {{ build-image.image }} variable will be dynamically populated with the relevant values.

    Docker step

    The most commonly used build step is prefect_docker.deployments.steps.build_docker_image which produces both the image_name and tag fields.

    For an example, check out the deployments tutorial.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-configurations","title":"Deployment Configurations","text":"

    You can create multiple deployments from one or more python file that use .deploy. Similarly a prefect.yaml file can have multiple deployment configurations that control the behavior of created deployments.

    These deployments can be managed independently of one another, allowing you to deploy the same flow with different configurations in the same codebase.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#working-with-multiple-deployments","title":"Working with multiple deployments","text":".deployprefect.yaml

    To create multiple work pool-based deployments at once you can use the deploy function, which is analogous to the serve function.

    from prefect import deploy, flow\n\n@flow(log_prints=True)\ndef buy():\n    print(\"Buying securities\")\n\n\nif __name__ == \"__main__\":\n    deploy(\n        buy.to_deployment(name=\"dev-deploy\", work_pool_name=\"my-dev-work-pool\"),\n        buy.to_deployment(name=\"prod-deploy\", work_pool_name=\"my-prod-work-pool\"),\n        image=\"my-registry/my-image:dev\",\n        push=False,\n    )\n

    Note that in the example above we created two deployments from the same flow, but with different work pools. Alternatively, we could have created two deployments from different flows.

    from prefect import deploy, flow\n\n@flow(log_prints=True)\ndef buy():\n    print(\"Buying securities.\")\n\n@flow(log_prints=True)\ndef sell():\n    print(\"Selling securities.\")\n\n\nif __name__ == \"__main__\":\n    deploy(\n        buy.to_deployment(name=\"buy-deploy\"),\n        sell.to_deployment(name=\"sell-deploy\"),\n        work_pool_name=\"my-dev-work-pool\"\n        image=\"my-registry/my-image:dev\",\n        push=False,\n    )\n

    In the example above the code for both flows gets baked into the same image.

    We can specify that one or more flows should be pulled from a remote location at runtime by using the from_source method. Here's an example of deploying two flows, one defined locally and one defined in a remote repository:

    from prefect import deploy, flow\n\n\n@flow(log_prints=True)\ndef local_flow():\n    print(\"I'm a flow!\")\n\nif __name__ == \"__main__\":\n    deploy(\n        local_flow.to_deployment(name=\"example-deploy-local-flow\"),\n        flow.from_source(\n            source=\"https://github.com/org/repo.git\",\n            entrypoint=\"flows.py:my_flow\",\n        ).to_deployment(\n            name=\"example-deploy-remote-flow\",\n        ),\n        work_pool_name=\"my-work-pool\",\n        image=\"my-registry/my-image:dev\",\n    )\n

    You could pass any number of flows to the deploy function. This behavior is useful if using a monorepo approach to your workflows.

    Prefect supports multiple deployment declarations within the prefect.yaml file. This method of declaring multiple deployments allows the configuration for all deployments to be version controlled and deployed with a single command.

    New deployment declarations can be added to the prefect.yaml file by adding a new entry to the deployments list. Each deployment declaration must have a unique name field which is used to select deployment declarations when using the prefect deploy command.

    For example, consider the following prefect.yaml file:

    build: ...\npush: ...\npull: ...\n\ndeployments:\n- name: deployment-1\n    entrypoint: flows/hello.py:my_flow\n    parameters:\n        number: 42,\n        message: Don't panic!\n    work_pool:\n        name: my-process-work-pool\n        work_queue_name: primary-queue\n\n- name: deployment-2\n    entrypoint: flows/goodbye.py:my_other_flow\n    work_pool:\n        name: my-process-work-pool\n        work_queue_name: secondary-queue\n\n- name: deployment-3\n    entrypoint: flows/hello.py:yet_another_flow\n    work_pool:\n        name: my-docker-work-pool\n        work_queue_name: tertiary-queue\n

    This file has three deployment declarations, each referencing a different flow. Each deployment declaration has a unique name field and can be deployed individually by using the --name flag when deploying.

    For example, to deploy deployment-1 you would run:

    prefect deploy --name deployment-1\n

    To deploy multiple deployments you can provide multiple --name flags:

    prefect deploy --name deployment-1 --name deployment-2\n

    To deploy multiple deployments with the same name, you can prefix the deployment name with its flow name:

    prefect deploy --name my_flow/deployment-1 --name my_other_flow/deployment-1\n

    To deploy all deployments you can use the --all flag:

    prefect deploy --all\n

    To deploy deployments that match a pattern you can run:

    prefect deploy -n my-flow/* -n *dev/my-deployment -n dep*prod\n

    The above command will deploy all deployments from the flow my-flow, all flows ending in dev with a deployment named my-deployment, and all deployments starting with dep and ending in prod.

    CLI Options When Deploying Multiple Deployments

    When deploying more than one deployment with a single prefect deploy command, any additional attributes provided via the CLI will be ignored.

    To provide overrides to a deployment via the CLI, you must deploy that deployment individually.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#reusing-configuration-across-deployments","title":"Reusing configuration across deployments","text":"

    Because a prefect.yaml file is a standard YAML file, you can use YAML aliases to reuse configuration across deployments.

    This functionality is useful when multiple deployments need to share the work pool configuration, deployment actions, or other configurations.

    You can declare a YAML alias by using the &{alias_name} syntax and insert that alias elsewhere in the file with the *{alias_name} syntax. When aliasing YAML maps, you can also override specific fields of the aliased map by using the <<: *{alias_name} syntax and adding additional fields below.

    We recommend adding a definitions section to your prefect.yaml file at the same level as the deployments section to store your aliases.

    For example, consider the following prefect.yaml file:

    build: ...\npush: ...\npull: ...\n\ndefinitions:\n    work_pools:\n        my_docker_work_pool: &my_docker_work_pool\n            name: my-docker-work-pool\n            work_queue_name: default\n            job_variables:\n                image: \"{{ build-image.image }}\"\n    schedules:\n        every_ten_minutes: &every_10_minutes\n            interval: 600\n    actions:\n        docker_build: &docker_build\n            - prefect_docker.deployments.steps.build_docker_image: &docker_build_config\n                id: build-image\n                requires: prefect-docker>=0.3.0\n                image_name: my-example-image\n                tag: dev\n                dockerfile: auto\n                push: true\n\ndeployments:\n- name: deployment-1\n    entrypoint: flows/hello.py:my_flow\n    schedule: *every_10_minutes\n    parameters:\n        number: 42,\n        message: Don't panic!\n    work_pool: *my_docker_work_pool\n    build: *docker_build # Uses the full docker_build action with no overrides\n\n- name: deployment-2\n    entrypoint: flows/goodbye.py:my_other_flow\n    work_pool: *my_docker_work_pool\n    build:\n        - prefect_docker.deployments.steps.build_docker_image:\n            <<: *docker_build_config # Uses the docker_build_config alias and overrides the dockerfile field\n            dockerfile: Dockerfile.custom\n\n- name: deployment-3\n    entrypoint: flows/hello.py:yet_another_flow\n    schedule: *every_10_minutes\n    work_pool:\n        name: my-process-work-pool\n        work_queue_name: primary-queue\n

    In the above example, we are using YAML aliases to reuse work pool, schedule, and build configuration across multiple deployments:

    • deployment-1 and deployment-2 are using the same work pool configuration
    • deployment-1 and deployment-3 are using the same schedule
    • deployment-1 and deployment-2 are using the same build deployment action, but deployment-2 is overriding the dockerfile field to use a custom Dockerfile
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-declaration-reference","title":"Deployment declaration reference","text":"","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-fields","title":"Deployment fields","text":"

    Below are fields that can be added to each deployment declaration.

    Property Description name The name to give to the created deployment. Used with the prefect deploy command to create or update specific deployments. version An optional version for the deployment. tags A list of strings to assign to the deployment as tags. description An optional description for the deployment. schedule An optional schedule to assign to the deployment. Fields for this section are documented in the Schedule Fields section. triggers An optional array of triggers to assign to the deployment entrypoint Required path to the .py file containing the flow you want to deploy (relative to the root directory of your development folder) combined with the name of the flow function. Should be in the format path/to/file.py:flow_function_name. parameters Optional default values to provide for the parameters of the deployed flow. Should be an object with key/value pairs. enforce_parameter_schema Boolean flag that determines whether the API should validate the parameters passed to a flow run against the parameter schema generated for the deployed flow. work_pool Information on where to schedule flow runs for the deployment. Fields for this section are documented in the Work Pool Fields section.","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#schedule-fields","title":"Schedule fields","text":"

    Below are fields that can be added to a deployment declaration's schedule section.

    Property Description interval Number of seconds indicating the time between flow runs. Cannot be used in conjunction with cron or rrule. anchor_date Datetime string indicating the starting or \"anchor\" date to begin the schedule. If no anchor_date is supplied, the current UTC time is used. Can only be used with interval. timezone String name of a time zone, used to enforce localization behaviors like DST boundaries. See the IANA Time Zone Database for valid time zones. cron A valid cron string. Cannot be used in conjunction with interval or rrule. day_or Boolean indicating how croniter handles day and day_of_week entries. Must be used with cron. Defaults to True. rrule String representation of an RRule schedule. See the rrulestr examples for syntax. Cannot be used in conjunction with interval or cron.

    For more information about schedules, see the Schedules concept doc.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#work-pool-fields","title":"Work pool fields","text":"

    Below are fields that can be added to a deployment declaration's work_pool section.

    Property Description name The name of the work pool to schedule flow runs in for the deployment. work_queue_name The name of the work queue within the specified work pool to schedule flow runs in for the deployment. If not provided, the default queue for the specified work pool will be used. job_variables Values used to override the default values in the specified work pool's base job template. Maps directly to a created deployments infra_overrides attribute.","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#deployment-mechanics","title":"Deployment mechanics","text":"

    Anytime you run prefect deploy in a directory that contains a prefect.yaml file, the following actions are taken in order:

    • The prefect.yaml file is loaded. First, the build section is loaded and all variable and block references are resolved. The steps are then run in the order provided.
    • Next, the push section is loaded and all variable and block references are resolved; the steps within this section are then run in the order provided
    • Next, the pull section is templated with any step outputs but is not run. Note that block references are not hydrated for security purposes - block references are always resolved at runtime
    • Next, all variable and block references are resolved with the deployment declaration. All flags provided via the prefect deploy CLI are then overlaid on the values loaded from the file.
    • The final step occurs when the fully realized deployment specification is registered with the Prefect API

    Deployment Instruction Overrides

    The build, push, and pull sections in deployment definitions take precedence over the corresponding sections above them in prefect.yaml.

    Each time a step is run, the following actions are taken in order:

    • The step's inputs and block / variable references are resolved (see the templating documentation above for more details).
    • The step's function is imported; if it cannot be found, the special requires keyword is used to install the necessary packages
    • The step's function is called with the resolved inputs.
    • The step's output is returned and used to resolve inputs for subsequent steps.
    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/prefect-deploy/#next-steps","title":"Next steps","text":"

    Now that you are familiar with creating deployments, you may want to explore push work pools or Kubernetes work pools.

    ","tags":["orchestration","deploy","CLI","flow runs","deployments","schedules","triggers","prefect.yaml","infrastructure","storage","work pool","worker"],"boost":2},{"location":"guides/runtime-context/","title":"Runtime context","text":"

    Prefect tracks information about the current flow or task run with a run context. The run context can be thought of as a global variable that allows the Prefect engine to determine relationships between your runs, e.g. which flow your task is called from.

    The run context itself contains many internal objects used by Prefect to manage execution of your run and is only available in specific situations. For this reason, we expose a simple interface that only includes the items you care about and dynamically retrieves additional information when necessary. We call this the \"runtime context\" as it contains information that can only be accessed when a run is happening.

    Mock values via environment variable

    Oftentimes, you may want to mock certain values for testing purposes. For example, by manually setting an ID or a scheduled start time to ensure your code is functioning properly. Starting in version 2.10.3, you can mock values in runtime via environment variable using the schema PREFECT__RUNTIME__{SUBMODULE}__{KEY_NAME}=value: ```bash $ export PREFECT__RUNTIME__TASK_RUN__FAKE_KEY='foo' $ python -c 'from prefect.runtime import task_run; print(task_run.fake_key)' # \"foo\"

    `` </div> If the environment variable mocks an existing runtime attribute, the value is cast to the same type. This works for runtime attributes of basic types (bool,int,floatandstr) andpendulum.DateTime. For complex types likelistordict`, we suggest mocking them using monkeypatch or a similar tool.

    ","tags":["flows","subflows","tasks","deployments"],"boost":2},{"location":"guides/runtime-context/#accessing-runtime-information","title":"Accessing runtime information","text":"

    The prefect.runtime module is the home for all runtime context access. Each major runtime concept has its own submodule:

    • deployment: Access information about the deployment for the current run
    • flow_run: Access information about the current flow run
    • task_run: Access information about the current task run

    For example:

    from prefect import flow, task\nfrom prefect import runtime\n\n@flow(log_prints=True)\ndef my_flow(x):\n    print(\"My name is\", runtime.flow_run.name)\n    print(\"I belong to deployment\", runtime.deployment.name)\n    my_task(2)\n\n@task\ndef my_task(y):\n    print(\"My name is\", runtime.task_run.name)\n    print(\"Flow run parameters:\", runtime.flow_run.parameters)\n\nmy_flow(1)\n

    Outputs:

    $ python my_runtime_info.py\n10:08:02.948 | INFO    | prefect.engine - Created flow run 'solid-gibbon' for flow 'my-flow'\n10:08:03.555 | INFO    | Flow run 'solid-gibbon' - My name is solid-gibbon\n10:08:03.558 | INFO    | Flow run 'solid-gibbon' - I belong to deployment None\n10:08:03.703 | INFO    | Flow run 'solid-gibbon' - Created task run 'my_task-0' for task 'my_task'\n10:08:03.704 | INFO    | Flow run 'solid-gibbon' - Executing 'my_task-0' immediately...\n10:08:04.006 | INFO    | Task run 'my_task-0' - My name is my_task-0\n10:08:04.007 | INFO    | Task run 'my_task-0' - Flow run parameters: {'x': 1}\n10:08:04.105 | INFO    | Task run 'my_task-0' - Finished in state Completed()\n10:08:04.968 | INFO    | Flow run 'solid-gibbon' - Finished in state Completed('All states completed.')\n

    Above, we demonstrate access to information about the current flow run, task run, and deployment. When run without a deployment (via python my_runtime_info.py), you should see \"I belong to deployment None\" logged. When information is not available, the runtime will always return an empty value. Because this flow was run without a deployment, there is no deployment data. If this flow were deployed and executed by a worker, we'd see the name of the deployment instead.

    See the runtime API reference for a full list of available attributes.

    ","tags":["flows","subflows","tasks","deployments"],"boost":2},{"location":"guides/runtime-context/#accessing-the-run-context-directly","title":"Accessing the run context directly","text":"

    The current run context can be accessed with prefect.context.get_run_context(). This will raise an exception if no run context is available, meaning you are not in a flow or task run. If a task run context is available, it will be returned even if a flow run context is available.

    Alternatively, you can access the flow run or task run context explicitly. This will, for example, allow you to access the flow run context from a task run. Note that we do not send the flow run context to distributed task workers because the context is costly to serialize and deserialize.

    from prefect.context import FlowRunContext, TaskRunContext\n\nflow_run_ctx = FlowRunContext.get()\ntask_run_ctx = TaskRunContext.get()\n

    Unlike get_run_context, this will not raise an error if the context is not available. Instead, it will return None.

    ","tags":["flows","subflows","tasks","deployments"],"boost":2},{"location":"guides/settings/","title":"Profiles & Configuration","text":"

    Prefect's settings are documented and type-validated. By modifying the default settings, you can customize various aspects of the system.

    Settings can be viewed from the CLI or the UI.

    You can override the setting for a profile with environment variables.

    Prefect profiles are groups of settings that you can persist on your machine. When you change profiles, all of the settings configured in the profile are applied. You can apply profiles to individual commands or set a profile for your environment.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#commonly-configured-settings","title":"Commonly configured settings","text":"

    This section describes some commonly configured settings for Prefect installations. See Configuring settings for details on setting and unsetting configuration values.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#prefect_api_url","title":"PREFECT_API_URL","text":"

    The PREFECT_API_URL value specifies the API endpoint of your Prefect Cloud workspace or a Prefect server instance.

    For example, using a local Prefect server instance.

    PREFECT_API_URL=\"http://127.0.0.1:4200/api\"\n

    Using Prefect Cloud:

    PREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]\"\n

    View your Account ID and Workspace ID in your browser URL when logged into Prefect Cloud. For example: https://app.prefect.cloud/account/abc-my-account-id-is-here/workspaces/123-my-workspace-id-is-here.

    PREFECT_API_URL setting for workers

    When using workers and work pools (agent and block-based deployments are legacy) that can create flow runs for deployments in remote environments, PREFECT_API_URL must be set for the environment in which your worker is running.

    If you want the worker to communicate with Prefect Cloud or a Prefect server instance from a remote execution environment such as a VM or Docker container, you must configure PREFECT_API_URL in that environment.

    Running the Prefect UI behind a reverse proxy

    When using a reverse proxy (such as Nginx or Traefik) to proxy traffic to a locally-hosted Prefect UI instance, the Prefect server also needs to be configured to know how to connect to the API. The PREFECT_UI_API_URL should be set to the external proxy URL (e.g. if your external URL is https://prefect-server.example.com/ then set PREFECT_UI_API_URL=https://prefect-server.example.com/api for the Prefect server process). You can also accomplish this by setting PREFECT_API_URL to the API URL, as this setting is used as a fallback if PREFECT_UI_API_URL is not set.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#prefect_api_key","title":"PREFECT_API_KEY","text":"

    The PREFECT_API_KEY value specifies the API key used to authenticate with your Prefect Cloud workspace.

    PREFECT_API_KEY=\"[API-KEY]\"\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#prefect_home","title":"PREFECT_HOME","text":"

    The PREFECT_HOME value specifies the local Prefect directory for configuration files, profiles, and the location of the default Prefect SQLite database.

    PREFECT_HOME='~/.prefect'\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#prefect_local_storage_path","title":"PREFECT_LOCAL_STORAGE_PATH","text":"

    The PREFECT_LOCAL_STORAGE_PATH value specifies the default location of local storage for flow runs.

    PREFECT_LOCAL_STORAGE_PATH='${PREFECT_HOME}/storage'\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#database-settings","title":"Database settings","text":"

    Prefect provides several self-hosting database configuration settings you can read about here.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#logging-settings","title":"Logging settings","text":"

    Prefect provides several logging configuration settings that you can read about in the logging docs.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#configuring-settings","title":"Configuring settings","text":"

    The prefect config CLI commands enable you to view, set, and unset settings.

    Command Description set Change the value for a setting. unset Restore the default value for a setting. view Display the current settings.","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#viewing-settings-from-the-cli","title":"Viewing settings from the CLI","text":"

    The prefect config view command will display settings that override default values.

    $ prefect config view\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='DEBUG'\n

    You can show the sources of values with --show-sources:

    $ prefect config view --show-sources\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='DEBUG' (from env)\n

    You can also include default values with --show-defaults:

    $ prefect config view --show-defaults\nPREFECT_PROFILE='default'\nPREFECT_AGENT_PREFETCH_SECONDS='10' (from defaults)\nPREFECT_AGENT_QUERY_INTERVAL='5.0' (from defaults)\nPREFECT_API_KEY='None' (from defaults)\nPREFECT_API_REQUEST_TIMEOUT='60.0' (from defaults)\nPREFECT_API_URL='None' (from defaults)\n...\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#setting-and-clearing-values","title":"Setting and clearing values","text":"

    The prefect config set command lets you change the value of a default setting.

    A commonly used example is setting the PREFECT_API_URL, which you may need to change when interacting with different Prefect server instances or Prefect Cloud.

    # use a local Prefect server\nprefect config set PREFECT_API_URL=\"http://127.0.0.1:4200/api\"\n\n# use Prefect Cloud\nprefect config set PREFECT_API_URL=\"https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]\"\n

    If you want to configure a setting to use its default value, use the prefect config unset command.

    prefect config unset PREFECT_API_URL\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#overriding-defaults-with-environment-variables","title":"Overriding defaults with environment variables","text":"

    All settings have keys that match the environment variable that can be used to override them.

    For example, configuring the home directory:

    # environment variable\nexport PREFECT_HOME=\"/path/to/home\"\n
    # python\nimport prefect.settings\nprefect.settings.PREFECT_HOME.value()  # PosixPath('/path/to/home')\n

    Configuring the server's port:

    # environment variable\nexport PREFECT_SERVER_API_PORT=4242\n
    # python\nprefect.settings.PREFECT_SERVER_API_PORT.value()  # 4242\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#configuration-profiles","title":"Configuration profiles","text":"

    Prefect allows you to persist settings instead of setting an environment variable each time you open a new shell. Settings are persisted to profiles, which allow you to move between groups of settings quickly.

    The prefect profile CLI commands enable you to create, review, and manage profiles.

    Command Description create Create a new profile. delete Delete the given profile. inspect Display settings from a given profile; defaults to active. ls List profile names. rename Change the name of a profile. use Switch the active profile.

    If you configured settings for a profile, prefect profile inspect displays those settings:

    $ prefect profile inspect\nPREFECT_PROFILE = \"default\"\nPREFECT_API_KEY = \"pnu_XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX\"\nPREFECT_API_URL = \"http://127.0.0.1:4200/api\"\n

    You can pass the name of a profile to view its settings:

    $ prefect profile create test\n$ prefect profile inspect test\nPREFECT_PROFILE=\"test\"\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#creating-and-removing-profiles","title":"Creating and removing profiles","text":"

    Create a new profile with no settings:

    $ prefect profile create test\nCreated profile 'test' at /Users/terry/.prefect/profiles.toml.\n

    Create a new profile foo with settings cloned from an existing default profile:

    $ prefect profile create foo --from default\nCreated profile 'cloud' matching 'default' at /Users/terry/.prefect/profiles.toml.\n

    Rename a profile:

    $ prefect profile rename temp test\nRenamed profile 'temp' to 'test'.\n

    Remove a profile:

    $ prefect profile delete test\nRemoved profile 'test'.\n

    Removing the default profile resets it:

    $ prefect profile delete default\nReset profile 'default'.\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#change-values-in-profiles","title":"Change values in profiles","text":"

    Set a value in the current profile:

    $ prefect config set VAR=X\nSet variable 'VAR' to 'X'\nUpdated profile 'default'\n

    Set multiple values in the current profile:

    $ prefect config set VAR2=Y VAR3=Z\nSet variable 'VAR2' to 'Y'\nSet variable 'VAR3' to 'Z'\nUpdated profile 'default'\n

    You can set a value in another profile by passing the --profile NAME option to a CLI command:

    $ prefect --profile \"foo\" config set VAR=Y\nSet variable 'VAR' to 'Y'\nUpdated profile 'foo'\n

    Unset values in the current profile to restore the defaults:

    $ prefect config unset VAR2 VAR3\nUnset variable 'VAR2'\nUnset variable 'VAR3'\nUpdated profile 'default'\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#inspecting-profiles","title":"Inspecting profiles","text":"

    See a list of available profiles:

    $ prefect profile ls\n* default\ncloud\ntest\nlocal\n

    View all settings for a profile:

    $ prefect profile inspect cloud\nPREFECT_API_URL='https://api.prefect.cloud/api/accounts/xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxx\nx/workspaces/xxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx'\nPREFECT_API_KEY='xxx_xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'          \n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#using-profiles","title":"Using profiles","text":"

    The profile default is used by default. There are several methods to switch to another profile.

    The recommended method is to use the prefect profile use command with the name of the profile:

    $ prefect profile use foo\nProfile 'test' now active.\n

    Alternatively, you may set the environment variable PREFECT_PROFILE to the name of the profile:

    export PREFECT_PROFILE=foo\n

    Or, specify the profile in the CLI command for one-time usage:

    prefect --profile \"foo\" ...\n

    Note that this option must come before the subcommand. For example, to list flow runs using the profile foo:

    prefect --profile \"foo\" flow-run ls\n

    You may use the -p flag as well:

    prefect -p \"foo\" flow-run ls\n

    You may also create an 'alias' to automatically use your profile:

    $ alias prefect-foo=\"prefect --profile 'foo' \"\n# uses our profile!\n$ prefect-foo config view  \n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#conflicts-with-environment-variables","title":"Conflicts with environment variables","text":"

    If setting the profile from the CLI with --profile, environment variables that conflict with settings in the profile will be ignored.

    In all other cases, environment variables will take precedence over the value in the profile.

    For example, a value set in a profile will be used by default:

    $ prefect config set PREFECT_LOGGING_LEVEL=\"ERROR\"\n$ prefect config view --show-sources\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='ERROR' (from profile)\n

    But, setting an environment variable will override the profile setting:

    $ export PREFECT_LOGGING_LEVEL=\"DEBUG\"\n$ prefect config view --show-sources\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='DEBUG' (from env)\n

    Unless the profile is explicitly requested when using the CLI:

    $ prefect --profile default config view --show-sources\nPREFECT_PROFILE=\"default\"\nPREFECT_LOGGING_LEVEL='ERROR' (from profile)\n
    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/settings/#profile-files","title":"Profile files","text":"

    Profiles are persisted to the PREFECT_PROFILES_PATH, which can be changed with an environment variable.

    By default, it is stored in your PREFECT_HOME directory:

    $ prefect config view --show-defaults | grep PROFILES_PATH\nPREFECT_PROFILES_PATH='~/.prefect/profiles.toml'\n

    The TOML format is used to store profile data.

    ","tags":["configuration","settings","environment variables","profiles"],"boost":2},{"location":"guides/state-change-hooks/","title":"State Change Hooks","text":"

    You've read about how state change hooks execute code in response to changes in flow or task run states, enabling you to define actions for specific state transitions in a workflow. Now let's see some real-world use cases!

    ","tags":["state change","hooks","triggers"],"boost":2},{"location":"guides/state-change-hooks/#example-use-cases","title":"Example use cases","text":"","tags":["state change","hooks","triggers"],"boost":2},{"location":"guides/state-change-hooks/#send-a-notification-when-a-flow-run-fails","title":"Send a notification when a flow run fails","text":"

    State change hooks enable you to customize messages sent when tasks transition between states, such as sending notifications containing sensitive information when tasks enter a Failed state. Let's run a client-side hook upon a flow run entering a Failed state.

    from prefect import flow\nfrom prefect.blocks.core import Block\nfrom prefect.settings import PREFECT_API_URL\n\ndef notify_slack(flow, flow_run, state):\n    slack_webhook_block = Block.load(\n        \"slack-webhook/my-slack-webhook\"\n    )\n\n    slack_webhook_block.notify(\n        (\n            f\"Your job {flow_run.name} entered {state.name} \"\n            f\"with message:\\n\\n\"\n            f\"See <https://{PREFECT_API_URL.value()}/flow-runs/\"\n            f\"flow-run/{flow_run.id}|the flow run in the UI>\\n\\n\"\n            f\"Tags: {flow_run.tags}\\n\\n\"\n            f\"Scheduled start: {flow_run.expected_start_time}\"\n        )\n    )\n\n@flow(on_failure=[notify_slack], retries=1)\ndef failing_flow():\n    raise ValueError(\"oops!\")\n\nif __name__ == \"__main__\":\n    failing_flow()\n

    Note that because we've configured retries here, the on_failure hook will not run until all retries have completed, when the flow run finally enters a Failed state.

    ","tags":["state change","hooks","triggers"],"boost":2},{"location":"guides/state-change-hooks/#delete-a-cloud-run-job-when-a-flow-crashes","title":"Delete a Cloud Run job when a flow crashes","text":"

    State change hooks can aid in managing infrastructure cleanup in scenarios where tasks spin up individual infrastructure resources independently of Prefect. When a flow run crashes, tasks may exit abruptly, resulting in the potential omission of cleanup logic within the tasks. State change hooks can be used to ensure infrastructure is properly cleaned up even when a flow run enters a Crashed state.

    Let's create a hook that deletes a Cloud Run job if the flow run crashes.

    import os\nfrom prefect import flow, task\nfrom prefect.blocks.system import String\nfrom prefect.client import get_client\nimport prefect.runtime\n\nasync def delete_cloud_run_job(flow, flow_run, state):\n    \"\"\"Flow run state change hook that deletes a Cloud Run Job if\n    the flow run crashes.\"\"\"\n\n    # retrieve Cloud Run job name\n    cloud_run_job_name = await String.load(\n        name=\"crashing-flow-cloud-run-job\"\n    )\n\n    # delete Cloud Run job\n    delete_job_command = f\"yes | gcloud beta run jobs delete \n    {cloud_run_job_name.value} --region us-central1\"\n    os.system(delete_job_command)\n\n    # clean up the Cloud Run job string block as well\n    async with get_client() as client:\n        block_document = await client.read_block_document_by_name(\n            \"crashing-flow-cloud-run-job\", block_type_slug=\"string\"\n        )\n        await client.delete_block_document(block_document.id)\n\n@task\ndef my_task_that_crashes():\n    raise SystemExit(\"Crashing on purpose!\")\n\n@flow(on_crashed=[delete_cloud_run_job])\ndef crashing_flow():\n    \"\"\"Save the flow run name (i.e. Cloud Run job name) as a \n    String block. It then executes a task that ends up crashing.\"\"\"\n    flow_run_name = prefect.runtime.flow_run.name\n    cloud_run_job_name = String(value=flow_run_name)\n    cloud_run_job_name.save(\n        name=\"crashing-flow-cloud-run-job\", overwrite=True\n    )\n\n    my_task_that_crashes()\n\nif __name__ == \"__main__\":\n    crashing_flow()\n
    ","tags":["state change","hooks","triggers"],"boost":2},{"location":"guides/testing/","title":"Testing","text":"

    Once you have some awesome flows, you probably want to test them!

    ","tags":["testing","unit testing","development"],"boost":2},{"location":"guides/testing/#unit-testing-flows","title":"Unit testing flows","text":"

    Prefect provides a simple context manager for unit tests that allows you to run flows and tasks against a temporary local SQLite database.

    from prefect import flow\nfrom prefect.testing.utilities import prefect_test_harness\n\n@flow\ndef my_favorite_flow():\n    return 42\n\ndef test_my_favorite_flow():\n  with prefect_test_harness():\n      # run the flow against a temporary testing database\n      assert my_favorite_flow() == 42\n

    For more extensive testing, you can leverage prefect_test_harness as a fixture in your unit testing framework. For example, when using pytest:

    from prefect import flow\nimport pytest\nfrom prefect.testing.utilities import prefect_test_harness\n\n@pytest.fixture(autouse=True, scope=\"session\")\ndef prefect_test_fixture():\n    with prefect_test_harness():\n        yield\n\n@flow\ndef my_favorite_flow():\n    return 42\n\ndef test_my_favorite_flow():\n    assert my_favorite_flow() == 42\n

    Note

    In this example, the fixture is scoped to run once for the entire test session. In most cases, you will not need a clean database for each test and just want to isolate your test runs to a test database. Creating a new test database per test creates significant overhead, so we recommend scoping the fixture to the session. If you need to isolate some tests fully, you can use the test harness again to create a fresh database.

    ","tags":["testing","unit testing","development"],"boost":2},{"location":"guides/testing/#unit-testing-tasks","title":"Unit testing tasks","text":"

    To test an individual task, you can access the original function using .fn:

    from prefect import flow, task\n\n@task\ndef my_favorite_task():\n    return 42\n\n@flow\ndef my_favorite_flow():\n    val = my_favorite_task()\n    return val\n\ndef test_my_favorite_task():\n    assert my_favorite_task.fn() == 42\n

    Disable logger

    If your task makes uses a logger, you can disable the logger in order to avoid the RuntimeError raised from a missing flow context.

    from prefect.logging import disable_run_logger\n\ndef test_my_favorite_task():\n    with disable_run_logger():\n        assert my_favorite_task.fn() == 42\n

    ","tags":["testing","unit testing","development"],"boost":2},{"location":"guides/troubleshooting/","title":"Troubleshooting","text":"

    Don't Panic! If you experience an error with Prefect, there are many paths to understanding and resolving it. The first troubleshooting step is confirming that you are running the latest version of Prefect. If you are not, be sure to upgrade to the latest version, since the issue may have already been fixed. Beyond that, there are several categories of errors:

    • The issue may be in your flow code, in which case you should carefully read the logs.
    • The issue could be with how you are authenticated, and whether or not you are connected to Cloud.
    • The issue might have to do with how your code is executed.
    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#upgrade","title":"Upgrade","text":"

    Prefect is constantly evolving, adding new features and fixing bugs. Chances are that a patch has already been identified and released. Search existing issues for similar reports and check out the Release Notes. Upgrade to the newest version with the following command:

    pip install --upgrade prefect\n

    Different components may use different versions of Prefect:

    • Cloud will generally always be the newest version. Cloud is continuously deployed by the Prefect team. When using a self-hosted server, you can control this version.
    • Workers and agents typically don't change versions frequently, and are usually whatever the latest version was at the time of creation. Workers and agents provision infrastructure for flow runs, so upgrading them may help with infrastructure problems.
    • Flows could use a different version than the worker or agent that created them, especially when running in different environments. Suppose your worker and flow both use the latest official Docker image, but your worker was created a month ago. Your worker will often be on an older version than your flow.

    Integration Versions

    Keep in mind that integrations are versioned and released independently of the core Prefect library. They should be upgraded simultaneously with the core library, using the same method.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#logs","title":"Logs","text":"

    In many cases, there will be an informative stack trace in Prefect's logs. Read it carefully, locate the source of the error, and try to identify the cause.

    There are two types of logs:

    • Flow and task logs are always scoped to a flow. They are sent to Prefect and are viewable in the UI.
    • Worker and agent logs are not scoped to a flow and may have more information on what happened before the flow started. These logs are generally only available where the worker or agent is running.

    If your flow and task logs are empty, there may have been an infrastructure issue that prevented your flow from starting. Check your worker logs for more details.

    If there is no clear indication of what went wrong, try updating the logging level from the default INFO level to the DEBUG level. Settings such as the logging level are propagated from the worker environment to the flow run environment and can be set via environment variables or the prefect config set CLI:

    # Using the CLI\nprefect config set PREFECT_LOGGING_LEVEL=DEBUG\n\n# Using environment variables\nexport PREFECT_LOGGING_LEVEL=DEBUG\n

    The DEBUG logging level produces a high volume of logs so consider setting it back to INFO once any issues are resolved.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#cloud","title":"Cloud","text":"

    When using Prefect Cloud, there are the additional concerns of authentication and authorization. The Prefect API authenticates users and service accounts - collectively known as actors - with API keys. Missing, incorrect, or expired API keys will result in a 401 response with detail Invalid authentication credentials. Use the following command to check your authentication, replacing $PREFECT_API_KEY with your API key:

    curl -s -H \"Authorization: Bearer $PREFECT_API_KEY\" \"https://api.prefect.cloud/api/me/\"\n

    Users vs Service Accounts

    Service accounts - sometimes referred to as bots - represent non-human actors that interact with Prefect such as workers and CI/CD systems. Each human that interacts with Prefect should be represented as a user. User API keys start with pnu_ and service account API keys start with pnb_.

    Supposing the response succeeds, let's check our authorization. Actors can be members of workspaces. An actor attempting an action in a workspace they are not a member of will result in a 404 response. Use the following command to check your actor's workspace memberships:

    curl -s -H \"Authorization: Bearer $PREFECT_API_KEY\" \"https://api.prefect.cloud/api/me/workspaces\"\n

    Formatting JSON

    Python comes with a helpful tool for formatting JSON. Append the following to the end of the command above to make the output more readable: | python -m json.tool

    Make sure your actor is a member of the workspace you are working in. Within a workspace, an actor has a role which grants them certain permissions. Insufficient permissions will result in an error. For example, starting an agent or worker with the Viewer role, will result in errors.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#execution","title":"Execution","text":"

    Prefect flows can be executed locally by the user, or remotely by a worker or agent. Local execution generally means that you - the user - run your flow directly with a command like python flow.py. Remote execution generally means that a worker runs your flow via a deployment, optionally on different infrastructure.

    With remote execution, the creation of your flow run happens separately from its execution. Flow runs are assigned to a work pool and a work queue. For flow runs to execute, a worker must be subscribed to the work pool and work queue, otherwise the flow runs will go from Scheduled to Late. Ensure that your work pool and work queue have a subscribed worker.

    Local and remote execution can also differ in their treatment of relative imports. If switching from local to remote execution results in local import errors, try replicating the behavior by executing the flow locally with the -m flag (i.e. python -m flow instead of python flow.py). Read more about -m here.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#api-tests-return-an-unexpected-307-redirected","title":"API tests return an unexpected 307 Redirected","text":"

    Summary: Requests require a trailing / in the request URL.

    If you write a test that does not include a trailing / when making a request to a specific endpoint:

    async def test_example(client):\n    response = await client.post(\"/my_route\")\n    assert response.status_code == 201\n

    You'll see a failure like:

    E       assert 307 == 201\nE        +  where 307 = <Response [307 Temporary Redirect]>.status_code\n

    To resolve this, include the trailing /:

    async def test_example(client):\n    response = await client.post(\"/my_route/\")\n    assert response.status_code == 201\n

    Note: requests to nested URLs may exhibit the opposite behavior and require no trailing slash:

    async def test_nested_example(client):\n    response = await client.post(\"/my_route/filter/\")\n    assert response.status_code == 307\n\n    response = await client.post(\"/my_route/filter\")\n    assert response.status_code == 200\n

    Reference: \"HTTPX disabled redirect following by default\" in 0.22.0.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/troubleshooting/#pytestpytestunraisableexceptionwarning-or-resourcewarning","title":"pytest.PytestUnraisableExceptionWarning or ResourceWarning","text":"

    As you're working with one of the FlowRunner implementations, you may get an error like this one:

    E               pytest.PytestUnraisableExceptionWarning: Exception ignored in: <ssl.SSLSocket fd=-1, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0>\nE\nE               Traceback (most recent call last):\nE                 File \".../pytest_asyncio/plugin.py\", line 306, in setup\nE                   res = await func(**_add_kwargs(func, kwargs, event_loop, request))\nE               ResourceWarning: unclosed <ssl.SSLSocket fd=10, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 60605), raddr=('127.0.0.1', 6443)>\n\n.../_pytest/unraisableexception.py:78: PytestUnraisableExceptionWarning\n

    This error is saying that your test suite (or the prefect library code) opened a connection to something (like a Docker daemon or a Kubernetes cluster) and didn't close it.

    It may help to re-run the specific test with PYTHONTRACEMALLOC=25 pytest ... so that Python can display more of the stack trace where the connection was opened.

    ","tags":["troubleshooting","guides","how to"]},{"location":"guides/upgrade-guide-agents-to-workers/","title":"Upgrade from Agents to Workers","text":"

    Upgrading from agents to workers significantly enhances the experience of deploying flows. It simplifies the specification of each flow's infrastructure and runtime environment.

    A worker is the fusion of an agent with an infrastructure block. Like agents, workers poll a work pool for flow runs that are scheduled to start. Like infrastructure blocks, workers are typed - they work with only one kind of infrastructure, and they specify the default configuration for jobs submitted to that infrastructure.

    Accordingly, workers are not a drop-in replacement for agents. Using workers requires deploying flows differently. In particular, deploying a flow with a worker does not involve specifying an infrastructure block. Instead, infrastructure configuration is specified on the work pool and passed to each worker that polls work from that pool.

    This guide provides an overview of the differences between agents and workers. It also describes how to upgrade from agents to workers in just a few quick steps.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#enhancements","title":"Enhancements","text":"","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#workers","title":"Workers","text":"
    • Improved visibility into the status of each worker, including when a worker was started and when it last polled.
    • Better handling of race conditions for high availability use cases.
    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#work-pools","title":"Work pools","text":"
    • Work pools allow greater customization and governance of infrastructure parameters for deployments via their base job template.
    • Prefect Cloud push work pools enable flow execution in your environment without needing to host a worker.
    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#improved-deployment-apis","title":"Improved deployment APIs","text":"
    • The new YAML-based and Python deployment APIs are more flexible and easier to use than block and agent-based deployments.
    • Both options allow you to deploy multiple flows with a single command.
    • Both options allow you to build Docker images for your flows to create portable execution environments.
    • The YAML-based API supports templating to enable dryer deployment definitions.
    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#whats-different","title":"What's different","text":"
    1. Deployment CLI and Python API:

      prefect deployment build <entrypoint>/prefect deployment apply --> prefect deploy

      Prefect will now automatically detect flows in your repo and provide a wizard \ud83e\uddd9 to guide you through setting required attributes for your deployments.

      Deployment.build_from_flow --> flow.deploy

    2. Configuring remote flow code storage:

      storage blocks --> pull action

      When using the YAML-based deployment API, you can configure a pull action in your prefect.yaml file to specify how to retrieve flow code for your deployments. You can use configuration from your existing storage blocks to define your pull action via templating.

      When using the Python deployment API, you can pass any storage block to the flow.deploy method to specify how to retrieve flow code for your deployment.

    3. Configuring flow run infrastructure:

      infrastructure blocks --> typed work pool

      Default infrastructure config is now set on the typed work pool, and can be overwritten by individual deployments.

    4. Managing multiple deployments:

      Create and/or update many deployments at once through a prefect.yaml file or use the deploy function.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#whats-similar","title":"What's similar","text":"
    • Storage blocks can be set as the pull action in a prefect.yaml file.
    • Infrastructure blocks have configuration fields similar to typed work pools.
    • Deployment-level infrastructure overrides operate in much the same way.

      infra_override -> job_variable

    • The process for starting an agent and starting a worker in your environment are virtually identical.

      prefect agent start --pool <work pool name> --> prefect worker start --pool <work pool name>

      Worker Helm chart

      If you host your agents in a Kubernetes cluster, you can use the Prefect worker Helm chart to host workers in your cluster.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#upgrade-guide","title":"Upgrade guide","text":"

    If you have existing deployments that use infrastructure blocks, you can quickly upgrade them to be compatible with workers by following these steps:

    1. Create a work pool

      This new work pool will replace your infrastructure block.

      You can use the .publish_as_work_pool method on any infrastructure block to create a work pool with the same configuration.

      For example, if you have a KubernetesJob infrastructure block named 'my-k8s-job', you can create a work pool with the same configuration with this script:

      from prefect.infrastructure import KubernetesJob\n\nKubernetesJob.load(\"my-k8s-job\").publish_as_work_pool()\n

      Running this script will create a work pool named 'my-k8s-job' with the same configuration as your infrastructure block.

      Serving flows

      If you are using a Process infrastructure block and a LocalFilesystem storage block (or aren't using an infrastructure and storage block at all), you can use flow.serve to create a deployment without needing to specify a work pool name or start a worker.

      This is a quick way to create a deployment for a flow and is a great way to manage your deployments if you don't need dynamic infrastructure creation or configuration offered by workers.

      Check out our Docker guide for how to build a served flow into a Docker image and host it in your environment.

    2. Start a worker

      This worker will replace your agent and poll your new work pool for flow runs to execute.

      prefect worker start -p <work pool name>\n

    3. Deploy your flows to the new work pool

      To deploy your flows to the new work pool, you can use flow.deploy for a Pythonic deployment experience or prefect deploy for a YAML-based deployment experience.

      If you currently use Deployment.build_from_flow, we recommend using flow.deploy.

      If you currently use prefect deployment build and prefect deployment apply, we recommend using prefect deploy.

      .deployprefect deploy

      If you have a Python script that uses Deployment.build_from_flow, you can replace it with flow.deploy.

      Most arguments to Deployment.build_from_flow can be translated directly to flow.deploy, but here are some changes that you may need to make:

      • Replace infrastructure with work_pool_name.
        • If you've used the .publish_as_work_pool method on your infrastructure block, use the name of the created work pool.
      • Replace infra_overrides with job_variables.
      • Replace storage with a call to flow.from_source.
        • flow.from_source will load your flow from a remote storage location and make it deployable. Your existing storage block can be passed to the source argument of flow.from_source.

      Below are some examples of how to translate Deployment.build_from_flow to flow.deploy.

      Always run prefect deploy commands from the root level of your repo!

      With agents, you might have had multiple deployment.yaml files, but under worker deployment patterns, each repo will have a single prefect.yaml file located at the root of the repo that contains deployment configuration for all flows in that repo.

      To set up a new prefect.yaml file for your deployments, run the following command from the root level of your repo:

      perfect deploy\n

      This will start a wizard that will guide you through setting up your first deployment.

      For step 4, select y on the last prompt to save the configuration for the deployment.

      Saving the configuration for your deployment will result in a prefect.yaml file populated with your first deployment. You can use this YAML file to edit and define multiple deployments for this repo.

      You can add more deployments to the deployments list in your prefect.yaml file and/or by continuing to use the deployment creation wizard.

    For more information, check out our in-depth guide for deploying flows to work pools.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#deploying-without-any-blocks","title":"Deploying without any blocks","text":"

    If you aren't using any blocks:

        from prefect import flow\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a Python script!\")\n\n    if __name__ == \"__main__\":\n        Deployment.build_from_flow(\n            my_flow,\n            name=\"my-deployment\",\n            parameters=dict(name=\"Marvin\"),\n        )\n

    You can replace Deployment.build_from_flow with flow.serve :

        from prefect import flow\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a Python script!\")\n\n    if __name__ == \"__main__\":\n        my_flow.serve(\n            name=\"my-deployment\",\n            parameters=dict(name=\"Marvin\"),\n        )\n

    This will start a process that will serve your flow and execute any flow runs that are scheduled to start.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#deploying-using-a-storage-block","title":"Deploying using a storage block","text":"

    If you currently use a storage block to load your flow code but no infrastructure block:

        from prefect import flow\n    from prefect.storage import GitHub\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a GitHub repo!\")\n\n    if __name__ == \"__main__\":\n        Deployment.build_from_flow(\n            my_flow,\n            name=\"my-deployment\",\n            storage=GitHub.load(\"demo-repo\"),\n            parameters=dict(name=\"Marvin\"),\n        )\n

    you can use flow.from_source to load your flow from the same location and flow.serve to create a deployment:

        from prefect import flow\n    from prefect.storage import GitHub\n\n    if __name__ == \"__main__\":\n        flow.from_source(\n            source=GitHub.load(\"demo-repo\"),\n            entrypoint=\"example.py:my_flow\"\n        ).serve(\n            name=\"my-deployment\",\n            parameters=dict(name=\"Marvin\"),\n        )\n

    This will allow you to execute scheduled flow runs without starting a worker. Additionally, the process serving your flow will regularly check for updates to your flow code and automatically update the flow if it detects any changes to the code.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#deploying-using-an-infrastructure-and-storage-block","title":"Deploying using an infrastructure and storage block","text":"

    For the code below, we'll need to create a work pool from our infrastructure block and pass it to flow.deploy as the work_pool_name argument. We'll also need to pass our storage block to flow.from_source as the source argument.

        from prefect import flow\n    from prefect.deployments import Deployment\n    from prefect.filesystems import GitHub\n    from prefect.infrastructure.kubernetes import KubernetesJob\n\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a GitHub repo!\")\n\n\n    if __name__ == \"__main__\":\n        Deployment.build_from_flow(\n            my_flow,\n            name=\"my-deployment\",\n            storage=GitHub.load(\"demo-repo\"),\n            entrypoint=\"example.py:my_flow\",\n            infrastructure=KubernetesJob.load(\"my-k8s-job\"),\n            infra_overrides=dict(pull_policy=\"Never\"),\n            parameters=dict(name=\"Marvin\"),\n        )\n

    The equivalent deployment code using flow.deploy would look like this:

        from prefect import flow\n    from prefect.storage import GitHub\n\n    if __name__ == \"__main__\":\n        flow.from_source(\n            source=GitHub.load(\"demo-repo\"),\n            entrypoint=\"example.py:my_flow\"\n        ).deploy(\n            name=\"my-deployment\",\n            work_pool_name=\"my-k8s-job\",\n            job_variables=dict(pull_policy=\"Never\"),\n            parameters=dict(name=\"Marvin\"),\n        )\n

    Note that when using flow.from_source(...).deploy(...), the flow you're deploying does not need to be available locally before running your script.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/upgrade-guide-agents-to-workers/#deploying-via-a-docker-image","title":"Deploying via a Docker image","text":"

    If you currently bake your flow code into a Docker image before deploying, you can use the image argument of flow.deploy to build a Docker image as part of your deployment process:

        from prefect import flow\n\n    @flow(log_prints=True)\n    def my_flow(name: str = \"world\"):\n        print(f\"Hello {name}! I'm a flow from a Docker image!\")\n\n\n    if __name__ == \"__main__\":\n        my_flow.deploy(\n            name=\"my-deployment\",\n            image=\"my-repo/my-image:latest\",\n            work_pool_name=\"my-k8s-job\",\n            job_variables=dict(pull_policy=\"Never\"),\n            parameters=dict(name=\"Marvin\"),\n        )\n

    You can skip a flow.from_source call when building an image with flow.deploy. Prefect will keep track of the flow's source code location in the image and load it from that location when the flow is executed.

    ","tags":["worker","agent","deployments","infrastructure","work pool"],"boost":2},{"location":"guides/using-the-client/","title":"Using the Prefect Orchestration Client","text":"","tags":["client","API","filters","orchestration"]},{"location":"guides/using-the-client/#overview","title":"Overview","text":"

    In the API reference for the PrefectClient, you can find a bunch of useful client methods that make it simpler to do things like:

    • reschedule late flow runs
    • get the last N completed flow runs from my workspace

    The PrefectClient is an async context manager, so you can use it like this:

    from prefect import get_client\n\nasync with get_client() as client:\n    response = await client.hello()\n    print(response.json()) # \ud83d\udc4b\n

    ","tags":["client","API","filters","orchestration"]},{"location":"guides/using-the-client/#examples","title":"Examples","text":"","tags":["client","API","filters","orchestration"]},{"location":"guides/using-the-client/#rescheduling-late-flow-runs","title":"Rescheduling late flow runs","text":"

    Sometimes, you may need to bulk reschedule flow runs that are late - for example, if you've accidentally scheduled many flow runs of a deployment to an inactive work pool.

    To do this, we can delete late flow runs and create new ones in a Scheduled state with a delay.

    This example reschedules the last 3 late flow runs of a deployment named healthcheck-storage-test to run 6 hours later than their original expected start time. It also deletes any remaining late flow runs of that deployment.

    import asyncio\nfrom datetime import datetime, timedelta, timezone\nfrom typing import Optional\n\nfrom prefect import get_client\nfrom prefect.client.schemas.filters import (\n    DeploymentFilter, FlowRunFilter\n)\nfrom prefect.client.schemas.objects import FlowRun\nfrom prefect.client.schemas.sorting import FlowRunSort\nfrom prefect.states import Scheduled\n\nasync def reschedule_late_flow_runs(\n    deployment_name: str,\n    delay: timedelta,\n    most_recent_n: int,\n    delete_remaining: bool = True,\n    states: Optional[list[str]] = None\n) -> list[FlowRun]:\n    if not states:\n        states = [\"Late\"]\n\n    async with get_client() as client:\n        flow_runs = await client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state=dict(name=dict(any_=states)),\n                expected_start_time=dict(\n                    before_=datetime.now(timezone.utc)\n                ),\n            ),\n            deployment_filter=DeploymentFilter(\n                name={'like_': deployment_name}\n            ),\n            sort=FlowRunSort.START_TIME_DESC,\n            limit=most_recent_n if not delete_remaining else None\n        )\n\n        if not flow_runs:\n            print(f\"No flow runs found in states: {states!r}\")\n            return []\n\n        rescheduled_flow_runs = []\n        for i, run in enumerate(flow_runs):\n            await client.delete_flow_run(flow_run_id=run.id)\n            if i < most_recent_n:\n                new_run = await client.create_flow_run_from_deployment(\n                    deployment_id=run.deployment_id,\n                    state=Scheduled(\n                        scheduled_time=run.expected_start_time + delay\n                    ),\n                )\n                rescheduled_flow_runs.append(new_run)\n\n        return rescheduled_flow_runs\n\nif __name__ == \"__main__\":\n    rescheduled_flow_runs = asyncio.run(\n        reschedule_late_flow_runs(\n            deployment_name=\"healthcheck-storage-test\",\n            delay=timedelta(hours=6),\n            most_recent_n=3,\n        )\n    )\n\n    print(f\"Rescheduled {len(rescheduled_flow_runs)} flow runs\")\n\n    assert all(\n        run.state.is_scheduled() for run in rescheduled_flow_runs\n    )\n    assert all(\n        run.expected_start_time > datetime.now(timezone.utc)\n        for run in rescheduled_flow_runs\n    )\n
    ","tags":["client","API","filters","orchestration"]},{"location":"guides/using-the-client/#get-the-last-n-completed-flow-runs-from-my-workspace","title":"Get the last N completed flow runs from my workspace","text":"

    To get the last N completed flow runs from our workspace, we can make use of read_flow_runs and prefect.client.schemas.

    This example gets the last three completed flow runs from our workspace:

    import asyncio\nfrom typing import Optional\n\nfrom prefect import get_client\nfrom prefect.client.schemas.filters import FlowRunFilter\nfrom prefect.client.schemas.objects import FlowRun\nfrom prefect.client.schemas.sorting import FlowRunSort\n\nasync def get_most_recent_flow_runs(\n    n: int = 3,\n    states: Optional[list[str]] = None\n) -> list[FlowRun]:\n    if not states:\n        states = [\"COMPLETED\"]\n\n    async with get_client() as client:\n        return await client.read_flow_runs(\n            flow_run_filter=FlowRunFilter(\n                state={'type': {'any_': states}}\n            ),\n            sort=FlowRunSort.END_TIME_DESC,\n            limit=n,\n        )\n\nif __name__ == \"__main__\":\n    last_3_flow_runs: list[FlowRun] = asyncio.run(\n        get_most_recent_flow_runs()\n    )\n    print(last_3_flow_runs)\n\n    assert all(\n        run.state.is_completed() for run in last_3_flow_runs\n    )\n    assert (\n        end_times := [run.end_time for run in last_3_flow_runs]\n    ) == sorted(end_times, reverse=True)\n

    Instead of the last three from the whole workspace, you could also use the DeploymentFilter like the previous example to get the last three completed flow runs of a specific deployment.

    There are other ways to filter objects like flow runs

    See the filters API reference for more ways to filter flow runs and other objects in your Prefect ecosystem.

    ","tags":["client","API","filters","orchestration"]},{"location":"guides/variables/","title":"Variables","text":"

    Variables enable you to store and reuse non-sensitive bits of data, such as configuration information. Variables are named, mutable string values, much like environment variables. Variables are scoped to a Prefect server instance or a single workspace in Prefect Cloud.

    Variables can be created or modified at any time, but are intended for values with infrequent writes and frequent reads. Variable values may be cached for quicker retrieval.

    While variable values are most commonly loaded during flow runtime, they can be loaded in other contexts, at any time, such that they can be used to pass configuration information to Prefect configuration files, such as deployment steps.

    Variables are not Encrypted

    Using variables to store sensitive information, such as credentials, is not recommended. Instead, use Secret blocks to store and access sensitive information.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#managing-variables","title":"Managing variables","text":"

    You can create, read, edit and delete variables via the Prefect UI, API, and CLI. Names must adhere to traditional variable naming conventions:

    • Have no more than 255 characters.
    • Only contain lowercase alphanumeric characters ([a-z], [0-9]) or underscores (_). Spaces are not allowed.
    • be unique.

    Values must:

    • have less than or equal to 5000 characters.

    Optionally, you can add tags to the variable.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#via-the-prefect-ui","title":"Via the Prefect UI","text":"

    You can see all the variables in your Prefect server instance or Prefect Cloud workspace on the Variables page of the Prefect UI. Both the name and value of all variables are visible to anyone with access to the server or workspace.

    To create a new variable, select the + button next to the header of the Variables page. Enter the name and value of the variable.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#via-the-rest-api","title":"Via the REST API","text":"

    Variables can be created and deleted via the REST API. You can also set and get variables via the API with either the variable name or ID. See the REST reference for more information.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#via-the-cli","title":"Via the CLI","text":"

    You can list, inspect, and delete variables via the command line interface with the prefect variable ls, prefect variable inspect <name>, and prefect variable delete <name> commands, respectively.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#accessing-variables","title":"Accessing variables","text":"

    In addition to the UI and API, variables can be referenced in code and in certain Prefect configuration files.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#in-python-code","title":"In Python code","text":"

    You can access any variable via the Python SDK via the .get() method. If you attempt to reference a variable that does not exist, the method will return None.

    from prefect import variables\n\n# from a synchronous context\nanswer = variables.get('the_answer')\nprint(answer)\n# 42\n\n# from an asynchronous context\nanswer = await variables.get('the_answer')\nprint(answer)\n# 42\n\n# without a default value\nanswer = variables.get('not_the_answer')\nprint(answer)\n# None\n\n# with a default value\nanswer = variables.get('not_the_answer', default='42')\nprint(answer)\n# 42\n
    ","tags":["variables","blocks"],"boost":2},{"location":"guides/variables/#in-prefectyaml-deployment-steps","title":"In prefect.yaml deployment steps","text":"

    In .yaml files, variables are denoted by quotes and double curly brackets, like so: \"{{ prefect.variables.my_variable }}\". You can use variables to templatize deployment steps by referencing them in the prefect.yaml file used to create deployments. For example, you could pass a variable in to specify a branch for a git repo in a deployment pull step:

    pull:\n- prefect.deployments.steps.git_clone:\n    repository: https://github.com/PrefectHQ/hello-projects.git\n    branch: \"{{ prefect.variables.deployment_branch }}\"\n

    The deployment_branch variable will be evaluated at runtime for the deployed flow, allowing changes to be made to variables used in a pull action without updating a deployment directly.

    ","tags":["variables","blocks"],"boost":2},{"location":"guides/webhooks/","title":"Webhooks","text":"

    Use webhooks in your Prefect Cloud workspace to receive, observe, and react to events from other systems in your ecosystem. Each webhook exposes a unique URL endpoint to receive events from other systems and transforms them into Prefect events for use in automations.

    Webhooks are defined by two essential components: a unique URL and a template which translates incoming web requests to a Prefect event.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#configuring-webhooks","title":"Configuring webhooks","text":"","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#via-the-prefect-cloud-api","title":"Via the Prefect Cloud API","text":"

    Webhooks are managed via the Webhooks API endpoints. This is a Prefect Cloud-only feature. You authenticate API calls using the standard authentication methods you use with Prefect Cloud.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#via-prefect-cloud","title":"Via Prefect Cloud","text":"

    Webhooks can be created and managed from the Prefect Cloud UI.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#via-the-prefect-cli","title":"Via the Prefect CLI","text":"

    Webhooks can be managed and interacted with via the prefect cloud webhook command group.

    prefect cloud webhook --help\n

    You can create your first webhook by invoking create:

    prefect cloud webhook create your-webhook-name \\\n    --description \"Receives webhooks from your system\" \\\n    --template '{ \"event\": \"your.event.name\", \"resource\": { \"prefect.resource.id\": \"your.resource.id\" } }'\n

    Note the template string, which is discussed in greater detail down below

    You can retrieve details for a specific webhook by ID using get, or optionally query all webhooks in your workspace via ls:

    # get webhook by ID\nprefect cloud webhook get <webhook-id>\n\n# list all configured webhooks in your workspace\nprefect cloud webhook ls\n

    If you ever need to disable an existing webhook without deleting it, use toggle:

    prefect cloud webhook toggle <webhook-id>\nWebhook is now disabled\n\nprefect cloud webhook toggle <webhook-id>\nWebhook is now enabled\n

    If you are concerned that your webhook endpoint may have been compromised, use rotate to generate a new, random endpoint

    prefect cloud webhook rotate <webhook-url-slug>\n
    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#webhook-endpoints","title":"Webhook endpoints","text":"

    The webhook endpoints have randomly generated opaque URLs that do not divulge any information about your Prefect Cloud workspace. They are rooted at https://api.prefect.cloud/hooks/. For example: https://api.prefect.cloud/hooks/AERylZ_uewzpDx-8fcweHQ. Prefect Cloud assigns this URL when you create a webhook; it cannot be set via the API. You may rotate your webhook URL at any time without losing the associated configuration.

    All webhooks may accept requests via the most common HTTP methods:

    • GET, HEAD, and DELETE may be used for webhooks that define a static event template, or a template that does not depend on the body of the HTTP request. The headers of the request will be available for templates.
    • POST, PUT, and PATCH may be used when the webhook request will include a body. See How HTTP request components are handled for more details on how the body is parsed.

    Prefect Cloud webhooks are deliberately quiet to the outside world, and will only return a 204 No Content response when they are successful, and a 400 Bad Request error when there is any error interpreting the request. For more visibility when your webhooks fail, see the Troubleshooting section below.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#webhook-templates","title":"Webhook templates","text":"

    The purpose of a webhook is to accept an HTTP request from another system and produce a Prefect event from it. You may find that you often have little influence or control over the format of those requests, so Prefect's webhook system gives you full control over how you turn those notifications from other systems into meaningful events in your Prefect Cloud workspace. The template you define for each webhook will determine how individual components of the incoming HTTP request become the event name and resource labels of the resulting Prefect event.

    As with the templates available in Prefect Cloud Automation for defining notifications and other parameters, you will write templates in Jinja2. All of the built-in Jinja2 blocks and filters are available, as well as the filters from the jinja2-humanize-extensions package.

    Your goal when defining your event template is to produce a valid JSON object that defines (at minimum) the event name and the resource[\"prefect.resource.id\"], which are required of all events. The simplest template is one in which these are statically defined.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#static-webhook-events","title":"Static webhook events","text":"

    Let's see a static webhook template example. Say you want to configure a webhook that will notify Prefect when your recommendations machine learning model has been updated, so you can then send a Slack notification to your team and run a few subsequent deployments. Those models are produced on a daily schedule by another team that is using cron for scheduling. They aren't able to use Prefect for their flows (yet!), but they are happy to add a curl to the end of their daily script to notify you. Because this webhook will only be used for a single event from a single resource, your template can be entirely static:

    {\n    \"event\": \"model.refreshed\",\n    \"resource\": {\n        \"prefect.resource.id\": \"product.models.recommendations\",\n        \"prefect.resource.name\": \"Recommendations [Products]\",\n        \"producing-team\": \"Data Science\"\n    }\n}\n

    Make sure to produce valid JSON

    The output of your template, when rendered, should be a valid string that can be parsed, for example, with json.loads.

    A webhook with this template may be invoked via any of the HTTP methods, including a GET request with no body, so the team you are integrating with can include this line at the end of their daily script:

    curl https://api.prefect.cloud/hooks/AERylZ_uewzpDx-8fcweHQ\n

    Each time the script hits the webhook, the webhook will produce a single Prefect event with that name and resource in your workspace.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#event-fields-that-prefect-cloud-populates-for-you","title":"Event fields that Prefect Cloud populates for you","text":"

    You may notice that you only had to provide the event and resource definition, which is not a completely fleshed out event. Prefect Cloud will set default values for any missing fields, such as occurred and id, so you don't need to set them in your template. Additionally, Prefect Cloud will add the webhook itself as a related resource on all of the events it produces.

    If your template does not produce a payload field, the payload will default to a standard set of debugging information, including the HTTP method, headers, and body.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#dynamic-webhook-events","title":"Dynamic webhook events","text":"

    Now let's say that after a few days you and the Data Science team are getting a lot of value from the automations you have set up with the static webhook. You've agreed to upgrade this webhook to handle all of the various models that the team produces. It's time to add some dynamic information to your webhook template.

    Your colleagues on the team have adjusted their daily cron scripts to POST a small body that includes the ID and name of the model that was updated:

    curl \\\n    -d \"model=recommendations\" \\\n    -d \"friendly_name=Recommendations%20[Products]\" \\\n    -X POST https://api.prefect.cloud/hooks/AERylZ_uewzpDx-8fcweHQ\n

    This script will send a POST request and the body will include a traditional URL-encoded form with two fields describing the model that was updated: model and friendly_name. Here's the webhook code that uses Jinja to receive these values in your template and produce different events for the different models:

    {\n    \"event\": \"model.refreshed\",\n    \"resource\": {\n        \"prefect.resource.id\": \"product.models.{{ body.model }}\",\n        \"prefect.resource.name\": \"{{ body.friendly_name }}\",\n        \"producing-team\": \"Data Science\"\n    }\n}\n

    All subsequent POST requests will produce events with those variable resource IDs and names. The other statically-defined parts, such as event or the producing-team label you included earlier will still be used.

    Use Jinja2's default filter to handle missing values

    Jinja2 has a helpful default filter that can compensate for missing values in the request. In this example, you may want to use the model's ID in place of the friendly name when the friendly name is not provided: {{ body.friendly_name|default(body.model) }}.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#how-http-request-components-are-handled","title":"How HTTP request components are handled","text":"

    The Jinja2 template context includes the three parts of the incoming HTTP request:

    • method is the uppercased string of the HTTP method, like GET or POST.
    • headers is a case-insensitive dictionary of the HTTP headers included with the request. To prevent accidental disclosures, the Authorization header is removed.
    • body represents the body that was posted to the webhook, with a best-effort approach to parse it into an object you can access.

    HTTP headers are available without any alteration as a dict-like object, but you may access them with header names in any case. For example, these template expressions all return the value of the Content-Length header:

    {{ headers['Content-Length'] }}\n\n{{ headers['content-length'] }}\n\n{{ headers['CoNtEnt-LeNgTh'] }}\n

    The HTTP request body goes through some light preprocessing to make it more useful in templates. If the Content-Type of the request is application/json, the body will be parsed as a JSON object and made available to the webhook templates. If the Content-Type is application/x-www-form-urlencoded (as in our example above), the body is parsed into a flat dict-like object of key-value pairs. Jinja2 supports both index and attribute access to the fields of these objects, so the following two expressions are equivalent:

    {{ body['friendly_name'] }}\n\n{{ body.friendly_name }}\n

    Only for Python identifiers

    Jinja2's syntax only allows attribute-like access if the key is a valid Python identifier, so body.friendly-name will not work. Use body['friendly-name'] in those cases.

    You may not have much control over the client invoking your webhook, but would still like for bodies that look like JSON to be parsed as such. Prefect Cloud will attempt to parse any other content type (like text/plain) as if it were JSON first. In any case where the body cannot be transformed into JSON, it will be made available to your templates as a Python str.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#accepting-prefect-events-directly","title":"Accepting Prefect events directly","text":"

    In cases where you have more control over the client, your webhook can accept Prefect events directly with a simple pass-through template:

    {{ body|tojson }}\n

    This template accepts the incoming body (assuming it was in JSON format) and just passes it through unmodified. This allows a POST of a partial Prefect event as in this example:

    POST /hooks/AERylZ_uewzpDx-8fcweHQ HTTP/1.1\nHost: api.prefect.cloud\nContent-Type: application/json\nContent-Length: 228\n\n{\n    \"event\": \"model.refreshed\",\n    \"resource\": {\n        \"prefect.resource.id\": \"product.models.recommendations\",\n        \"prefect.resource.name\": \"Recommendations [Products]\",\n        \"producing-team\": \"Data Science\"\n    }\n}\n

    The resulting event will be filled out with the default values for occurred, id, and other fields as described above.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#accepting-cloudevents","title":"Accepting CloudEvents","text":"

    The Cloud Native Computing Foundation has standardized CloudEvents for use by systems to exchange event information in a common format. These events are supported by major cloud providers and a growing number of cloud-native systems. Prefect Cloud can interpret a webhook containing a CloudEvent natively with the following template:

    {{ body|from_cloud_event(headers) }}\n

    The resulting event will use the CloudEvent's subject as the resource (or the source if no subject is available). The CloudEvent's data attribute will become the Prefect event's payload['data'], and the other CloudEvent metadata will be at payload['cloudevents']. If you would like to handle CloudEvents in a more specific way tailored to your use case, use a dynamic template to interpret the incoming body.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/webhooks/#troubleshooting","title":"Troubleshooting","text":"

    The initial configuration of your webhook may require some trial and error as you get the sender and your receiving webhook speaking a compatible language. While you are in this phase, you may find the Event Feed in the UI to be indispensable for seeing the events as they are happening.

    When Prefect Cloud encounters an error during receipt of a webhook, it will produce a prefect-cloud.webhook.failed event in your workspace. This event will include critical information about the HTTP method, headers, and body it received, as well as what the template rendered. Keep an eye out for these events when something goes wrong.

    ","tags":["events","automations","triggers","webhooks","Prefect Cloud"]},{"location":"guides/deployment/aci/","title":"Run an Agent with Azure Container Instances","text":"

    Microsoft Azure Container Instances (ACI) provides a convenient and simple service for quickly spinning up a Docker container that can host a Prefect Agent and execute flow runs.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#prerequisites","title":"Prerequisites","text":"

    To follow this quickstart, you'll need the following:

    • A Prefect Cloud account
    • A Prefect Cloud API key (Prefect Cloud Pro and Enterprise tier accounts can use a service account API key)
    • A Microsoft Azure account
    • Azure CLI installed and authenticated
    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#create-a-resource-group","title":"Create a resource group","text":"

    Like most Azure resources, ACI applications must live in a resource group. If you don\u2019t already have a resource group you\u2019d like to use, create a new one by running the az group create command. For example, this example creates a resource group called prefect-agents in the eastus region:

    az group create --name prefect-agents --location eastus\n

    Feel free to change the group name or location to match your use case. You can also run az account list-locations -o table to see all available resource group locations for your account.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#create-the-container-instance","title":"Create the container instance","text":"

    Prefect provides pre-configured Docker images you can use to quickly stand up a container instance. These Docker images include Python and Prefect. For example, the image prefecthq/prefect:2-python3.10 includes the latest release version of Prefect and Python 3.10.

    To create the container instance, use the az container create command. This example shows the syntax, but you'll need to provide the correct values for [ACCOUNT-ID],[WORKSPACE-ID], [API-KEY], and any dependencies you need to pip install on the instance. These options are discussed below.

    az container create \\\n--resource-group prefect-agents \\\n--name prefect-agent-example \\\n--image prefecthq/prefect:2-python3.10 \\\n--secure-environment-variables PREFECT_API_URL='https://api.prefect.cloud/api/accounts/[ACCOUNT-ID]/workspaces/[WORKSPACE-ID]' PREFECT_API_KEY='[API-KEY]' \\\n--command-line \"/bin/bash -c 'pip install adlfs s3fs requests pandas; prefect agent start -p default-agent-pool -q test'\"\n

    When the container instance is running, go to Prefect Cloud and select the Work Pools page. Select default-agent-pool, then select the Queues tab to see work queues configured on this work pool. When the container instance is running and the agent has started, the test work queue displays \"Healthy\" status. This work queue and agent are ready to execute deployments configured to run on the test queue.

    Agents and queues

    The agent running in this container instance can now pick up and execute flow runs for any deployment configured to use the test queue on the default-agent-pool work pool.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#container-create-options","title":"Container create options","text":"

    Let's break down the details of the az container create command used here.

    The az container create command creates a new ACI container.

    --resource-group prefect-agents tells Azure which resource group the new container is created in. Here, the examples uses the prefect-agents resource group created earlier.

    --name prefect-agent-example determines the container name you will see in the Azure Portal. You can set any name you\u2019d like here to suit your use case, but container instance names must be unique in your resource group.

    --image prefecthq/prefect:2-python3.10 tells ACI which Docker images to run. The script above pulls a public Prefect image from Docker Hub. You can also build custom images and push them to a public container registry so ACI can access them. Or you can push your image to a private Azure Container Registry and use it to create a container instance.

    --secure-environment-variables sets environment variables that are only visible from inside the container. They do not show up when viewing the container\u2019s metadata. You'll populate these environment variables with a few pieces of information to configure the execution environment of the container instance so it can communicate with your Prefect Cloud workspace:

    • A Prefect Cloud [PREFECT_API_KEY]/concepts/settings/#prefect_api_key) value specifying the API key used to authenticate with your Prefect Cloud workspace. (Pro and Enterprise tier accounts can use a service account API key.)
    • The PREFECT_API_URL value specifying the API endpoint of your Prefect Cloud workspace.

    --command-line lets you override the container\u2019s normal entry point and run a command instead. The script above uses this section to install the adlfs pip package so it can read flow code from Azure Blob Storage, along with s3fs, pandas, and requests. It then runs the Prefect agent, in this case using the default work pool and a test work queue. If you want to use a different work pool or queue, make sure to change these values appropriately.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#create-a-deployment","title":"Create a deployment","text":"

    Following the example of the Flow deployments tutorial, let's create a deployment that can be executed by the agent on this container instance.

    In an environment where you have installed Prefect, create a new folder called health_test, and within it create a new file called health_flow.py containing the following code.

    import prefect\nfrom prefect import task, flow\nfrom prefect import get_run_logger\n\n\n@task\ndef say_hi():\n    logger = get_run_logger()\n    logger.info(\"Hello from the Health Check Flow! \ud83d\udc4b\")\n\n\n@task\ndef log_platform_info():\n    import platform\n    import sys\n    from prefect.server.api.server import SERVER_API_VERSION\n\n    logger = get_run_logger()\n    logger.info(\"Host's network name = %s\", platform.node())\n    logger.info(\"Python version = %s\", platform.python_version())\n    logger.info(\"Platform information (instance type) = %s \", platform.platform())\n    logger.info(\"OS/Arch = %s/%s\", sys.platform, platform.machine())\n    logger.info(\"Prefect Version = %s \ud83d\ude80\", prefect.__version__)\n    logger.info(\"Prefect API Version = %s\", SERVER_API_VERSION)\n\n\n@flow(name=\"Health Check Flow\")\ndef health_check_flow():\n    hi = say_hi()\n    log_platform_info(wait_for=[hi])\n

    Now create a deployment for this flow script, making sure that it's configured to use the test queue on the default-agent-pool work pool.

    prefect deployment build --infra process --storage-block azure/flowsville/health_test --name health-test --pool default-agent-pool --work-queue test --apply health_flow.py:health_check_flow\n

    Once created, any flow runs for this deployment will be picked up by the agent running on this container instance.

    Infrastructure and storage

    This Prefect deployment example was built using the Process infrastructure type and Azure Blob Storage.

    You might wonder why your deployment needs process infrastructure rather than DockerContainer infrastructure when you are deploying a Docker image to ACI.

    A Prefect deployment\u2019s infrastructure type describes how you want Prefect agents to run flows for the deployment. With DockerContainer infrastructure, the agent will try to use Docker to spin up a new container for each flow run. Since you\u2019ll be starting your own container on ACI, you don\u2019t need Prefect to do it for you. Specifying process infrastructure on the deployment tells Prefect you want to agent to run flows by starting a process in your ACI container.

    You can use any storage type as long as you've configured a block for it before creating the deployment.

    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/aci/#cleaning-up","title":"Cleaning up","text":"

    Note that ACI instances may incur usage charges while running, but must be running for the agent to pick up and execute flow runs.

    To stop a container, use the az container stop command:

    az container stop --resource-group prefect-agents --name prefect-agent-example\n

    To delete a container, use the az container delete command:

    az container delete --resource-group prefect-agents --name prefect-agent-example\n
    ","tags":["Docker","containers","agents","cloud"],"boost":2},{"location":"guides/deployment/daemonize/","title":"Daemonize Processes for Prefect Deployments","text":"

    When running workflow applications, it can be helpful to create long-running processes that run at startup and are robust to failure. In this guide you'll learn how to set up a systemd service to create long-running Prefect processes that poll for scheduled flow runs.

    A systemd service is ideal for running a long-lived process on a Linux VM or physical Linux server. We will leverage systemd and see how to automatically start a Prefect worker or long-lived serve process when Linux starts. This approach provides resilience by automatically restarting the process if it crashes.

    In this guide we will:

    • Create a Linux user
    • Install and configure Prefect
    • Set up a systemd service for the Prefect worker or .serve process
    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#prerequisites","title":"Prerequisites","text":"
    • An environment with a linux operating system with systemd and Python 3.8 or later.
    • A superuser account (you can run sudo commands).
    • A Prefect Cloud account, or a local instance of a Prefect server running on your network.
    • If daemonizing a worker, you'll need a Prefect deployment with a work pool your worker can connect to.

    If using an AWS t2-micro EC2 instance with an AWS Linux image, you can install Python and pip with sudo yum install -y python3 python3-pip.

    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#step-1-add-a-user","title":"Step 1: Add a user","text":"

    Create a user account on your linux system for the Prefect process. While you can run a worker or serve process as root, it's good security practice to avoid doing so unless you are sure you need to.

    In a terminal, run:

    sudo useradd -m prefect\nsudo passwd prefect\n

    When prompted, enter a password for the prefect account.

    Next, log in to the prefect account by running:

    sudo su prefect\n
    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#step-2-install-prefect","title":"Step 2: Install Prefect","text":"

    Run:

    pip3 install prefect\n

    This guide assumes you are installing Prefect globally, not in a virtual environment. If running a systemd service in a virtual environment, you'll just need to change the ExecPath. For example, if using venv, change the ExecPath to target the prefect application in the bin subdirectory of your virtual environment.

    Next, set up your environment so that the Prefect client will know which server to connect to.

    If connecting to Prefect Cloud, follow the instructions to obtain an API key and then run the following:

    prefect cloud login -k YOUR_API_KEY\n

    When prompted, choose the Prefect workspace you'd like to log in to.

    If connecting to a self-hosted Prefect server instance instead of Prefect Cloud, run the following and substitute the IP address of your server:

    prefect config set PREFECT_API_URL=http://your-prefect-server-IP:4200\n

    Finally, run the exit command to sign out of the prefect Linux account. This command switches you back to your sudo-enabled account so you will can run the commands in the next section.

    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#step-3-set-up-a-systemd-service","title":"Step 3: Set up a systemd service","text":"Prefect worker.serve

    Move into the /etc/systemd/system folder and open a file for editing. We use the Vim text editor below.

    cd /etc/systemd/system\nsudo vim my-prefect-service.service\n
    my-prefect-service.service
    [Unit]\nDescription=Prefect worker\n\n[Service]\nUser=prefect\nWorkingDirectory=/home\nExecStart=prefect worker start --pool YOUR_WORK_POOL_NAME\nRestart=always\n\n[Install]\nWantedBy=multi-user.target\n

    Make sure you substitute your own work pool name.

    Copy your flow entrypoint Python file and any other files needed for your flow to run into the /home directory (or the directory of your choice).

    Here's a basic example flow:

    my_file.py
    from prefect import flow\n\n\n@flow(log_prints=True)\ndef say_hi():\n    print(\"Hello!\")\n\nif __name__==\"__main__\":\n    say_hi.serve(name=\"Greeting from daemonized .serve\")\n

    If you want to make changes to your flow code without restarting your process, you can push your code to git-based cloud storage (GitHub, BitBucket, GitLab) and use flow.from_source().serve(), as in the example below.

    my_remote_flow_code_file.py
    if __name__ == \"__main__\":\n  flow.from_source(\n      source=\"https://github.com/org/repo.git\",\n      entrypoint=\"path/to/my_remote_flow_code_file.py:say_hi\",\n  ).serve(name=\"deployment-with-github-storage\")\n

    Make sure you substitute your own flow code entrypoint path.

    Note that if you change the flow entrypoint parameters, you will need to restart the process.

    Move into the /etc/systemd/system folder and open a file for editing. We use the Vim text editor below.

    cd /etc/systemd/system\nsudo vim my-prefect-service.service\n
    my-prefect-service.service
    [Unit]\nDescription=Prefect serve \n\n[Service]\nUser=prefect\nWorkingDirectory=/home\nExecStart=python3 my_file.py\nRestart=always\n\n[Install]\nWantedBy=multi-user.target\n

    To save the file and exit Vim hit the escape key, type :wq!, then press the return key.

    Next, run sudo systemctl daemon-reload to make systemd aware of your new service.

    Then, run sudo systemctl enable my-prefect-service to enable the service. This command will ensure it runs when your system boots.

    Next, run sudo systemctl start my-prefect-service to start the service.

    Run your deployment from UI and check out the logs on the Flow Runs page.

    You can see if your daemonized Prefect worker or serve process is running and see the Prefect logs with systemctl status my-prefect-service.

    That's it! You now have a systemd service that starts when your system boots, and will restart if it ever crashes.

    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/daemonize/#next-steps","title":"Next steps","text":"

    If you want to set up a long-lived process on a Windows machine the pattern is similar. Instead of systemd, you can use NSSM.

    Check out other Prefect guides to see what else you can do with Prefect!

    ","tags":["systemd","daemonize","worker"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/","title":"Developing a New Worker Type","text":"

    Advanced Topic

    This tutorial is for users who want to extend the Prefect framework and completing this successfully will require deep knowledge of Prefect concepts. For standard use cases, we recommend using one of the available workers instead.

    Prefect workers are responsible for setting up execution infrastructure and starting flow runs on that infrastructure.

    A list of available workers can be found here. What if you want to execute your flow runs on infrastructure that doesn't have an available worker type? This tutorial will walk you through creating a custom worker that can run your flows on your chosen infrastructure.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#worker-configuration","title":"Worker configuration","text":"

    When setting up an execution environment for a flow run, a worker receives configuration for the infrastructure it is designed to work with. Examples of configuration values include memory allocation, CPU allocation, credentials, image name, etc. The worker then uses this configuration to create the execution environment and start the flow run.

    How are the configuration values populated?

    The work pool that a worker polls for flow runs has a base job template associated with it. The template is the contract for how configuration values populate for each flow run.

    The keys in the job_configuration section of this base job template match the worker's configuration class attributes. The values in the job_configuration section of the base job template are used to populate the attributes of the worker's configuration class.

    The work pool creator gets to decide how they want to populate the values in the job_configuration section of the base job template. The values can be hard-coded, templated using placeholders, or a mix of these two approaches. Because you, as the worker developer, don't know how the work pool creator will populate the values, you should set sensible defaults for your configuration class attributes as a matter of best practice.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#implementing-a-basejobconfiguration-subclass","title":"Implementing a BaseJobConfiguration subclass","text":"

    A worker developer defines their worker's configuration to function with a class extending BaseJobConfiguration.

    BaseJobConfiguration has attributes that are common to all workers:

    Attribute Description name The name to assign to the created execution environment. env Environment variables to set in the created execution environment. labels The labels assigned to the created execution environment for metadata purposes. command The command to use when starting a flow run.

    Prefect sets values for each attribute before giving the configuration to the worker. If you want to customize the values of these attributes, use the prepare_for_flow_run method.

    Here's an example prepare_for_flow_run method that adds a label to the execution environment:

    def prepare_for_flow_run(\n    self, flow_run, deployment = None, flow = None,\n):  \n    super().prepare_for_flow_run(flow_run, deployment, flow)  \n    self.labels.append(\"my-custom-label\")\n

    A worker configuration class is a Pydantic model, so you can add additional attributes to your configuration class as Pydantic fields. For example, if you want to allow memory and CPU requests for your worker, you can do so like this:

    from pydantic import Field\nfrom prefect.workers.base import BaseJobConfiguration\n\nclass MyWorkerConfiguration(BaseJobConfiguration):\n    memory: int = Field(\n            default=1024,\n            description=\"Memory allocation for the execution environment.\"\n        )\n    cpu: int = Field(\n            default=500, \n            description=\"CPU allocation for the execution environment.\"\n        )\n

    This configuration class will populate the job_configuration section of the resulting base job template.

    For this example, the base job template would look like this:

    job_configuration:\n    name: \"{{ name }}\"\n    env: \"{{ env }}\"\n    labels: \"{{ labels }}\"\n    command: \"{{ command }}\"\n    memory: \"{{ memory }}\"\n    cpu: \"{{ cpu }}\"\nvariables:\n    type: object\n    properties:\n        name:\n          title: Name\n          description: Name given to infrastructure created by a worker.\n          type: string\n        env:\n          title: Environment Variables\n          description: Environment variables to set when starting a flow run.\n          type: object\n          additionalProperties:\n            type: string\n        labels:\n          title: Labels\n          description: Labels applied to infrastructure created by a worker.\n          type: object\n          additionalProperties:\n            type: string\n        command:\n          title: Command\n          description: The command to use when starting a flow run. In most cases,\n            this should be left blank and the command will be automatically generated\n            by the worker.\n          type: string\n        memory:\n            title: Memory\n            description: Memory allocation for the execution environment.\n            type: integer\n            default: 1024\n        cpu:\n            title: CPU\n            description: CPU allocation for the execution environment.\n            type: integer\n            default: 500\n

    This base job template defines what values can be provided by deployment creators on a per-deployment basis and how those provided values will be translated into the configuration values that the worker will use to create the execution environment.

    Notice that each attribute for the class was added in the job_configuration section with placeholders whose name matches the attribute name. The variables section was also populated with the OpenAPI schema for each attribute. If a configuration class is used without explicitly declaring any template variables, the template variables will be inferred from the configuration class attributes.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#customizing-configuration-attribute-templates","title":"Customizing Configuration Attribute Templates","text":"

    You can customize the template for each attribute for situations where the configuration values should use more sophisticated templating. For example, if you want to add units for the memory attribute, you can do so like this:

    from pydantic import Field\nfrom prefect.workers.base import BaseJobConfiguration\n\nclass MyWorkerConfiguration(BaseJobConfiguration):\n    memory: str = Field(\n            default=\"1024Mi\",\n            description=\"Memory allocation for the execution environment.\"\n            template=\"{{ memory_request }}Mi\"\n        )\n    cpu: str = Field(\n            default=\"500m\", \n            description=\"CPU allocation for the execution environment.\"\n            template=\"{{ cpu_request }}m\"\n        )\n

    Notice that we changed the type of each attribute to str to accommodate the units, and we added a new template attribute to each attribute. The template attribute is used to populate the job_configuration section of the resulting base job template.

    For this example, the job_configuration section of the resulting base job template would look like this:

    job_configuration:\n    name: \"{{ name }}\"\n    env: \"{{ env }}\"\n    labels: \"{{ labels }}\"\n    command: \"{{ command }}\"\n    memory: \"{{ memory_request }}Mi\"\n    cpu: \"{{ cpu_request }}m\"\n

    Note that to use custom templates, you will need to declare the template variables used in the template because the names of those variables can no longer be inferred from the configuration class attributes. We will cover how to declare the default variable schema in the Worker Template Variables section.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#rules-for-template-variable-interpolation","title":"Rules for template variable interpolation","text":"

    When defining a job configuration model, it's useful to understand how template variables are interpolated into the job configuration. The templating engine follows a few simple rules:

    1. If a template variable is the only value for a key in the job_configuration section, the key will be replaced with the value template variable.
    2. If a template variable is part of a string (i.e., there is text before or after the template variable), the value of the template variable will be interpolated into the string.
    3. If a template variable is the only value for a key in the job_configuration section and no value is provided for the template variable, the key will be removed from the job_configuration section.

    These rules allow worker developers and work pool maintainers to define template variables that can be complex types like dictionaries and lists. These rules also mean that worker developers should give reasonable default values to job configuration fields whenever possible because values are not guaranteed to be provided if template variables are unset.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#template-variable-usage-strategies","title":"Template variable usage strategies","text":"

    Template variables define the interface that deployment creators interact with to configure the execution environments of their deployments. The complexity of this interface can be controlled via the template variables that are defined for a base job template. This control allows work pool maintainers to find a point along the spectrum of flexibility and simplicity appropriate for their organization.

    There are two patterns that are represented in current worker implementations:

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#pass-through","title":"Pass-through","text":"

    In the pass-through pattern, template variables are passed through to the job configuration with little change. This pattern exposes complete control to deployment creators but also requires them to understand the details of the execution environment.

    This pattern is useful when the execution environment is simple, and the deployment creators are expected to have high technical knowledge.

    The Docker worker is an example of a worker that uses this pattern.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#infrastructure-as-code-templating","title":"Infrastructure as code templating","text":"

    Depending on the infrastructure they interact with, workers can sometimes employ a declarative infrastructure syntax (i.e., infrastructure as code) to create execution environments (e.g., a Kubernetes manifest or an ECS task definition).

    In the IaC pattern, it's often useful to use template variables to template portions of the declarative syntax which then can be used to generate the declarative syntax into a final form.

    This approach allows work pool creators to provide a simpler interface to deployment creators while also controlling which portions of infrastructure are configurable by deployment creators.

    The Kubernetes worker is an example of a worker that uses this pattern.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#configuring-credentials","title":"Configuring credentials","text":"

    When executing flow runs within cloud services, workers will often need credentials to authenticate with those services. For example, a worker that executes flow runs in AWS Fargate will need AWS credentials. As a worker developer, you can use blocks to accept credentials configuration from the user.

    For example, if you want to allow the user to configure AWS credentials, you can do so like this:

    from prefect_aws import AwsCredentials\n\nclass MyWorkerConfiguration(BaseJobConfiguration):\n    aws_credentials: AwsCredentials = Field(\n        default=None,\n        description=\"AWS credentials to use when creating AWS resources.\"\n    )\n

    Users can create and assign a block to the aws_credentials attribute in the UI and the worker will use these credentials when interacting with AWS resources.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#worker-template-variables","title":"Worker template variables","text":"

    Providing template variables for a base job template defines the fields that deployment creators can override per deployment. The work pool creator ultimately defines the template variables for a base job template, but the worker developer is able to define default template variables for the worker to make it easier to use.

    Default template variables for a worker are defined by implementing the BaseVariables class. Like the BaseJobConfiguration class, the BaseVariables class has attributes that are common to all workers:

    Attribute Description name The name to assign to the created execution environment. env Environment variables to set in the created execution environment. labels The labels assigned the created execution environment for metadata purposes. command The command to use when starting a flow run.

    Additional attributes can be added to the BaseVariables class to define additional template variables. For example, if you want to allow memory and CPU requests for your worker, you can do so like this:

    from pydantic import Field\nfrom prefect.workers.base import BaseVariables\n\nclass MyWorkerTemplateVariables(BaseVariables):\n    memory_request: int = Field(\n            default=1024,\n            description=\"Memory allocation for the execution environment.\"\n        )\n    cpu_request: int = Field(\n            default=500, \n            description=\"CPU allocation for the execution environment.\"\n        )\n

    When MyWorkerTemplateVariables is used in conjunction with MyWorkerConfiguration from the Customizing Configuration Attribute Templates section, the resulting base job template will look like this:

    job_configuration:\n    name: \"{{ name }}\"\n    env: \"{{ env }}\"\n    labels: \"{{ labels }}\"\n    command: \"{{ command }}\"\n    memory: \"{{ memory_request }}Mi\"\n    cpu: \"{{ cpu_request }}m\"\nvariables:\n    type: object\n    properties:\n        name:\n          title: Name\n          description: Name given to infrastructure created by a worker.\n          type: string\n        env:\n          title: Environment Variables\n          description: Environment variables to set when starting a flow run.\n          type: object\n          additionalProperties:\n            type: string\n        labels:\n          title: Labels\n          description: Labels applied to infrastructure created by a worker.\n          type: object\n          additionalProperties:\n            type: string\n        command:\n          title: Command\n          description: The command to use when starting a flow run. In most cases,\n            this should be left blank and the command will be automatically generated\n            by the worker.\n          type: string\n        memory_request:\n            title: Memory Request\n            description: Memory allocation for the execution environment.\n            type: integer\n            default: 1024\n        cpu_request:\n            title: CPU Request\n            description: CPU allocation for the execution environment.\n            type: integer\n            default: 500\n

    Note that template variable classes are never used directly. Instead, they are used to generate a schema that is used to populate the variables section of a base job template and validate the template variables provided by the user.

    We don't recommend using template variable classes within your worker implementation for validation purposes because the work pool creator ultimately defines the template variables. The configuration class should handle any necessary run-time validation.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#worker-implementation","title":"Worker implementation","text":"

    Workers set up execution environments using provided configuration. Workers also observe the execution environment as the flow run executes and report any crashes to the Prefect API.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#attributes","title":"Attributes","text":"

    To implement a worker, you must implement the BaseWorker class and provide it with the following attributes:

    Attribute Description Required type The type of the worker. Yes job_configuration The configuration class for the worker. Yes job_configuration_variables The template variables class for the worker. No _documentation_url Link to documentation for the worker. No _logo_url Link to a logo for the worker. No _description A description of the worker. No","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#methods","title":"Methods","text":"","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#run","title":"run","text":"

    In addition to the attributes above, you must also implement a run method. The run method is called for each flow run the worker receives for execution from the work pool.

    The run method has the following signature:

     async def run(\n        self, flow_run: FlowRun, configuration: BaseJobConfiguration, task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> BaseWorkerResult:\n        ...\n

    The run method is passed: the flow run to execute, the execution environment configuration for the flow run, and a task status object that allows the worker to track whether the flow run was submitted successfully.

    The run method must also return a BaseWorkerResult object. The BaseWorkerResult object returned contains information about the flow run execution. For the most part, you can implement the BaseWorkerResult with no modifications like so:

    from prefect.workers.base import BaseWorkerResult\n\nclass MyWorkerResult(BaseWorkerResult):\n    \"\"\"Result returned by the MyWorker.\"\"\"\n

    If you would like to return more information about a flow run, then additional attributes can be added to the BaseWorkerResult class.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#kill_infrastructure","title":"kill_infrastructure","text":"

    Workers must implement a kill_infrastructure method to support flow run cancellation. The kill_infrastructure method is called when a flow run is canceled and is passed an identifier for the infrastructure to tear down and the execution environment configuration for the flow run.

    The infrastructure_pid passed to the kill_infrastructure method is the same identifier used to mark a flow run execution as started in the run method. The infrastructure_pid must be a string, but it can take on any format you choose.

    The infrastructure_pid should contain enough information to uniquely identify the infrastructure created for a flow run when used with the job_configuration passed to the kill_infrastructure method. Examples of useful information include: the cluster name, the hostname, the process ID, the container ID, etc.

    If a worker cannot tear down infrastructure created for a flow run, the kill_infrastructure command should raise an InfrastructureNotFound or InfrastructureNotAvailable exception.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#worker-implementation-example","title":"Worker implementation example","text":"

    Below is an example of a worker implementation. This example is not intended to be a complete implementation but to illustrate the aforementioned concepts.

    from prefect.workers.base import BaseWorker, BaseWorkerResult, BaseJobConfiguration, BaseVariables\n\nclass MyWorkerConfiguration(BaseJobConfiguration):\n    memory: str = Field(\n            default=\"1024Mi\",\n            description=\"Memory allocation for the execution environment.\"\n            template=\"{{ memory_request }}Mi\"\n        )\n    cpu: str = Field(\n            default=\"500m\", \n            description=\"CPU allocation for the execution environment.\"\n            template=\"{{ cpu_request }}m\"\n        )\n\nclass MyWorkerTemplateVariables(BaseVariables):\n    memory_request: int = Field(\n            default=1024,\n            description=\"Memory allocation for the execution environment.\"\n        )\n    cpu_request: int = Field(\n            default=500, \n            description=\"CPU allocation for the execution environment.\"\n        )\n\nclass MyWorkerResult(BaseWorkerResult):\n    \"\"\"Result returned by the MyWorker.\"\"\"\n\nclass MyWorker(BaseWorker):\n    type = \"my-worker\"\n    job_configuration = MyWorkerConfiguration\n    job_configuration_variables = MyWorkerTemplateVariables\n    _documentation_url = \"https://example.com/docs\"\n    _logo_url = \"https://example.com/logo\"\n    _description = \"My worker description.\"\n\n    async def run(\n        self, flow_run: FlowRun, configuration: BaseJobConfiguration, task_status: Optional[anyio.abc.TaskStatus] = None,\n    ) -> BaseWorkerResult:\n        # Create the execution environment and start execution\n        job = await self._create_and_start_job(configuration)\n\n        if task_status:\n            # Use a unique ID to mark the run as started. This ID is later used to tear down infrastructure\n            # if the flow run is cancelled.\n            task_status.started(job.id) \n\n        # Monitor the execution\n        job_status = await self._watch_job(job, configuration)\n\n        exit_code = job_status.exit_code if job_status else -1 # Get result of execution for reporting\n        return MyWorkerResult(\n            status_code=exit_code,\n            identifier=job.id,\n        )\n\n    async def kill_infrastructure(self, infrastructure_pid: str, configuration: BaseJobConfiguration) -> None:\n        # Tear down the execution environment\n        await self._kill_job(infrastructure_pid, configuration)\n

    Most of the execution logic is omitted from the example above, but it shows that the typical order of operations in the run method is: 1. Create the execution environment and start the flow run execution 2. Mark the flow run as started via the passed task_status object 3. Monitor the execution 4. Get the execution's final status from the infrastructure and return a BaseWorkerResult object

    To see other examples of worker implementations, see the ProcessWorker and KubernetesWorker implementations.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/developing-a-new-worker-type/#integrating-with-the-prefect-cli","title":"Integrating with the Prefect CLI","text":"

    Workers can be started via the Prefect CLI by providing the --type option to the prefect worker start CLI command. To make your worker type available via the CLI, it must be available at import time.

    If your worker is in a package, you can add an entry point to your setup file in the following format:

    entry_points={\n    \"prefect.collections\": [\n        \"my_package_name = my_worker_module\",\n    ]\n},\n

    Prefect will discover this entry point and load your work module in the specified module. The entry point will allow the worker to be available via the CLI.

    ","tags":["work pools","workers","orchestration","flow runs","deployments","storage","infrastructure","tutorial","recipes"],"boost":2},{"location":"guides/deployment/kubernetes/","title":"Running Flows with Kubernetes","text":"

    This guide will walk you through running your flows on Kubernetes. Though much of the guide is general to any Kubernetes cluster, there are differences between the managed Kubernetes offerings between cloud providers, especially when it comes to container registries and access management. We'll focus on Amazon Elastic Kubernetes Service (EKS).

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#prerequisites","title":"Prerequisites","text":"

    Before we begin, there are a few pre-requisites:

    1. A Prefect Cloud account
    2. A cloud provider (AWS, GCP, or Azure) account
    3. Install Python and Prefect
    4. Install Helm
    5. Install the Kubernetes CLI (kubectl)

    Administrator Access

    Though not strictly necessary, you may want to ensure you have admin access, both in Prefect Cloud and in your cloud provider. Admin access is only necessary during the initial setup and can be downgraded after.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-cluster","title":"Create a cluster","text":"

    Let's start by creating a new cluster. If you already have one, skip ahead to the next section.

    AWSGCPAzure

    One easy way to get set up with a cluster in EKS is with eksctl. Node pools can be backed by either EC2 instances or FARGATE. Let's choose FARGATE so there's less to manage. The following command takes around 15 minutes and must not be interrupted:

    # Replace the cluster name with your own value\neksctl create cluster --fargate --name <CLUSTER-NAME>\n\n# Authenticate to the cluster.\naws eks update-kubeconfig --name <CLUSTER-NAME>\n

    You can get a GKE cluster up and running with a few commands using the gcloud CLI. We'll build a bare-bones cluster that is accessible over the open internet - this should not be used in a production environment. To deploy the cluster, your project must have a VPC network configured.

    First, authenticate to GCP by setting the following configuration options.

    # Authenticate to gcloud\ngcloud auth login\n\n# Specify the project & zone to deploy the cluster to\n# Replace the project name with your GCP project name\ngcloud config set project <GCP-PROJECT-NAME>\ngcloud config set compute/zone <AVAILABILITY-ZONE>\n

    Next, deploy the cluster - this command will take ~15 minutes to complete. Once the cluster has been created, authenticate to the cluster.

    # Create cluster\n# Replace the cluster name with your own value\ngcloud container clusters create <CLUSTER-NAME> --num-nodes=1 \\\n--machine-type=n1-standard-2\n\n# Authenticate to the cluster\ngcloud container clusters <CLUSTER-NAME> --region <AVAILABILITY-ZONE>\n

    GCP Gotchas

    • You'll need to enable the default service account in the IAM console, or specify a different service account with the appropriate permissions to be used.
    ERROR: (gcloud.container.clusters.create) ResponseError: code=400, message=Service account \"000000000000-compute@developer.gserviceaccount.com\" is disabled.\n
    • Organization policy blocks creation of external (public) IPs. You can override this policy (if you have the appropriate permissions) under the Organizational Policy page within IAM.
    creation failed: Constraint constraints/compute.vmExternalIpAccess violated for project 000000000000. Add instance projects/<GCP-PROJECT-NAME>/zones/us-east1-b/instances/gke-gke-guide-1-default-pool-c369c84d-wcfl to the constraint to use external IP with it.\"\n

    You can quickly create an AKS cluster using the Azure CLI, or use the Cloud Shell directly from the Azure portal shell.azure.com.

    First, authenticate to Azure if not already done.

      az login\n

    Next, deploy the cluster - this command will take ~4 minutes to complete. Once the cluster has been created, authenticate to the cluster.

      # Create a Resource Group at the desired location, e.g. westus\n  az group create --name <RESOURCE-GROUP-NAME> --location <LOCATION>\n\n  # Create a kubernetes cluster with default kubernetes version, default SKU load balancer (Standard) and default vm set type (VirtualMachineScaleSets)\n  az aks create --resource-group <RESOURCE-GROUP-NAME> --name <CLUSTER-NAME>\n\n  # Configure kubectl to connect to your Kubernetes cluster\n  az aks get-credentials --resource-group <RESOURCE-GROUP-NAME> --name <CLUSTER-NAME>\n\n  # Verify the connection by listing the cluster nodes\n  kubectl get nodes\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-container-registry","title":"Create a container registry","text":"

    Besides a cluster, the other critical resource we'll need is a container registry. A registry is not strictly required, but in most cases you'll want to use custom images and/or have more control over where images are stored. If you already have a registry, skip ahead to the next section.

    AWSGCPAzure

    Let's create a registry using the AWS CLI and authenticate the docker daemon to said registry:

    # Replace the image name with your own value\naws ecr create-repository --repository-name <IMAGE-NAME>\n\n# Login to ECR\n# Replace the region and account ID with your own values\naws ecr get-login-password --region <REGION> | docker login \\\n  --username AWS --password-stdin <AWS_ACCOUNT_ID>.dkr.ecr.<REGION>.amazonaws.com\n

    Let's create a registry using the gcloud CLI and authenticate the docker daemon to said registry:

    # Create artifact registry repository to host your custom image\n# Replace the repository name with your own value; it can be the \n# same name as your image\ngcloud artifacts repositories create <REPOSITORY-NAME> \\\n--repository-format=docker --location=us\n\n# Authenticate to artifact registry\ngcloud auth configure-docker us-docker.pkg.dev\n

    Let's create a registry using the Azure CLI and authenticate the docker daemon to said registry:

    # Name must be a lower-case alphanumeric\n# Tier SKU can easily be updated later, e.g. az acr update --name <REPOSITORY-NAME> --sku Standard\naz acr create --resource-group <RESOURCE-GROUP-NAME> \\\n  --name <REPOSITORY-NAME> \\\n  --sku Basic\n\n# Attach ACR to AKS cluster\n# You need Owner, Account Administrator, or Co-Administrator role on your Azure subscription as per Azure docs\naz aks update --resource-group <RESOURCE-GROUP-NAME> --name <CLUSTER-NAME> --attach-acr <REPOSITORY-NAME>\n\n# You can verify AKS can now reach ACR\naz aks check-acr --resource-group RESOURCE-GROUP-NAME> --name <CLUSTER-NAME> --acr <REPOSITORY-NAME>.azurecr.io\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-kubernetes-work-pool","title":"Create a Kubernetes work pool","text":"

    Work pools allow you to manage deployment infrastructure. We'll configure the default values for our Kubernetes base job template. Note that these values can be overridden by individual deployments.

    Let's switch to the Prefect Cloud UI, where we'll create a new Kubernetes work pool (alternatively, you could use the Prefect CLI to create a work pool).

    1. Click on the Work Pools tab on the left sidebar
    2. Click the + button at the top of the page
    3. Select Kubernetes as the work pool type
    4. Click Next to configure the work pool settings

    Let's look at a few popular configuration options.

    Environment Variables

    Add environment variables to set when starting a flow run. So long as you are using a Prefect-maintained image and haven't overwritten the image's entrypoint, you can specify Python packages to install at runtime with {\"EXTRA_PIP_PACKAGES\":\"my_package\"}. For example {\"EXTRA_PIP_PACKAGES\":\"pandas==1.2.3\"} will install pandas version 1.2.3. Alternatively, you can specify package installation in a custom Dockerfile, which can allow you to take advantage of image caching. As we'll see below, Prefect can help us create a Dockerfile with our flow code and the packages specified in a requirements.txt file baked in.

    Namespace

    Set the Kubernetes namespace to create jobs within, such as prefect. By default, set to default.

    Image

    Specify the Docker container image for created jobs. If not set, the latest Prefect 2 image will be used (i.e. prefecthq/prefect:2-latest). Note that you can override this on each deployment through job_variables.

    Image Pull Policy

    Select from the dropdown options to specify when to pull the image. When using the IfNotPresent policy, make sure to use unique image tags, as otherwise old images could get cached on your nodes.

    Finished Job TTL

    Number of seconds before finished jobs are automatically cleaned up by Kubernetes' controller. You may want to set to 60 so that completed flow runs are cleaned up after a minute.

    Pod Watch Timeout Seconds

    Number of seconds for pod creation to complete before timing out. Consider setting to 300, especially if using a serverless type node pool, as these tend to have longer startup times.

    Kubernetes Cluster Config

    You can configure the Kubernetes cluster to use for job creation by specifying a KubernetesClusterConfig block. Generally you should leave the cluster config blank as the worker should be provisioned with appropriate access and permissions. Typically this setting is used when a worker is deployed to a cluster that is different from the cluster where flow runs are executed.

    Advanced Settings

    Want to modify the default base job template to add other fields or delete existing fields?

    Select the Advanced tab and edit the JSON representation of the base job template.

    For example, to set a CPU request, add the following section under variables:

    \"cpu_request\": {\n  \"title\": \"CPU Request\",\n  \"description\": \"The CPU allocation to request for this pod.\",\n  \"default\": \"default\",\n  \"type\": \"string\"\n},\n

    Next add the following to the first containers item under job_configuration:

    ...\n\"containers\": [\n  {\n    ...,\n    \"resources\": {\n      \"requests\": {\n        \"cpu\": \"{{ cpu_request }}\"\n      }\n    }\n  }\n],\n...\n

    Running deployments with this work pool will now request the specified CPU.

    After configuring the work pool settings, move to the next screen.

    Give the work pool a name and save.

    Our new Kubernetes work pool should now appear in the list of work pools.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-prefect-cloud-api-key","title":"Create a Prefect Cloud API key","text":"

    While in the Prefect Cloud UI, create a Prefect Cloud API key if you don't already have one. Click on your profile avatar picture, then click your name to go to your profile settings, click API Keys and hit the plus button to create a new API key here. Make sure to store it safely along with your other passwords, ideally via a password manager.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#deploy-a-worker-using-helm","title":"Deploy a worker using Helm","text":"

    With our cluster and work pool created, it's time to deploy a worker, which will set up Kubernetes infrastructure to run our flows. The best way to deploy a worker is using the Prefect Helm Chart.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#add-the-prefect-helm-repository","title":"Add the Prefect Helm repository","text":"

    Add the Prefect Helm repository to your Helm client:

    helm repo add prefect https://prefecthq.github.io/prefect-helm\nhelm repo update\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-namespace","title":"Create a namespace","text":"

    Create a new namespace in your Kubernetes cluster to deploy the Prefect worker:

    kubectl create namespace prefect\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-kubernetes-secret-for-the-prefect-api-key","title":"Create a Kubernetes secret for the Prefect API key","text":"
    kubectl create secret generic prefect-api-key \\\n--namespace=prefect --from-literal=key=your-prefect-cloud-api-key\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#configure-helm-chart-values","title":"Configure Helm chart values","text":"

    Create a values.yaml file to customize the Prefect worker configuration. Add the following contents to the file:

    worker:\n  cloudApiConfig:\n    accountId: <target account ID>\n    workspaceId: <target workspace ID>\n  config:\n    workPool: <target work pool name>\n

    These settings will ensure that the worker connects to the proper account, workspace, and work pool.

    View your Account ID and Workspace ID in your browser URL when logged into Prefect Cloud. For example: https://app.prefect.cloud/account/abc-my-account-id-is-here/workspaces/123-my-workspace-id-is-here.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#create-a-helm-release","title":"Create a Helm release","text":"

    Let's install the Prefect worker using the Helm chart with your custom values.yaml file:

    helm install prefect-worker prefect/prefect-worker \\\n  --namespace=prefect \\\n  -f values.yaml\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#verify-deployment","title":"Verify deployment","text":"

    Check the status of your Prefect worker deployment:

    kubectl get pods -n prefect\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#define-a-flow","title":"Define a flow","text":"

    Let's start simple with a flow that just logs a message. In a directory named flows, create a file named hello.py with the following contents:

    from prefect import flow, get_run_logger, tags\n\n@flow\ndef hello(name: str = \"Marvin\"):\n    logger = get_run_logger()\n    logger.info(f\"Hello, {name}!\")\n\nif __name__ == \"__main__\":\n    with tags(\"local\"):\n        hello()\n

    Run the flow locally with python hello.py to verify that it works. Note that we use the tags context manager to tag the flow run as local. This step is not required, but does add some helpful metadata.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#define-a-prefect-deployment","title":"Define a Prefect deployment","text":"

    Prefect has two recommended options for creating a deployment with dynamic infrastructure. You can define a deployment in a Python script using the flow.deploy mechanics or in a prefect.yaml definition file. The prefect.yaml file currently allows for more customization in terms of push and pull steps. Kubernetes objects are defined in YAML, so we expect many teams using Kubernetes work pools to create their deployments with YAML as well. To learn about the Python deployment creation method with flow.deploy refer to the Workers & Work Pools tutorial page.

    The prefect.yaml file is used by the prefect deploy command to deploy our flows. As a part of that process it will also build and push our image. Create a new file named prefect.yaml with the following contents:

    # Generic metadata about this project\nname: flows\nprefect-version: 2.13.8\n\n# build section allows you to manage and build docker images\nbuild:\n- prefect_docker.deployments.steps.build_docker_image:\n    id: build-image\n    requires: prefect-docker>=0.4.0\n    image_name: \"{{ $PREFECT_IMAGE_NAME }}\"\n    tag: latest\n    dockerfile: auto\n    platform: \"linux/amd64\"\n\n# push section allows you to manage if and how this project is uploaded to remote locations\npush:\n- prefect_docker.deployments.steps.push_docker_image:\n    requires: prefect-docker>=0.4.0\n    image_name: \"{{ build-image.image_name }}\"\n    tag: \"{{ build-image.tag }}\"\n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect.deployments.steps.set_working_directory:\n    directory: /opt/prefect/flows\n\n# the definitions section allows you to define reusable components for your deployments\ndefinitions:\n  tags: &common_tags\n    - \"eks\"\n  work_pool: &common_work_pool\n    name: \"kubernetes\"\n    job_variables:\n      image: \"{{ build-image.image }}\"\n\n# the deployments section allows you to provide configuration for deploying flows\ndeployments:\n- name: \"default\"\n  tags: *common_tags\n  schedule: null\n  entrypoint: \"flows/hello.py:hello\"\n  work_pool: *common_work_pool\n\n- name: \"arthur\"\n  tags: *common_tags\n  schedule: null\n  entrypoint: \"flows/hello.py:hello\"\n  parameters:\n    name: \"Arthur\"\n  work_pool: *common_work_pool\n

    We define two deployments of the hello flow: default and arthur. Note that by specifying dockerfile: auto, Prefect will automatically create a dockerfile that installs any requirements.txt and copies over the current directory. You can pass a custom Dockerfile instead with dockerfile: Dockerfile or dockerfile: path/to/Dockerfile. Also note that we are specifically building for the linux/amd64 platform. This specification is often necessary when images are built on Macs with M series chips but run on cloud provider instances.

    Deployment specific build, push, and pull

    The build, push, and pull steps can be overridden for each deployment. This allows for more custom behavior, such as specifying a different image for each deployment.

    Let's make sure we define our requirements in a requirements.txt file:

    prefect>=2.13.8\nprefect-docker>=0.4.0\nprefect-kubernetes>=0.3.1\n

    The directory should now look something like this:

    .\n\u251c\u2500\u2500 prefect.yaml\n\u2514\u2500\u2500 flows\n    \u251c\u2500\u2500 requirements.txt\n    \u2514\u2500\u2500 hello.py\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#tag-images-with-a-git-sha","title":"Tag images with a Git SHA","text":"

    If your code is stored in a GitHub repository, it's good practice to tag your images with the Git SHA of the code used to build it. This can be done in the prefect.yaml file with a few minor modifications, and isn't yet an option with the Python deployment creation method. Let's use the run_shell_script command to grab the SHA and pass it to the tag parameter of build_docker_image:

    build:\n- prefect.deployments.steps.run_shell_script:\n    id: get-commit-hash\n    script: git rev-parse --short HEAD\n    stream_output: false\n- prefect_docker.deployments.steps.build_docker_image:\n    id: build-image\n    requires: prefect-docker>=0.4.0\n    image_name: \"{{ $PREFECT_IMAGE_NAME }}\"\n    tag: \"{{ get-commit-hash.stdout }}\"\n    dockerfile: auto\n    platform: \"linux/amd64\"\n

    Let's also set the SHA as a tag for easy identification in the UI:

    definitions:\n  tags: &common_tags\n    - \"eks\"\n    - \"{{ get-commit-hash.stdout }}\"\n  work_pool: &common_work_pool\n    name: \"kubernetes\"\n    job_variables:\n      image: \"{{ build-image.image }}\"\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#authenticate-to-prefect","title":"Authenticate to Prefect","text":"

    Before we deploy the flows to Prefect, we will need to authenticate via the Prefect CLI. We will also need to ensure that all of our flow's dependencies are present at deploy time.

    This example uses a virtual environment to ensure consistency across environments.

    # Create a virtualenv & activate it\nvirtualenv prefect-demo\nsource prefect-demo/bin/activate\n\n# Install dependencies of your flow\nprefect-demo/bin/pip install -r requirements.txt\n\n# Authenticate to Prefect & select the appropriate \n# workspace to deploy your flows to\nprefect-demo/bin/prefect cloud login\n
    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#deploy-the-flows","title":"Deploy the flows","text":"

    Now we're ready to deploy our flows which will build our images. The image name determines which registry it will end up in. We have configured our prefect.yaml file to get the image name from the PREFECT_IMAGE_NAME environment variable, so let's set that first:

    AWSGCPAzure
    export PREFECT_IMAGE_NAME=<AWS_ACCOUNT_ID>.dkr.ecr.<REGION>.amazonaws.com/<IMAGE-NAME>\n
    export PREFECT_IMAGE_NAME=us-docker.pkg.dev/<GCP-PROJECT-NAME>/<REPOSITORY-NAME>/<IMAGE-NAME>\n
    export PREFECT_IMAGE_NAME=<REPOSITORY-NAME>.azurecr.io/<IMAGE-NAME>\n

    To deploy your flows, ensure your Docker daemon is running first. Deploy all the flows with prefect deploy --all or deploy them individually by name: prefect deploy -n hello/default or prefect deploy -n hello/arthur.

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/kubernetes/#run-the-flows","title":"Run the flows","text":"

    Once the deployments are successfully created, we can run them from the UI or the CLI:

    prefect deployment run hello/default\nprefect deployment run hello/arthur\n

    Congratulations! You just ran two deployments in Kubernetes. Head over to the UI to check their status!

    ","tags":["kubernetes","containers","orchestration","infrastructure","deployments"],"boost":2},{"location":"guides/deployment/push-work-pools/","title":"Push Work to Serverless Computing Infrastructure","text":"

    Push work pools are a special type of work pool that allows Prefect Cloud to submit flow runs for execution to serverless computing infrastructure without running a worker. Push work pools currently support execution in GCP Cloud Run Jobs, Azure Container Instances, and AWS ECS Tasks.

    In this guide you will:

    • Create a push work pool that sends work to Google Cloud Run, Amazon Elastic Container Service (AWS ECS), or Azure Container Instances (ACI)
    • Deploy a flow to that work pool
    • Execute a flow without having to run a worker or agent process to poll for flow runs

    You can automatically provision infrastructure and create your push work pool using the prefect work-pool create CLI command with the --provision-infra flag. This approach greatly simplifies the setup process.

    Let's explore automatic infrastructure provisioning for push work pools first, and then we'll cover how to manually set up your push work pool.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#automatic-infrastructure-provisioning","title":"Automatic infrastructure provisioning","text":"

    With Perfect Cloud you can provision infrastructure for use with an AWS ECS, Google Cloud Run, ACI push work pool. Push work pools in Prefect Cloud simplify the setup and management of the infrastructure necessary to run your flows. However, setting up infrastructure on your cloud provider can still be a time-consuming process. Prefect can dramatically simplify this process by automatically provisioning the necessary infrastructure for you.

    We'll use the prefect work-pool create CLI command with the --provision-infra flag to automatically provision your serverless cloud resources and set up your Prefect workspace to use a new push pool.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#prerequisites","title":"Prerequisites","text":"

    To use automatic infrastructure provisioning, you'll need to have the relevant cloud CLI library installed and to have authenticated with your cloud provider.

    AWS ECSAzure Container InstancesGoogle Cloud Run

    Install the AWS CLI, authenticate with your AWS account, and set a default region.

    If you already have the AWS CLI installed, be sure to update to the latest version.

    You will need the following permissions in your authenticated AWS account:

    IAM Permissions:

    • iam:CreatePolicy
    • iam:GetPolicy
    • iam:ListPolicies
    • iam:CreateUser
    • iam:GetUser
    • iam:AttachUserPolicy
    • iam:CreateRole
    • iam:GetRole
    • iam:AttachRolePolicy
    • iam:ListRoles
    • iam:PassRole

    Amazon ECS Permissions:

    • ecs:CreateCluster
    • ecs:DescribeClusters

    Amazon EC2 Permissions:

    • ec2:CreateVpc
    • ec2:DescribeVpcs
    • ec2:CreateInternetGateway
    • ec2:AttachInternetGateway
    • ec2:CreateRouteTable
    • ec2:CreateRoute
    • ec2:CreateSecurityGroup
    • ec2:DescribeSubnets
    • ec2:CreateSubnet
    • ec2:DescribeAvailabilityZones
    • ec2:AuthorizeSecurityGroupIngress
    • ec2:AuthorizeSecurityGroupEgress

    Amazon ECR Permissions:

    • ecr:CreateRepository
    • ecr:DescribeRepositories
    • ecr:GetAuthorizationToken

    If you want to use AWS managed policies, you can use the following:

    • AmazonECS_FullAccess
    • AmazonEC2FullAccess
    • IAMFullAccess
    • AmazonEC2ContainerRegistryFullAccess

    Note that the above policies will give you all the permissions needed, but are more permissive than necessary.

    Docker is also required to build and push images to your registry. You can install Docker here.

    Install the Azure CLI and authenticate with your Azure account.

    If you already have the Azure CLI installed, be sure to update to the latest version with az upgrade.

    You will also need the following roles in your Azure subscription:

    • Contributor
    • User Access Administrator
    • Application Administrator
    • Managed Identity Operator
    • Azure Container Registry Contributor

    Docker is also required to build and push images to your registry. You can install Docker here.

    Install the gcloud CLI and authenticate with your GCP project.

    If you already have the gcloud CLI installed, be sure to update to the latest version with gcloud components update.

    You will also need the following permissions in your GCP project:

    • resourcemanager.projects.list
    • serviceusage.services.enable
    • iam.serviceAccounts.create
    • iam.serviceAccountKeys.create
    • resourcemanager.projects.setIamPolicy
    • artifactregistry.repositories.create

    Docker is also required to build and push images to your registry. You can install Docker here.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#automatically-creating-a-new-push-work-pool-and-provisioning-infrastructure","title":"Automatically creating a new push work pool and provisioning infrastructure","text":"

    Here's the command to create a new push work pool and configure the necessary infrastructure.

    AWS ECSAzure Container InstancesGoogle Cloud Run
    prefect work-pool create --type ecs:push --provision-infra my-ecs-pool\n

    Using the --provision-infra flag will automatically set up your default AWS account to be ready to execute flows via ECS tasks. In your AWS account, this command will create a new IAM user, IAM policy, ECS cluster that uses AWS Fargate, VPC, and ECR repository if they don't already exist. In your Prefect workspace, this command will create an AWSCredentials block for storing the generated credentials.

    Here's an abbreviated example output from running the command:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-ecs-pool will require:                                          \u2502\n\u2502                                                                                                                   \u2502\n\u2502          - Creating an IAM user for managing ECS tasks: prefect-ecs-user                                          \u2502\n\u2502          - Creating and attaching an IAM policy for managing ECS tasks: prefect-ecs-policy                        \u2502\n\u2502          - Storing generated AWS credentials in a block                                                           \u2502\n\u2502          - Creating an ECS cluster for running Prefect flows: prefect-ecs-cluster                                 \u2502\n\u2502          - Creating a VPC with CIDR 172.31.0.0/16 for running ECS tasks: prefect-ecs-vpc                          \u2502\n\u2502          - Creating an ECR repository for storing Prefect images: prefect-flows                                   \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]: y\nProvisioning IAM user\nCreating IAM policy\nGenerating AWS credentials\nCreating AWS credentials block\nProvisioning ECS cluster\nProvisioning VPC\nCreating internet gateway\nSetting up subnets\nSetting up security group\nProvisioning ECR repository\nAuthenticating with ECR\nSetting default Docker build namespace\nProvisioning Infrastructure \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned!\nCreated work pool 'my-ecs-pool'!\n

    Default Docker build namespace

    After infrastructure provisioning completes, you will be logged into your new ECR repository and the default Docker build namespace will be set to the URL of the registry.

    While the default namespace is set, you will not need to provide the registry URL when building images as part of your deployment process.

    To take advantage of this, you can write your deploy scripts like this:

    example_deploy_script.py
    from prefect import flow\nfrom prefect.deployments import DeploymentImage\n\n@flow(log_prints=True)            \ndef my_flow(name: str = \"world\"):                          \n    print(f\"Hello {name}! I'm a flow running in a ECS task!\") \n\n\nif __name__ == \"__main__\":\n    my_flow.deploy(\n        name=\"my-deployment\", \n        work_pool_name=\"my-work-pool\",\n        image=DeploymentImage(                                                 \n            name=\"my-repository:latest\",\n            platform=\"linux/amd64\",\n        )                                                                      \n    )       \n

    This will build an image with the tag <ecr-registry-url>/my-image:latest and push it to the registry.

    Your image name will need to match the name of the repository created with your work pool. You can create new repositories in the ECR console.

    prefect work-pool create --type azure-container-instance:push --provision-infra my-aci-pool\n

    Using the --provision-infra flag will automatically set up your default Azure account to be ready to execute flows via Azure Container Instances. In your Azure account, this command will create a resource group, app registration, service account with necessary permission, generate a secret for the app registration, and create an Azure Container Registry, if they don't already exist. In your Prefect workspace, this command will create an AzureContainerInstanceCredentials block for storing the client secret value from the generated secret.

    Here's an abbreviated example output from running the command:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-aci-work-pool will require:                     \u2502\n\u2502                                                                                           \u2502\n\u2502     Updates in subscription Azure subscription 1                                          \u2502\n\u2502                                                                                           \u2502\n\u2502         - Create a resource group in location eastus                                      \u2502\n\u2502         - Create an app registration in Azure AD prefect-aci-push-pool-app                \u2502\n\u2502         - Create/use a service principal for app registration                             \u2502\n\u2502         - Generate a secret for app registration                                          \u2502\n\u2502         - Create an Azure Container Registry with prefix prefect                          \u2502\n\u2502         - Create an identity prefect-acr-identity to allow access to the created registry \u2502\n\u2502         - Assign Contributor role to service account                                      \u2502\n\u2502         - Create an ACR registry for image hosting                                        \u2502\n\u2502         - Create an identity for Azure Container Instance to allow access to the registry \u2502\n\u2502                                                                                           \u2502\n\u2502     Updates in Prefect workspace                                                          \u2502\n\u2502                                                                                           \u2502\n\u2502         - Create Azure Container Instance credentials block aci-push-pool-credentials     \u2502\n\u2502                                                                                           \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]:     \nCreating resource group\nCreating app registration\nGenerating secret for app registration\nCreating ACI credentials block\nACI credentials block 'aci-push-pool-credentials' created in Prefect Cloud\nAssigning Contributor role to service account\nCreating Azure Container Registry\nCreating identity\nProvisioning infrastructure... \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned for 'my-aci-work-pool' work pool!\nCreated work pool 'my-aci-work-pool'!\n

    Default Docker build namespace

    After infrastructure provisioning completes, you will be logged into your new Azure Container Registry and the default Docker build namespace will be set to the URL of the registry.

    While the default namespace is set, any images you build without specifying a registry or username/organization will be pushed to the registry.

    To take advantage of this functionality, you can write your deploy scripts like this:

    example_deploy_script.py
    from prefect import flow                                                       \nfrom prefect.deployments import DeploymentImage                                \n\n\n@flow(log_prints=True)                                                         \ndef my_flow(name: str = \"world\"):                                              \n    print(f\"Hello {name}! I'm a flow running on an Azure Container Instance!\") \n\n\nif __name__ == \"__main__\":                                                     \n    my_flow.deploy(                                                            \n        name=\"my-deployment\",\n        work_pool_name=\"my-work-pool\",                                                \n        image=DeploymentImage(                                                 \n            name=\"my-image:latest\",                                            \n            platform=\"linux/amd64\",                                            \n        )                                                                      \n    )       \n

    This will build an image with the tag <acr-registry-url>/my-image:latest and push it to the registry.

    prefect work-pool create --type cloud-run:push --provision-infra my-cloud-run-pool \n

    Using the --provision-infra flag will allow you to select a GCP project to use for your work pool and automatically configure it to be ready to execute flows via Cloud Run. In your GCP project, this command will activate the Cloud Run API, create a service account, and create a key for the service account, if they don't already exist. In your Prefect workspace, this command will create a GCPCredentials block for storing the service account key.

    Here's an abbreviated example output from running the command:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-cloud-run-pool will require:                           \u2502\n\u2502                                                                                                          \u2502\n\u2502     Updates in GCP project central-kit-405415 in region us-central1                                      \u2502\n\u2502                                                                                                          \u2502\n\u2502         - Activate the Cloud Run API for your project                                                    \u2502\n\u2502         - Activate the Artifact Registry API for your project                                            \u2502\n\u2502         - Create an Artifact Registry repository named prefect-images                                    \u2502\n\u2502         - Create a service account for managing Cloud Run jobs: prefect-cloud-run                        \u2502\n\u2502             - Service account will be granted the following roles:                                       \u2502\n\u2502                 - Service Account User                                                                   \u2502\n\u2502                 - Cloud Run Developer                                                                    \u2502\n\u2502         - Create a key for service account prefect-cloud-run                                             \u2502\n\u2502                                                                                                          \u2502\n\u2502     Updates in Prefect workspace                                                                         \u2502\n\u2502                                                                                                          \u2502\n\u2502         - Create GCP credentials block my--pool-push-pool-credentials to store the service account key   \u2502\n\u2502                                                                                                          \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]: y\nActivating Cloud Run API\nActivating Artifact Registry API\nCreating Artifact Registry repository\nConfiguring authentication to Artifact Registry\nSetting default Docker build namespace\nCreating service account\nAssigning roles to service account\nCreating service account key\nCreating GCP credentials block\nProvisioning Infrastructure \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned!\nCreated work pool 'my-cloud-run-pool'!\n

    Default Docker build namespace

    After infrastructure provisioning completes, you will be logged into your new Artifact Registry repository and the default Docker build namespace will be set to the URL of the repository.

    While the default namespace is set, any images you build without specifying a registry or username/organization will be pushed to the repository.

    To take advantage of this functionality, you can write your deploy scripts like this:

    example_deploy_script.py
    from prefect import flow                                                       \nfrom prefect.deployments import DeploymentImage                                \n\n\n@flow(log_prints=True)\ndef my_flow(name: str = \"world\"):\n    print(f\"Hello {name}! I'm a flow running on Cloud Run!\")\n\n\nif __name__ == \"__main__\":                                                     \n    my_flow.deploy(                                                            \n        name=\"my-deployment\",\n        work_pool_name=\"above-ground\",\n        image=DeploymentImage(\n            name=\"my-image:latest\",\n            platform=\"linux/amd64\",\n        )\n    )\n

    This will build an image with the tag <region>-docker.pkg.dev/<project>/<repository-name>/my-image:latest and push it to the repository.

    That's it! You're ready to create and schedule deployments that use your new push work pool. Reminder that no worker is needed to run flows with a push work pool.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#using-existing-resources-with-automatic-infrastructure-provisioning","title":"Using existing resources with automatic infrastructure provisioning","text":"

    If you already have the necessary infrastructure set up, Prefect will detect that upon work pool creation and the infrastructure provisioning for that resource will be skipped.

    For example, here's how prefect work-pool create my-work-pool --provision-infra looks when existing Azure resources are detected:

    Proceed with infrastructure provisioning? [y/n]: y\nCreating resource group\nResource group 'prefect-aci-push-pool-rg' already exists in location 'eastus'.\nCreating app registration\nApp registration 'prefect-aci-push-pool-app' already exists.\nGenerating secret for app registration\nProvisioning infrastructure\nACI credentials block 'bb-push-pool-credentials' created\nAssigning Contributor role to service account...\nService principal with object ID '4be6fed7-...' already has the 'Contributor' role assigned in \n'/subscriptions/.../'\nCreating Azure Container Instance\nContainer instance 'prefect-aci-push-pool-container' already exists.\nCreating Azure Container Instance credentials block\nProvisioning infrastructure... \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned!\nCreated work pool 'my-work-pool'!\n
    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#provisioning-infrastructure-for-an-existing-push-work-pool","title":"Provisioning infrastructure for an existing push work pool","text":"

    If you already have a push work pool set up, but haven't configured the necessary infrastructure, you can use the provision-infra sub-command to provision the infrastructure for that work pool. For example, you can run the following command if you have a work pool named \"my-work-pool\".

    prefect work-pool provision-infra my-work-pool\n

    Prefect will create the necessary infrastructure for the my-work-pool work pool and provide you with a summary of the changes to be made:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-work-pool will require:                                      \u2502\n\u2502                                                                                                                \u2502\n\u2502     Updates in subscription Azure subscription 1                                                               \u2502\n\u2502                                                                                                                \u2502\n\u2502         - Create a resource group in location eastus                                                           \u2502\n\u2502         - Create an app registration in Azure AD prefect-aci-push-pool-app                                     \u2502\n\u2502         - Create/use a service principal for app registration                                                  \u2502\n\u2502         - Generate a secret for app registration                                                               \u2502\n\u2502         - Assign Contributor role to service account                                                           \u2502\n\u2502         - Create Azure Container Instance 'aci-push-pool-container' in resource group prefect-aci-push-pool-rg \u2502\n\u2502                                                                                                                \u2502\n\u2502     Updates in Prefect workspace                                                                               \u2502\n\u2502                                                                                                                \u2502\n\u2502         - Create Azure Container Instance credentials block aci-push-pool-credentials                          \u2502\n\u2502                                                                                                                \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]: y\n

    This command can speed up your infrastructure setup process.

    As with the examples above, you will need to have the related cloud CLI library installed and be authenticated with your cloud provider.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#manual-infrastructure-provisioning","title":"Manual infrastructure provisioning","text":"

    If you prefer to set up your infrastructure manually, don't include the --provision-infra flag in the CLI command. In the examples below, we'll create a push work pool via the Prefect Cloud UI.

    AWS ECSAzure Container InstancesGoogle Cloud Run

    To push work to ECS, AWS credentials are required.

    Create a user and attach the AmazonECS_FullAccess permissions.

    From that user's page create credentials and store them somewhere safe for use in the next section.

    To push work to Azure, an Azure subscription, resource group and tenant secret are required.

    Create Subscription and Resource Group

    1. In the Azure portal, create a subscription.
    2. Create a resource group within your subscription.

    Create App Registration

    1. In the Azure portal, create an app registration.
    2. In the app registration, create a client secret. Copy the value and store it somewhere safe.

    Add App Registration to Resource Group

    1. Navigate to the resource group you created earlier.
    2. Choose the \"Access control (IAM)\" blade in the left-hand side menu. Click \"+ Add\" button at the top, then \"Add role assignment\".
    3. Go to the \"Privileged administrator roles\" tab, click on \"Contributor\", then click \"Next\" at the bottom of the page.
    4. Click on \"+ Select members\". Type the name of the app registration (otherwise it may not autopopulate) and click to add it. Then hit \"Select\" and click \"Next\". The default permissions associated with a role like \"Contributor\" might not always be sufficient for all operations related to Azure Container Instances (ACI). The specific permissions required can depend on the operations you need to perform (like creating, running, and deleting ACI container groups) and your organization's security policies. In some cases, additional permissions or custom roles might be necessary.
    5. Click \"Review + assign\" to finish.

    A GCP service account and an API Key are required, to push work to Cloud Run.

    Create a service account by navigating to the service accounts page and clicking Create. Name and describe your service account, and click continue to configure permissions.

    The service account must have two roles at a minimum, Cloud Run Developer, and Service Account User.

    Once the Service account is created, navigate to its Keys page to add an API key. Create a JSON type key, download it, and store it somewhere safe for use in the next section.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#work-pool-configuration","title":"Work pool configuration","text":"

    Our push work pool will store information about what type of infrastructure our flow will run on, what default values to provide to compute jobs, and other important execution environment parameters. Because our push work pool needs to integrate securely with your serverless infrastructure, we need to start by storing our credentials in Prefect Cloud, which we'll do by making a block.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#creating-a-credentials-block","title":"Creating a Credentials block","text":"AWS ECSAzure Container InstancesGoogle Cloud Run

    Navigate to the blocks page, click create new block, and select AWS Credentials for the type.

    For use in a push work pool, this block must have the region and cluster name filled out, in addition to access key and access key secret.

    Provide any other optional information and create your block.

    Navigate to the blocks page and click the \"+\" at the top to create a new block. Find the Azure Container Instance Credentials block and click \"Add +\".

    Locate the client ID and tenant ID on your app registration and use the client secret you saved earlier. Be sure to use the value of the secret, not the secret ID!

    Provide any other optional information and click \"Create\".

    Navigate to the blocks page, click create new block, and select GCP Credentials for the type.

    For use in a push work pool, this block must have the contents of the JSON key stored in the Service Account Info field, as such:

    Provide any other optional information and create your block.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#creating-a-push-work-pool","title":"Creating a push work pool","text":"

    Now navigate to the work pools page. Click Create to start configuring your push work pool by selecting a push option in the infrastructure type step.

    AWS ECSAzure Container InstancesGoogle Cloud Run

    Each step has several optional fields that are detailed in the work pools documentation. Select the block you created under the AWS Credentials field. This will allow Prefect Cloud to securely interact with your ECS cluster.

    Fill in the subscription ID and resource group name from the resource group you created. Add the Azure Container Instance Credentials block you created in the step above.

    Each step has several optional fields that are detailed in the work pools documentation. Select the block you created under the GCP Credentials field. This will allow Prefect Cloud to securely interact with your GCP project.

    Create your pool and you are ready to deploy flows to your Push work pool.

    Push work pool concurrency

    Push work pools do not have a concurrency setting. If you would like to control concurrency at the flow level, you can use global concurrency limits.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#deployment","title":"Deployment","text":"

    Deployment details are described in the deployments concept section. Your deployment needs to be configured to send flow runs to our push work pool. For example, if you create a deployment through the interactive command line experience, choose the work pool you just created. If you are deploying an existing prefect.yaml file, the deployment would contain:

      work_pool:\n    name: my-push-pool\n

    Deploying your flow to the my-push-pool work pool will ensure that runs that are ready for execution will be submitted immediately, without the need for a worker to poll for them.

    Serverless infrastructure may require a certain image architecture

    Note that serverless infrastructure may assume a certain Docker image architecture; for example, Google Cloud Run will fail to run images built with linux/arm64 architecture. If using Prefect to build your image, you can change the image architecture through the platform keyword (e.g., platform=\"linux/amd64\").

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/push-work-pools/#putting-it-all-together","title":"Putting it all together","text":"

    With your deployment created, navigate to its detail page and create a new flow run. You'll see the flow start running without ever having to poll the work pool, because Prefect Cloud securely connected to your serverless infrastructure, created a job, ran the job, and began reporting on its execution.

    ","tags":["work pools","deployments","Cloud Run","AWS ECS","Azure Container Instances","ACI","elastic container service","GCP","Google Cloud Run","serverless","Amazon Web Services","push work pools"],"boost":2},{"location":"guides/deployment/serverless-workers/","title":"Run Deployments on Serverless Infrastructure with Prefect Workers","text":"

    Prefect provides work pools for workers to run flows on the serverless platforms of major cloud providers. The following options are available:

    • AWS Elastic Container Service (ECS)
    • Azure Container Instances (ACI)
    • Google Cloud Run
    • Google Cloud Run V2
    • Google Vertex AI

    • Create a work pool that sends work to your chosen serverless infrastructure
    • Deploy a flow to that work pool
    • Start a worker in your serverless cloud provider that will poll its matched work pool for scheduled runs
    • Schedule a deployment run that a worker will pick up from the work pool and run on your serverless infrastructure

    Push work pools don't require a worker

    Options for push work pool versions of AWS ECS, Azure Container Instances, and Google Cloud Run that do not require a worker are available with Prefect Cloud. These push work pool options require connection configuration information to be stored on Prefect Cloud. Read more in the Serverless Push Work Pool Guide.

    This is a brief overview of the options to run workflows on serverless infrastructure. For in-depth guides, see the Prefect integration libraries:

    • AWS ECS guide in the prefect-aws docs
    • Azure Container Instances guide (forthcoming)
    • Google Cloud Run guide in the prefect-gcp docs.
    • For Google Vertex AI, follow the Cloud Run guide, substituting Google Vertex AI where Google Cloud Run is mentioned.

    Choosing between Google Cloud Run and Google Vertex AI

    Google Vertex AI is well-suited for machine learning model training applications in which GPUs or TPUs and high resource levels are desired.

    ","tags":["work pools","deployments","Cloud Run","GCP","Vertex AI","AWS ECS","Azure Container Instances","ACI"],"boost":2},{"location":"guides/deployment/serverless-workers/#steps","title":"Steps","text":"
    1. Make sure you have an user or service account on your chosen cloud provider with the necessary permissions to run serverless jobs
    2. Create the appropriate serverless work pool that uses a worker in the Prefect UI
    3. Create a deployment that references the work pool
    4. Start a worker in your chose serverless cloud provider infrastructure
    5. Run the deployment
    ","tags":["work pools","deployments","Cloud Run","GCP","Vertex AI","AWS ECS","Azure Container Instances","ACI"],"boost":2},{"location":"guides/deployment/serverless-workers/#next-steps","title":"Next steps","text":"

    Options for push versions on AWS ECS, Azure Container Instances, and Google Cloud Run work pools that do not require a worker are available with Prefect Cloud. Read more in the Serverless Push Work Pool Guide.

    Learn more about workers and work pools in the Prefect concept documentation.

    ","tags":["work pools","deployments","Cloud Run","GCP","Vertex AI","AWS ECS","Azure Container Instances","ACI"],"boost":2},{"location":"guides/deployment/storage-guide/","title":"Where to Store Your Flow Code","text":"

    When a flow runs, the execution environment needs access to its code. Flow code is not stored in a Prefect server database instance or Prefect Cloud. When deploying a flow, you have several flow code storage options.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#option-1-local-storage","title":"Option 1: Local storage","text":"

    Local flow code storage is often used with a Local Subprocess work pool for initial experimentation.

    To create a deployment with local storage and a Local Subprocess work pool, do the following:

    1. Run prefect deploy from the root of the directory containing your flow code.
    2. Select that you want to create a new deployment, select the flow code entrypoint, and name your deployment.
    3. Select a process work pool.

    You are then shown the location that your flow code will be fetched from when a flow is run. For example:

    Your Prefect workers will attempt to load your flow from: \n/my-path/my-flow-file.py. To see more options for managing your flow's code, run:\n\n    $ prefect init\n

    When deploying a flow to production, you most likely want code to run with infrastructure-specific configuration. The flow code storage options shown below are recommended for production deployments.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#option-2-git-based-storage","title":"Option 2: Git-based storage","text":"

    Git-based version control platforms are popular locations for code storage. They provide redundancy, version control, and easier collaboration.

    GitHub is the most popular cloud-based repository hosting provider. GitLab and Bitbucket are other popular options. Prefect supports each of these platforms.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#creating-a-deployment-with-git-based-storage","title":"Creating a deployment with git-based storage","text":"

    Run prefect deploy from within a git repository and create a new deployment. You will see a series of prompts. Select that you want to create a new deployment, select the flow code entrypoint, and name your deployment.

    Prefect detects that you are in a git repository and asks if you want to store your flow code in a git repository. Select \"y\" and you will be prompted to confirm the URL of your git repository and the branch name, as in the example below:

    ? Your Prefect workers will need access to this flow's code in order to run it. \nWould you like your workers to pull your flow code from its remote repository when running this flow? [y/n] (y): \n? Is https://github.com/my_username/my_repo.git the correct URL to pull your flow code from? [y/n] (y): \n? Is main the correct branch to pull your flow code from? [y/n] (y): \n? Is this a private repository? [y/n]: y\n

    In this example, the git repository is hosted on GitHub. If you are using Bitbucket or GitLab, the URL will match your provider. If the repository is public, enter \"n\" and you are on your way.

    If the repository is private, you can enter a token to access your private repository. This token will be saved in an encrypted Prefect Secret block.

    ? Please enter a token that can be used to access your private repository. This token will be saved as a Secret block via the Prefect API: \"123_abc_this_is_my_token\"\n

    Verify that you have a new Secret block in your active workspace named in the format \"deployment-my-deployment-my-flow-name-repo-token\".

    Creating access tokens differs for each provider.

    GitHubBitbucketGitLab

    We recommend using HTTPS with fine-grained Personal Access Tokens so that you can limit access by repository. See the GitHub docs for Personal Access Tokens (PATs).

    Under Your Profile->Developer Settings->Personal access tokens->Fine-grained token choose Generate New Token and fill in the required fields. Under Repository access choose Only select repositories and grant the token permissions for Contents.

    We recommend using HTTPS with Repository, Project, or Workspace Access Tokens.

    You can create a Repository Access Token with Scopes->Repositories->Read.

    Bitbucket requires you prepend the token string with x-token-auth: So the full string looks like x-token-auth:abc_123_this_is_my_token.

    We recommend using HTTPS with Project Access Tokens.

    In your repository in the GitLab UI, select Settings->Repository->Project Access Tokens and check read_repository under Select scopes.

    If you want to configure a Secret block ahead of time for use when deploying a prefect.yaml file, create the block via code or the Prefect UI and reference it like this:

    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://bitbucket.org/org/my-private-repo.git\n        access_token: \"{{ prefect.blocks.secret.my-block-name }}\"\n

    Alternatively, you can create a Credentials block ahead of time and reference it in the prefect.yaml pull step.

    GitHubBitbucketGitLab
    1. Install the Prefect-Github library with pip install -U prefect-github
    2. Register the blocks in that library to make them available on the server with prefect block register -m prefect_github.
    3. Create a GitHub Credentials block via code or the Prefect UI and reference it as shown above.
    4. In addition to the block name, most users will need to fill in the GitHub Username and GitHub Personal Access Token fields.
    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://github.com/discdiver/my-private-repo.git\n        credentials: \"{{ prefect.blocks.github-credentials.my-block-name }}\"\n
    1. Install the relevant library with pip install -U prefect-bitbucket
    2. Register the blocks in that library with prefect block register -m prefect_bitbucket
    3. Create a Bitbucket credentials block via code or the Prefect UI and reference it as shown above.
    4. In addition to the block name, most users will need to fill in the Bitbucket Username and Bitbucket Personal Access Token fields.
    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://bitbucket.org/org/my-private-repo.git\n        credentials: \"{{ prefect.blocks.bitbucket-credentials.my-block-name }}\"\n
    1. Install the relevant library with pip install -U prefect-gitlab
    2. Register the blocks in that library with prefect block register -m prefect_gitlab
    3. Create a GitLab credentials block via code or the Prefect UI and reference it as shown above.
    4. In addition to the block name, most users will need to fill in the GitLab Username and GitLab Personal Access Token fields.
    pull:\n    - prefect.deployments.steps.git_clone:\n        repository: https://gitlab.com/org/my-private-repo.git\n        credentials: \"{{ prefect.blocks.gitlab-credentials.my-block-name }}\"\n

    Push your code

    When you make a change to your code, Prefect does not push your code to your git-based version control platform. You need to push your code manually or as part of your CI/CD pipeline. This design decision is an intentional one to avoid confusion about the git history and push process.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#option-3-docker-based-storage","title":"Option 3: Docker-based storage","text":"

    Another popular way to store your flow code is to include it in a Docker image. The following work pools use Docker containers, so the flow code can be directly baked into the image:

    • Docker
    • Kubernetes
    • Serverless cloud-based options
      • AWS Elastic Container Service
      • Azure Container Instances
      • Google Cloud Run
    • Push-based serverless cloud-based options (no worker required)

      • AWS Elastic Container Service - Push
      • Azure Container Instances - Push
      • Google Cloud Run - Push
    • Run prefect init in the root of your repository and choose docker for the project name and answer the prompts to create a prefect.yaml file with a build step that will create a Docker image with the flow code built in. See the Workers and Work Pools page of the tutorial for more info.

    • Run prefect deploy to create a deployment.
    • Upon deployment run the worker will pull the Docker image and spin up a container.
    • The flow code baked into the image will run inside the container.

    CI/CD may not require push or pull steps

    You don't need push or pull steps in the prefect.yaml file if using CI/CD to build a Docker image outside of Prefect. Instead, the work pool can reference the image directly.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#option-4-cloud-provider-storage","title":"Option 4: Cloud-provider storage","text":"

    You can store your code in an AWS S3 bucket, Azure Blob Storage container, or GCP GCS bucket and specify the destination directly in the push and pull steps of your prefect.yaml file.

    To create a templated prefect.yaml file run prefect init and select the recipe for the applicable cloud-provider storage. Below are the recipe options and the relevant portions of the prefect.yaml file.

    AWS S3 bucketAzure Blob Storage containerGCP GCS bucket

    Choose s3 as the recipe and enter the bucket name when prompted.

    # push section allows you to manage if and how this project is uploaded to remote locations\npush:\n- prefect_aws.deployments.steps.push_to_s3:\n    id: push_code\n    requires: prefect-aws>=0.3.4\n    bucket: my-bucket\n    folder: my-folder\n    credentials: \"{{ prefect.blocks.aws-credentials.my-credentials-block }}\" # if private\n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect_aws.deployments.steps.pull_from_s3:\n    id: pull_code\n    requires: prefect-aws>=0.3.4\n    bucket: '{{ push_code.bucket }}'\n    folder: '{{ push_code.folder }}'\n    credentials: \"{{ prefect.blocks.aws-credentials.my-credentials-block }}\" # if private \n

    If the bucket requires authentication to access it, you can do the following:

    1. Install the Prefect-AWS library with pip install -U prefect-aws
    2. Register the blocks in Prefect-AWS with prefect block register -m prefect_aws
    3. Create a user with a role with read and write permissions to access the bucket. If using the UI, create an access key pair with IAM->Users->Security credentials->Access keys->Create access key. Choose Use case->Other and then copy the Access key and Secret access key values.
    4. Create an AWS Credentials block via code or the Prefect UI. In addition to the block name, most users will fill in the AWS Access Key ID and AWS Access Key Secret fields.
    5. Reference the block as shown in the push and pull steps above.

    Choose azure as the recipe and enter the container name when prompted.

    # push section allows you to manage if and how this project is uploaded to remote locations\npush:\n- prefect_azure.deployments.steps.push_to_azure_blob_storage:\n    id: push_code\n    requires: prefect-azure>=0.2.8\n    container: my-prefect-azure-container\n    folder: my-folder\n    credentials: \"{{ prefect.blocks.azure-blob-storage-credentials.my-credentials-block }}\" # if private\n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect_azure.deployments.steps.pull_from_azure_blob_storage:\n    id: pull_code\n    requires: prefect-azure>=0.2.8\n    container: '{{ push_code.container }}'\n    folder: '{{ push_code.folder }}'\n    credentials: \"{{ prefect.blocks.azure-blob-storage-credentials.my-credentials-block }}\" # if private\n

    If the blob requires authentication to access it, you can do the following:

    1. Install the Prefect-Azure library with pip install -U prefect-azure
    2. Register the blocks in Prefect-Azure with prefect block register -m prefect_azure
    3. Create an access key for a role with sufficient (read and write) permissions to access the blob. A connection string that will contain all needed information can be created in the UI under Storage Account->Access keys.
    4. Create an Azure Blob Storage Credentials block via code or the Prefect UI. Enter a name for the block and paste the connection string into the Connection String field.
    5. Reference the block as shown in the push and pull steps above.

    Choose `gcs`` as the recipe and enter the bucket name when prompted.

    # push section allows you to manage if and how this project is uploaded to remote locations\npush:\n- prefect_gcp.deployment.steps.push_to_gcs:\n    id: push_code\n    requires: prefect-gcp>=0.4.3\n    bucket: my-bucket\n    folder: my-folder\n    credentials: \"{{ prefect.blocks.gcp-credentials.my-credentials-block }}\" # if private \n\n# pull section allows you to provide instructions for cloning this project in remote locations\npull:\n- prefect_gcp.deployment.steps.pull_from_gcs:\n    id: pull_code\n    requires: prefect-gcp>=0.4.3\n    bucket: '{{ push_code.bucket }}'\n    folder: '{{ pull_code.folder }}'\n    credentials: \"{{ prefect.blocks.gcp-credentials.my-credentials-block }}\" # if private \n

    If the bucket requires authentication to access it, you can do the following:

    1. Install the Prefect-GCP library with pip install -U prefect-gcp
    2. Register the blocks in Prefect-GCP with prefect block register -m prefect_gcp
    3. Create a service account in GCP for a role with read and write permissions to access the bucket contents. If using the GCP console, go to IAM & Admin->Service accounts->Create service account. After choosing a role with the required permissions, see your service account and click on the three dot menu in the Actions column. Select Manage Keys->ADD KEY->Create new key->JSON. Download the JSON file.
    4. Create a GCP Credentials block via code or the Prefect UI. Enter a name for the block and paste the entire contents of the JSON key file into the Service Account Info field.
    5. Reference the block as shown in the push and pull steps above.

    Another option for authentication is for the worker to have access to the the storage location at runtime via SSH keys.

    Alternatively, you can inject environment variables into your deployment like this example that uses an environment variable named CUSTOM_FOLDER:

     push:\n    - prefect_gcp.deployment.steps.push_to_gcs:\n        id: push_code\n        requires: prefect-gcp>=0.4.3\n        bucket: my-bucket\n        folder: '{{ $CUSTOM_FOLDER }}'\n
    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#including-and-excluding-files-from-storage","title":"Including and excluding files from storage","text":"

    By default, Prefect uploads all files in the current folder to the configured storage location when you create a deployment.

    When using a git repository, Docker image, or cloud-provider storage location, you may want to exclude certain files or directories. - If you are familiar with git you are likely familiar with the .gitignore file. - If you are familiar with Docker you are likely familiar with the .dockerignore file. - For cloud-provider storage the .prefectignore file serves the same purpose and follows a similar syntax as those files. So an entry of *.pyc will exclude all .pyc files from upload.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#other-code-storage-creation-methods","title":"Other code storage creation methods","text":"

    In earlier versions of Prefect storage blocks were the recommended way to store flow code. Storage blocks are still supported, but not recommended.

    As shown above, repositories can be referenced directly through interactive prompts with prefect deploy or in a prefect.yaml. When authentication is needed, Secret or Credential blocks can be referenced, and in some cases created automatically through interactive deployment creation prompts.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"guides/deployment/storage-guide/#next-steps","title":"Next steps","text":"

    You've seen options for where to store your flow code.

    We recommend using Docker-based storage or git-based storage for your production deployments.

    Check out more guides to reach your goals with Prefect.

    ","tags":["guides","guide","flow code","storage","code storage","repository","github","git","gitlab","bitbucket","s3","azure","blob storage","bucket","AWS","GCP","GCS","Google Cloud Storage","Azure Blob Storage","Docker","storage"],"boost":2},{"location":"integrations/","title":"Integrations","text":"

    Prefect integrations are organized into collections of pre-built tasks, flows, blocks and more that are installable as PyPI packages.

    Airbyte

    Maintained by Prefect

    Alert

    Maintained by Khuyen Tran

    AWS

    Maintained by Prefect

    Azure

    Maintained by Prefect

    Bitbucket

    Maintained by Prefect

    Census

    Maintained by Prefect

    CubeJS

    Maintained by Alessandro Lollo

    Dask

    Maintained by Prefect

    Databricks

    Maintained by Prefect

    dbt

    Maintained by Prefect

    Docker

    Maintained by Prefect

    Earthdata

    Maintained by Giorgio Basile

    Email

    Maintained by Prefect

    Firebolt

    Maintained by Prefect

    Fivetran

    Maintained by Fivetran

    Fugue

    Maintained by The Fugue Development Team

    GCP

    Maintained by Prefect

    GitHub

    Maintained by Prefect

    GitLab

    Maintained by Prefect

    Google Sheets

    Maintained by Stefano Cascavilla

    Great Expectations

    Maintained by Prefect

    HashiCorp Vault

    Maintained by Pavel Chekin

    Hex

    Maintained by Prefect

    Hightouch

    Maintained by Prefect

    Jupyter

    Maintained by Prefect

    Kubernetes

    Maintained by Prefect

    KV

    Maintained by Michael Adkins

    MetricFlow

    Maintained by Alessandro Lollo

    Monday

    Maintained by Prefect

    MonteCarlo

    Maintained by Prefect

    OpenAI

    Maintained by Prefect

    OpenMetadata

    Maintained by Prefect

    Planetary Computer

    Maintained by Giorgio Basile

    Ray

    Maintained by Prefect

    Shell

    Maintained by Prefect

    Sifflet

    Maintained by Sifflet and Alessandro Lollo

    Slack

    Maintained by Prefect

    Snowflake

    Maintained by Prefect

    Soda Core

    Maintained by Soda and Alessandro Lollo

    Spark on Kubernetes

    Maintained by Manoj Babu Katragadda

    SQLAlchemy

    Maintained by Prefect

    Stitch

    Maintained by Alessandro Lollo

    Transform

    Maintained by Alessandro Lollo

    Twitter

    Maintained by Prefect

    ","tags":["tasks","flows","blocks","collections","task library","integrations","Airbyte","Alert","AWS","Azure","Bitbucket","Census","CubeJS","Dask","Databricks","dbt","Docker","Earthdata","Email","Firebolt","Fivetran","Fugue","GCP","GitHub","GitLab","Google Sheets","Great Expectations","HashiCorp Vault","Hex","Hightouch","Jupyter","Kubernetes","KV","MetricFlow","Monday","MonteCarlo","OpenAI","OpenMetadata","Planetary Computer","Ray","Shell","Sifflet","Slack","Snowflake","Soda Core","Spark on Kubernetes","SQLAlchemy","Stitch","Transform","Twitter"],"boost":2},{"location":"integrations/contribute/","title":"Contribute","text":"

    We welcome contributors! You can help contribute blocks and integrations by following these steps.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#contributing-blocks","title":"Contributing Blocks","text":"

    Building your own custom block is simple!

    1. Subclass from Block.
    2. Add a description alongside an Attributes and Example section in the docstring.
    3. Set a _logo_url to point to a relevant image.
    4. Create the pydantic.Fields of the block with a type annotation, default or default_factory, and a short description about the field.
    5. Define the methods of the block.

    For example, this is how the Secret block is implemented:

    from pydantic import Field, SecretStr\nfrom prefect.blocks.core import Block\n\nclass Secret(Block):\n    \"\"\"\n    A block that represents a secret value. The value stored in this block will be obfuscated when\n    this block is logged or shown in the UI.\n\n    Attributes:\n        value: A string value that should be kept secret.\n\n    Example:\n        ```python\n        from prefect.blocks.system import Secret\n        secret_block = Secret.load(\"BLOCK_NAME\")\n\n        # Access the stored secret\n        secret_block.get()\n        ```\n    \"\"\"\n\n    _logo_url = \"https://example.com/logo.png\"\n\n    value: SecretStr = Field(\n        default=..., description=\"A string value that should be kept secret.\"\n    )  # ... indicates it's a required field\n\n    def get(self):\n        return self.value.get_secret_value()\n

    To view in Prefect Cloud or the Prefect server UI, register the block.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#contributing-integrations","title":"Contributing Integrations","text":"

    Anyone can create and share a Prefect Integration and we encourage anyone interested in creating an integration to do so!

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#generate-a-project","title":"Generate a project","text":"

    To help you get started with your integration, we've created a template that gives the tools you need to create and publish your integration.

    Use the Prefect Integration template to get started creating an integration with a bootstrapped project!

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#list-a-project-in-the-integrations-catalog","title":"List a project in the Integrations Catalog","text":"

    To list your integration in the Prefect Integrations Catalog, submit a PR to the Prefect repository adding a file to the docs/integrations/catalog directory with details about your integration. Please use TEMPLATE.yaml in that folder as a guide.

    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/contribute/#contribute-fixes-or-enhancements-to-integrations","title":"Contribute fixes or enhancements to Integrations","text":"

    If you'd like to help contribute to fix an issue or add a feature to any of our Integrations, please propose changes through a pull request from a fork of the repository.

    1. Fork the repository
    2. Clone the forked repository
    3. Install the repository and its dependencies:
      pip install -e \".[dev]\"\n
    4. Make desired changes
    5. Add tests
    6. Insert an entry to the Integration's CHANGELOG.md
    7. Install pre-commit to perform quality checks prior to commit:
      pre-commit install\n
    8. git commit, git push, and create a pull request
    ","tags":["blocks","storage","secrets","configuration","infrastructure","integrations","integrations","contributing"],"boost":2},{"location":"integrations/usage/","title":"Using Integrations","text":"","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#installing-an-integration","title":"Installing an Integration","text":"

    Install the Integration via pip.

    For example, to use prefect-aws:

    pip install prefect-aws\n
    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#registering-blocks-from-an-integration","title":"Registering Blocks from an Integration","text":"

    Once the Prefect Integration is installed, register the blocks within the integration to view them in the Prefect Cloud UI:

    For example, to register the blocks available in prefect-aws:

    prefect block register -m prefect_aws\n

    Updating blocks from an integrations

    If you install an updated Prefect integration that adds fields to a block type, you will need to re-register that block type.

    Loading a block in code

    To use the load method on a Block, you must already have a block document saved either through code or through the Prefect UI.

    Learn more about Blocks here!

    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#using-tasks-and-flows-from-an-integration","title":"Using Tasks and Flows from an Integration","text":"

    Integrations also contain pre-built tasks and flows that can be imported and called within your code.

    As an example, to read a secret from AWS Secrets Manager with the read_secret task:

    from prefect import flow\nfrom prefect_aws import AwsCredentials\nfrom prefect_aws.secrets_manager import read_secret\n\n@flow\ndef connect_to_database():\n    aws_credentials = AwsCredentials.load(\"MY_BLOCK_NAME\")\n    secret_value = read_secret(\n        secret_name=\"db_password\",\n        aws_credentials=aws_credentials\n    )\n\n    # Use secret_value to connect to a database\n
    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#customizing-tasks-and-flows-from-an-integration","title":"Customizing Tasks and Flows from an Integration","text":"

    To customize the settings of a task or flow pre-configured in a collection, use with_options:

    from prefect import flow\nfrom prefect_dbt.cloud import DbtCloudCredentials\nfrom prefect_dbt.cloud.jobs import trigger_dbt_cloud_job_run_and_wait_for_completion\n\ncustom_run_dbt_cloud_job = trigger_dbt_cloud_job_run_and_wait_for_completion.with_options(\n    name=\"Run My DBT Cloud Job\",\n    retries=2,\n    retry_delay_seconds=10\n)\n\n@flow\ndef run_dbt_job_flow():\n    run_result = custom_run_dbt_cloud_job(\n        dbt_cloud_credentials=DbtCloudCredentials.load(\"my-dbt-cloud-credentials\"),\n        job_id=1\n    )\n\nrun_dbt_job_flow()\n
    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"integrations/usage/#recipes-and-tutorials","title":"Recipes and Tutorials","text":"

    To learn more about how to use Integrations, check out Prefect recipes on GitHub. These recipes provide examples of how Integrations can be used in various scenarios.

    ","tags":["tasks","flows","blocks","integrations","task library","contributing"],"boost":2},{"location":"recipes/recipes/","title":"Prefect Recipes","text":"

    Prefect recipes are common, extensible examples for setting up Prefect in your execution environment with ready-made ingredients such as Dockerfiles, Terraform files, and GitHub Actions.

    Recipes are useful when you are looking for tutorials on how to deploy a worker, use event-driven flows, set up unit testing, and more.

    The following are Prefect recipes specific to Prefect 2. You can find a full repository of recipes at https://github.com/PrefectHQ/prefect-recipes and additional recipes at Prefect Discourse.

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#recipe-catalog","title":"Recipe catalog","text":"Agent on Azure with Kubernetes

    Configure Prefect on Azure with Kubernetes, running a Prefect agent to execute deployment flow runs.

    Maintained by Prefect

    This recipe uses:

    Agent on ECS Fargate with AWS CLI

    Run a Prefect 2 agent on ECS Fargate using the AWS CLI.

    Maintained by Prefect

    This recipe uses:

    Agent on ECS Fargate with Terraform

    Run a Prefect 2 agent on ECS Fargate using Terraform.

    Maintained by Prefect

    This recipe uses:

    Agent on an Azure VM

    Set up an Azure VM and run a Prefect agent.

    Maintained by Prefect

    This recipe uses:

    Flow Deployment with GitHub Actions

    Deploy a Prefect flow with storage and infrastructure blocks, update and push Docker image to container registry.

    Maintained by Prefect

    This recipe uses:

    Flow Deployment with GitHub Storage and Docker Infrastructure

    Create a deployment with GitHub as a storage and Docker Container as an infrastructure

    Maintained by Prefect

    This recipe uses:

    Prefect server on an AKS Cluster

    Deploy a Prefect server to an Azure Kubernetes Service (AKS) Cluster with Azure Blob Storage.

    Maintained by Prefect

    This recipe uses:

    Serverless Prefect with AWS Chalice

    Execute Prefect flows in an AWS Lambda function managed by Chalice.

    Maintained by Prefect

    This recipe uses:

    Serverless Workflows with ECSTask Blocks

    Deploy a Prefect agent to AWS ECS Fargate using GitHub Actions and ECSTask infrastructure blocks.

    Maintained by Prefect

    This recipe uses:

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#contributing-recipes","title":"Contributing recipes","text":"

    We're always looking for new recipe contributions! See the Prefect Recipes repository for details on how you can add your Prefect recipe, share best practices with fellow Prefect users, and earn some swag.

    Prefect recipes provide a vital cookbook where users can find helpful code examples and, when appropriate, common steps for specific Prefect use cases.

    We love recipes from anyone who has example code that another Prefect user can benefit from (e.g. a Prefect flow that loads data into Snowflake).

    Have a blog post, Discourse article, or tutorial you\u2019d like to share as a recipe? All submissions are welcome. Clone the prefect-recipes repo, create a branch, add a link to your recipe to the README, and submit a PR. Have more questions? Read on.

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#what-is-a-recipe","title":"What is a recipe?","text":"

    A Prefect recipe is like a cookbook recipe: it tells you what you need \u2014 the ingredients \u2014 and some basic steps, but assumes you can put the pieces together. Think of the Hello Fresh meal experience, but for dataflows.

    A tutorial, on the other hand, is Julia Child holding your hand through the entire cooking process: explaining each ingredient and procedure, demonstrating best practices, pointing out potential problems, and generally making sure you can\u2019t stray from the happy path to a delicious meal.

    We love Julia, and we love tutorials. But we don\u2019t expect that a Prefect recipe should handhold users through every step and possible contingency of a solution. A recipe can start from an expectation of more expertise and problem-solving ability on the part of the reader.

    To see an example of a high quality recipe, check out Serverless with AWS Chalice. This recipe includes all of the elements we like to see.

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#steps-to-add-your-recipe","title":"Steps to add your recipe","text":"

    Here\u2019s our guide to creating a recipe:

    # Clone the repository\ngit clone git@github.com:PrefectHQ/prefect-recipes.git\ncd prefect-recipes\n\n# Create and checkout a new branch\n\ngit checkout -b new_recipe_branch_name\n
    1. Add your recipe. Your code may simply be a copy/paste of a single Python file or an entire folder. Unsure of where to add your file or folder? Just add under the flows-advanced/ folder. A Prefect Recipes maintainer will help you find the best place for your recipe. Just want to direct others to a project you made, whether it be a repo or a blogpost? Simply link to it in the Prefect Recipes README!
    2. (Optional) Write a README.
    3. Include a dependencies file, if applicable.
    4. Push your code and make a PR to the repository.

    That\u2019s it!

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#what-makes-a-good-recipe","title":"What makes a good recipe?","text":"

    Every recipe is useful, as other Prefect users can adapt the recipe to their needs. Particularly good ones help a Prefect user bake a great dataflow solution! Take a look at the prefect-recipes repo to see some examples.

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#what-are-the-common-ingredients-of-a-good-recipe","title":"What are the common ingredients of a good recipe?","text":"
    • Easy to understand: Can a user easily follow your recipe? Would a README or code comments help? A simple explanation providing context on how to use the example code is useful, but not required. A good README can set a recipe apart, so we have some additional suggestions for README files below.
    • Code and more: Sometimes a use case is best represented in Python code or shell scripts. Sometimes a configuration file is the most important artifact \u2014 think of a Dockerfile or Terraform file for configuring infrastructure.
    • All-inclusive: Share as much code as you can. Even boilerplate code like Dockerfiles or Terraform or Helm files are useful. Just don\u2019t share company secrets or IP.
    • Specific: Don't worry about generalizing your code, aside from removing anything internal/secret! Other users will extrapolate their own unique solutions from your example.
    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#what-are-some-tips-for-a-good-recipe-readme","title":"What are some tips for a good recipe README?","text":"

    A thoughtful README can take a recipe from good to great. Here are some best practices that we\u2019ve found make for a great recipe README:

    • Provide a brief explanation of what your recipe demonstrates. This helps users determine quickly whether the recipe is relevant to their needs or answers their questions.
    • List which files are included and what each is meant to do. Each explanation can contain only a few words.
    • Describe any dependencies and prerequisites (in addition to any dependencies you include in a requirements file). This includes both libraries or modules and any services your recipes depends on.
    • If steps are involved or there\u2019s an order to do things, a simple list of steps is helpful.
    • Bonus: troubleshooting steps you encountered to get here or tips where other users might get tripped up.
    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"recipes/recipes/#next-steps","title":"Next steps","text":"

    We hope you\u2019ll feel comfortable sharing your Prefect solutions as recipes in the prefect-recipes repo. Collaboration and knowledge sharing are defining attributes of our Prefect Community!

    Have questions about sharing or using recipes? Reach out on our active Prefect Slack Community!

    Happy engineering!

    ","tags":["recipes","best practices","examples"],"boost":2},{"location":"tutorial/","title":"Tutorial Overview","text":"

    This tutorial provides a guided walk-through of Prefect core concepts and instructions on how to use them.

    By the end of this tutorial you will have:

    1. Created a flow
    2. Added tasks to it
    3. Deployed and run the flow locally
    4. Created a work pool and run the flow remotely

    These four topics will get most users to their first production deployment.

    Advanced users that need more governance and control of their workflow infrastructure can go one step further by:

    1. Using a worker-based deployment

    If you're looking for examples of more advanced operations (like deploying on Kubernetes), check out Prefect's guides.

    ","tags":["tutorial","getting started","basics","tasks","flows","subflows","deployments","workers","work pools"],"boost":2},{"location":"tutorial/#prerequisites","title":"Prerequisites","text":"
    1. Before you start, make sure you have Python installed, then install Prefect: pip install -U prefect

    See the install guide for more detailed instructions, if needed.

    1. To use Prefect, you need to self-host a Prefect server or connect to Prefect Cloud.

    To get the most out of this tutorial, we recommend using Prefect Cloud. Sign up for a forever free Prefect Cloud account or accept your organization's invite to join their Prefect Cloud account.

    1. Create a new account or sign in at https://app.prefect.cloud/.
    2. Use the prefect cloud login CLI command to authenticate to Prefect Cloud from your environment.
    prefect cloud login\n

    Choose Log in with a web browser and click the Authorize button in the browser window that opens.

    As an alternative to using Prefect Cloud, you can self-host a Prefect server instance. If you choose this option, run prefect server start to start a local Prefect server instance.

    ","tags":["tutorial","getting started","basics","tasks","flows","subflows","deployments","workers","work pools"],"boost":2},{"location":"tutorial/#what-is-prefect","title":"What is Prefect?","text":"

    Prefect orchestrates workflows \u2014 it simplifies the creation, scheduling, and monitoring of complex data pipelines. With Prefect, you define workflows as Python code and let it handle the rest.

    Prefect also provides error handling, retry mechanisms, and a user-friendly dashboard for monitoring. It's the easiest way to transform any Python function into a unit of work that can be observed and orchestrated.

    Just bring your Python code, sprinkle in a few decorators, and go!

    ","tags":["tutorial","getting started","basics","tasks","flows","subflows","deployments","workers","work pools"],"boost":2},{"location":"tutorial/#first-steps-flows","title":"First steps: Flows","text":"

    Let's begin by learning how to create your first Prefect flow - click here to get started.

    ","tags":["tutorial","getting started","basics","tasks","flows","subflows","deployments","workers","work pools"],"boost":2},{"location":"tutorial/deployments/","title":"Deploying Flows","text":"

    Reminder to connect to Prefect Cloud or a self-hosted Prefect server instance

    Some features in this tutorial, such as scheduling, require you to be connected to a Prefect server. If using a self-hosted setup, run prefect server start to run both the webserver and UI. If using Prefect Cloud, make sure you have successfully authenticated your local environment.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#why-deployments","title":"Why deployments?","text":"

    Some of the most common reasons to use an orchestration tool such as Prefect are for scheduling and event-based triggering. Up to this point, we\u2019ve demonstrated running Prefect flows as scripts, but this means you have been the one triggering and managing flow runs. You can certainly continue to trigger your workflows in this way and use Prefect as a monitoring layer for other schedulers or systems, but you will miss out on many of the other benefits and features that Prefect offers.

    Deploying a flow exposes an API and UI so that you can:

    • trigger new runs, cancel active runs, pause scheduled runs, customize parameters, and more
    • remotely configure schedules and automation rules for your deployments
    • dynamically provision infrastructure using workers
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#what-is-a-deployment","title":"What is a deployment?","text":"

    Deploying a flow is the act of specifying where and how it will run. This information is encapsulated and sent to Prefect as a deployment that contains the crucial metadata needed for remote orchestration. Deployments elevate workflows from functions that you call manually to API-managed entities.

    Attributes of a deployment include (but are not limited to):

    • Flow entrypoint: path to your flow function
    • Schedule or Trigger: optional schedule or triggering rule for this deployment
    • Tags: optional text labels for organizing your deployments
    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#create-a-deployment","title":"Create a deployment","text":"

    Using our get_repo_info flow from the previous sections, we can easily create a deployment for it by calling a single method on the flow object: flow.serve.

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.serve(name=\"my-first-deployment\")\n

    Running this script will do two things:

    • create a deployment called \"my-first-deployment\" for your flow in the Prefect API
    • stay running to listen for flow runs for this deployment; when a run is found, it will be asynchronously executed within a subprocess

    Deployments must be defined in static files

    Flows can be defined and run interactively, that is, within REPLs or Notebooks. Deployments, on the other hand, require that your flow definition be in a known file (which can be located on a remote filesystem in certain setups, as we'll see in the next section of the tutorial).

    Because this deployment has no schedule or triggering automation, you will need to use the UI or API to create runs for it. Let's use the CLI (in a separate terminal window) to create a run for this deployment:

    prefect deployment run 'get-repo-info/my-first-deployment'\n

    If you are watching either your terminal or your UI, you should see the newly created run execute successfully! Let's take this example further by adding a schedule and additional metadata.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#additional-options","title":"Additional options","text":"

    The serve method on flows exposes many options for the deployment. Let's use a few of these options now:

    • cron: a keyword that allows us to set a cron string schedule for the deployment; see schedules for more advanced scheduling options
    • tags: a keyword that allows us to tag this deployment and its runs for bookkeeping and filtering purposes
    • description: a keyword that allows us to document what this deployment does; by default the description is set from the docstring of the flow function, but we did not document our flow function
    • version: a keyword that allows us to track changes to our deployment; by default a hash of the file containing the flow is used; popular options include semver tags or git commit hashes

    Let's add these options to our deployment:

    if __name__ == \"__main__\":\n    get_repo_info.serve(\n        name=\"my-first-deployment\",\n        cron=\"* * * * *\",\n        tags=[\"testing\", \"tutorial\"],\n        description=\"Given a GitHub repository, logs repository statistics for that repo.\",\n        version=\"tutorial/deployments\",\n    )\n

    When you rerun this script, you will find an updated deployment in the UI that is actively scheduling work! Stop the script in the CLI using CTRL+C and your schedule will be automatically paused.

    .serve is a long-running process

    For remotely triggered or scheduled runs to be executed, your script with flow.serve must be actively running.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#running-multiple-deployments-at-once","title":"Running multiple deployments at once","text":"

    This method is useful for creating deployments for single flows, but what if we have two or more flows? This situation only requires a few additional method calls and imports to get up and running:

    multi_flow_deployment.py
    import time\nfrom prefect import flow, serve\n\n\n@flow\ndef slow_flow(sleep: int = 60):\n    \"Sleepy flow - sleeps the provided amount of time (in seconds).\"\n    time.sleep(sleep)\n\n\n@flow\ndef fast_flow():\n    \"Fastest flow this side of the Mississippi.\"\n    return\n\n\nif __name__ == \"__main__\":\n    slow_deploy = slow_flow.to_deployment(name=\"sleeper\", interval=45)\n    fast_deploy = fast_flow.to_deployment(name=\"fast\")\n    serve(slow_deploy, fast_deploy)\n

    A few observations are in order:

    • the flow.to_deployment interface exposes the exact same options as flow.serve; this method produces a deployment object
    • the deployments are only registered with the API once serve(...) is called
    • when serving multiple deployments, the only requirement is that they share a Python environment; they can be executed and scheduled independently of each other

    Spend some time experimenting with this setup. A few potential next steps for exploration include:

    • pausing and unpausing the schedule for the \"sleeper\" deployment
    • using the UI to submit ad-hoc runs for the \"sleeper\" deployment with different values for sleep
    • cancelling an active run for the \"sleeper\" deployment from the UI (good luck cancelling the \"fast\" one \ud83d\ude09)

    Hybrid execution option

    Another implication of Prefect's deployment interface is that you can choose to use our hybrid execution model. Whether you use Prefect Cloud or host a Prefect server instance yourself, you can run work flows in the environments best suited to their execution. This model allows you efficient use of your infrastructure resources while maintaining the privacy of your code and data. There is no ingress required. For more information read more about our hybrid model.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/deployments/#next-steps","title":"Next steps","text":"

    Congratulations! You now have your first working deployment.

    Deploying flows through the serve method is a fast way to start scheduling flows with Prefect. However, if your team has more complex infrastructure requirements or you'd like to have Prefect manage flow execution, you can deploy flows to a work pool.

    Learn about work pools and how Prefect Cloud can handle infrastructure configuration for you in the next step of the tutorial.

    ","tags":["orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/flows/","title":"Flows","text":"

    Prerequisites

    This tutorial assumes you have already installed Prefect and connected to Prefect Cloud or a self-hosted server instance. See the prerequisites section of the tutorial for more details.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#what-is-a-flow","title":"What is a flow?","text":"

    Flows are like functions. They can take inputs, perform work, and return an output. In fact, you can turn any function into a Prefect flow by adding the @flow decorator. When a function becomes a flow, its behavior changes, giving it the following advantages:

    • All runs of the flow have persistent state. Transitions between states are recorded, allowing for flow execution to be observed and acted upon.
    • Input arguments can be type validated as workflow parameters.
    • Retries can be performed on failure.
    • Timeouts can be enforced to prevent unintentional, long-running workflows.
    • Metadata about flow runs, such as run time and final state, is automatically tracked.
    • They can easily be elevated to a deployment, which exposes a remote API for interacting with it
    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#run-your-first-flow","title":"Run your first flow","text":"

    The simplest way to get started with Prefect is to annotate a Python function with the\u00a0@flow\u00a0decorator. The script below fetches statistics about the main Prefect repository. Let's turn it into a Prefect flow and run it:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow\ndef get_repo_info():\n    url = \"https://api.github.com/repos/PrefectHQ/prefect\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(\"PrefectHQ/prefect repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\nif __name__ == \"__main__\":\n    get_repo_info()\n

    Running this file will result in some interesting output:

    12:47:42.792 | INFO | prefect.engine - Created flow run 'ludicrous-warthog' for flow 'get-repo-info'\nPrefectHQ/prefect repository statistics \ud83e\udd13:\nStars \ud83c\udf20 : 12146\nForks \ud83c\udf74 : 1245\n12:47:45.008 | INFO | Flow run 'ludicrous-warthog' - Finished in state Completed()\n

    Flows can contain arbitrary Python

    As we can see above, flow definitions can contain arbitrary Python logic.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#parameters","title":"Parameters","text":"

    As with any Python function, you can pass arguments to a flow. The positional and keyword arguments defined on your flow function are called parameters. Prefect will automatically perform type conversion using any provided type hints. Let's make the repository a string parameter with a default value:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info(repo_name=\"PrefectHQ/marvin\")\n

    We can call our flow with varying values for the repo_name parameter (including \"bad\" values):

    python repo_info.py\n

    Try passing repo_name=\"missing-org/missing-repo\".

    You should see

    HTTPStatusError: Client error '404 Not Found' for url '<https://api.github.com/repos/missing-org/missing-repo>'\n

    Now navigate to your Prefect dashboard and compare the displays for these two runs.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#logging","title":"Logging","text":"

    Prefect enables you to log a variety of useful information about your flow and task runs, capturing information about your workflows for purposes such as monitoring, troubleshooting, and auditing. If we navigate to our dashboard and explore the runs we created above, we will notice that the repository statistics are not captured in the flow run logs. Let's fix that by adding some logging to our flow:

    repo_info.py
    import httpx\nfrom prefect import flow, get_run_logger\n\n\n@flow\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    logger = get_run_logger()\n    logger.info(\"%s repository statistics \ud83e\udd13:\", repo_name)\n    logger.info(f\"Stars \ud83c\udf20 : %d\", repo[\"stargazers_count\"])\n    logger.info(f\"Forks \ud83c\udf74 : %d\", repo[\"forks_count\"])\n

    Now the output looks more consistent and, more importantly, our statistics are stored in the Prefect backend and displayed in the UI for this flow run:

    12:47:42.792 | INFO    | prefect.engine - Created flow run 'ludicrous-warthog' for flow 'get-repo-info'\n12:47:43.016 | INFO    | Flow run 'ludicrous-warthog' - PrefectHQ/prefect repository statistics \ud83e\udd13:\n12:47:43.016 | INFO    | Flow run 'ludicrous-warthog' - Stars \ud83c\udf20 : 12146\n12:47:43.042 | INFO    | Flow run 'ludicrous-warthog' - Forks \ud83c\udf74 : 1245\n12:47:45.008 | INFO    | Flow run 'ludicrous-warthog' - Finished in state Completed()\n

    log_prints=True

    We could have achieved the exact same outcome by using Prefect's convenient log_prints keyword argument in the flow decorator:

    @flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    ...\n

    Logging vs Artifacts

    The example above is for educational purposes. In general, it is better to use Prefect artifacts for storing metrics and output. Logs are best for tracking progress and debugging errors.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#retries","title":"Retries","text":"

    So far our script works, but in the future unexpected errors may occur. For example the GitHub API may be temporarily unavailable or rate limited. Retries help make our flow more resilient. Let's add retry functionality to our example above:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(retries=3, retry_delay_seconds=5, log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\nif __name__ == \"__main__\":\n    get_repo_info()\n
    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/flows/#next-tasks","title":"Next: Tasks","text":"

    As you have seen, adding a flow decorator converts our Python function to a resilient and observable workflow. In the next section, you'll supercharge this flow by using tasks to break down the workflow's complexity and make it more performant and observable - click here to continue.

    ","tags":["tutorial","getting started","basics","flows","logging","parameters","retries"]},{"location":"tutorial/tasks/","title":"Tasks","text":"","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#what-is-a-task","title":"What is a task?","text":"

    A task is any Python function decorated with a @task decorator called within a flow. You can think of a flow as a recipe for connecting a known sequence of tasks together. Tasks, and the dependencies between them, are displayed in the flow run graph, enabling you to break down a complex flow into something you can observe, understand and control at a more granular level. When a function becomes a task, it can be executed concurrently and its return value can be cached.

    Flows and tasks share some common features:

    • Both are defined easily using their respective decorator, which accepts settings for that flow / task (see all task settings / flow settings).
    • Each can be given a name, description and tags for organization and bookkeeping.
    • Both provide functionality for retries, timeouts, and other hooks to handle failure and completion events.

    Network calls (such as our GET requests to the GitHub API) are particularly useful as tasks because they take advantage of task features such as retries, caching, and concurrency.

    Tasks must be called from flows

    All tasks must be called from within a flow. Tasks may not call other tasks directly.

    When to use tasks

    Not all functions in a flow need be tasks. Use them only when their features are useful.

    Let's take our flow from before and move the request into a task:

    repo_info.py
    import httpx\nfrom prefect import flow, task\n\n\n@task\ndef get_url(url: str, params: dict = None):\n    response = httpx.get(url, params=params)\n    response.raise_for_status()\n    return response.json()\n\n\n@flow(retries=3, retry_delay_seconds=5, log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    repo_stats = get_url(url)\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo_stats['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo_stats['forks_count']}\")\n\nif __name__ == \"__main__\":\n    get_repo_info()\n

    Running the flow in your terminal will result in something like this:

    09:55:55.412 | INFO    | prefect.engine - Created flow run 'great-ammonite' for flow 'get-repo-info'\n09:55:55.499 | INFO    | Flow run 'great-ammonite' - Created task run 'get_url-0' for task 'get_url'\n09:55:55.500 | INFO    | Flow run 'great-ammonite' - Executing 'get_url-0' immediately...\n09:55:55.825 | INFO    | Task run 'get_url-0' - Finished in state Completed()\n09:55:55.827 | INFO    | Flow run 'great-ammonite' - PrefectHQ/prefect repository statistics \ud83e\udd13:\n09:55:55.827 | INFO    | Flow run 'great-ammonite' - Stars \ud83c\udf20 : 12157\n09:55:55.827 | INFO    | Flow run 'great-ammonite' - Forks \ud83c\udf74 : 1251\n09:55:55.849 | INFO    | Flow run 'great-ammonite' - Finished in state Completed('All states completed.')\n

    And you should now see this task run tracked in the UI as well.

    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#caching","title":"Caching","text":"

    Tasks support the ability to cache their return value. Caching allows you to efficiently reuse results of tasks that may be expensive to reproduce with every flow run, or reuse cached results if the inputs to a task have not changed.

    To enable caching, specify a cache_key_fn \u2014 a function that returns a cache key \u2014 on your task. You may optionally provide a cache_expiration timedelta indicating when the cache expires. You can define a task that is cached based on its inputs by using the Prefect task_input_hash. Let's add caching to our get_url task:

    import httpx\nfrom datetime import timedelta\nfrom prefect import flow, task, get_run_logger\nfrom prefect.tasks import task_input_hash\n\n\n@task(cache_key_fn=task_input_hash, \n      cache_expiration=timedelta(hours=1),\n      )\ndef get_url(url: str, params: dict = None):\n    response = httpx.get(url, params=params)\n    response.raise_for_status()\n    return response.json()\n

    You can test this caching behavior by using a personal repository as your workflow parameter - give it a star, or remove a star and see how the output of this task changes (or doesn't) by running your flow multiple times.

    Task results and caching

    Task results are cached in memory during a flow run and persisted to your home directory by default. Prefect Cloud only stores the cache key, not the data itself.

    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#concurrency","title":"Concurrency","text":"

    Tasks enable concurrency, allowing you to execute multiple tasks asynchronously. This concurrency can greatly enhance the efficiency and performance of your workflows. Let's expand our script to calculate the average open issues per user. This will require making more requests:

    repo_info.py
    import httpx\nfrom datetime import timedelta\nfrom prefect import flow, task\nfrom prefect.tasks import task_input_hash\n\n\n@task(cache_key_fn=task_input_hash, cache_expiration=timedelta(hours=1))\ndef get_url(url: str, params: dict = None):\n    response = httpx.get(url, params=params)\n    response.raise_for_status()\n    return response.json()\n\n\ndef get_open_issues(repo_name: str, open_issues_count: int, per_page: int = 100):\n    issues = []\n    pages = range(1, -(open_issues_count // -per_page) + 1)\n    for page in pages:\n        issues.append(\n            get_url(\n                f\"https://api.github.com/repos/{repo_name}/issues\",\n                params={\"page\": page, \"per_page\": per_page, \"state\": \"open\"},\n            )\n        )\n    return [i for p in issues for i in p]\n\n\n@flow(retries=3, retry_delay_seconds=5, log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    repo_stats = get_url(f\"https://api.github.com/repos/{repo_name}\")\n    issues = get_open_issues(repo_name, repo_stats[\"open_issues_count\"])\n    issues_per_user = len(issues) / len(set([i[\"user\"][\"id\"] for i in issues]))\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo_stats['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo_stats['forks_count']}\")\n    print(f\"Average open issues per user \ud83d\udc8c : {issues_per_user:.2f}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info()\n

    Now we're fetching the data we need, but the requests are happening sequentially. Tasks expose a submit method that changes the execution from sequential to concurrent. In our specific example, we also need to use the result method because we are unpacking a list of return values:

    def get_open_issues(repo_name: str, open_issues_count: int, per_page: int = 100):\n    issues = []\n    pages = range(1, -(open_issues_count // -per_page) + 1)\n    for page in pages:\n        issues.append(\n            get_url.submit(\n                f\"https://api.github.com/repos/{repo_name}/issues\",\n                params={\"page\": page, \"per_page\": per_page, \"state\": \"open\"},\n            )\n        )\n    return [i for p in issues for i in p.result()]\n

    The logs show that each task is running concurrently:

    12:45:28.241 | INFO    | prefect.engine - Created flow run 'intrepid-coua' for flow 'get-repo-info'\n12:45:28.311 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-0' for task 'get_url'\n12:45:28.312 | INFO    | Flow run 'intrepid-coua' - Executing 'get_url-0' immediately...\n12:45:28.543 | INFO    | Task run 'get_url-0' - Finished in state Completed()\n12:45:28.583 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-1' for task 'get_url'\n12:45:28.584 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-1' for execution.\n12:45:28.594 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-2' for task 'get_url'\n12:45:28.594 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-2' for execution.\n12:45:28.609 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-4' for task 'get_url'\n12:45:28.610 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-4' for execution.\n12:45:28.624 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-5' for task 'get_url'\n12:45:28.625 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-5' for execution.\n12:45:28.640 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-6' for task 'get_url'\n12:45:28.641 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-6' for execution.\n12:45:28.708 | INFO    | Flow run 'intrepid-coua' - Created task run 'get_url-3' for task 'get_url'\n12:45:28.708 | INFO    | Flow run 'intrepid-coua' - Submitted task run 'get_url-3' for execution.\n12:45:29.096 | INFO    | Task run 'get_url-6' - Finished in state Completed()\n12:45:29.565 | INFO    | Task run 'get_url-2' - Finished in state Completed()\n12:45:29.721 | INFO    | Task run 'get_url-5' - Finished in state Completed()\n12:45:29.749 | INFO    | Task run 'get_url-4' - Finished in state Completed()\n12:45:29.801 | INFO    | Task run 'get_url-3' - Finished in state Completed()\n12:45:29.817 | INFO    | Task run 'get_url-1' - Finished in state Completed()\n12:45:29.820 | INFO    | Flow run 'intrepid-coua' - PrefectHQ/prefect repository statistics \ud83e\udd13:\n12:45:29.820 | INFO    | Flow run 'intrepid-coua' - Stars \ud83c\udf20 : 12159\n12:45:29.821 | INFO    | Flow run 'intrepid-coua' - Forks \ud83c\udf74 : 1251\nAverage open issues per user \ud83d\udc8c : 2.27\n12:45:29.838 | INFO    | Flow run 'intrepid-coua' - Finished in state Completed('All states completed.')\n
    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#subflows","title":"Subflows","text":"

    Not only can you call tasks within a flow, but you can also call other flows! Child flows are called\u00a0subflows\u00a0and allow you to efficiently manage, track, and version common multi-task logic.

    Subflows are a great way to organize your workflows and offer more visibility within the UI.

    Let's add a flow decorator to our get_open_issues function:

    @flow\ndef get_open_issues(repo_name: str, open_issues_count: int, per_page: int = 100):\n    issues = []\n    pages = range(1, -(open_issues_count // -per_page) + 1)\n    for page in pages:\n        issues.append(\n            get_url.submit(\n                f\"https://api.github.com/repos/{repo_name}/issues\",\n                params={\"page\": page, \"per_page\": per_page, \"state\": \"open\"},\n            )\n        )\n    return [i for p in issues for i in p.result()]\n

    Whenever we run the parent flow, a new run will be generated for related functions within that as well. Not only is this run tracked as a subflow run of the main flow, but you can also inspect it independently in the UI!

    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/tasks/#next-deployments","title":"Next: Deployments","text":"

    We now have a flow with tasks, subflows, retries, logging, caching, and concurrent execution. In the next section, we'll see how we can deploy this flow in order to run it on a schedule and/or external infrastructure - click here to learn how to create your first deployment.

    ","tags":["tutorial","getting started","basics","tasks","caching","concurrency","subflows"]},{"location":"tutorial/work-pools/","title":"Work Pools","text":"","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#why-work-pools","title":"Why work pools?","text":"

    Work pools are a bridge between the Prefect orchestration layer and infrastructure for flow runs that can be dynamically provisioned. To transition from persistent infrastructure to dynamic infrastructure, use flow.deploy instead of flow.serve.

    Choosing Between flow.deploy() and flow.serve()

    Earlier in the tutorial you used serve to deploy your flows. For many use cases, serve is sufficient to meet scheduling and orchestration needs. Work pools are optional. If infrastructure needs escalate, work pools can become a handy tool. The best part? You're not locked into one method. You can seamlessly combine approaches as needed.

    Deployment definition methods differ slightly for work pools

    When you use work-pool-based execution, you define deployments differently. Deployments for workers are configured with deploy, which requires additional configuration. A deployment created with serve cannot be used with a work pool.

    The primary reason to use work pools is for dynamic infrastructure provisioning and configuration. For example, you might have a workflow that has expensive infrastructure requirements and is run infrequently. In this case, you don't want an idle process running within that infrastructure.

    Other advantages to using work pools include:

    • You can configure default infrastructure configurations on your work pools that all jobs inherit and can override.
    • Platform teams can use work pools to expose opinionated (and enforced!) interfaces to the infrastructure that they oversee.
    • Work pools can be used to prioritize (or limit) flow runs through the use of work queues.

    Prefect provides several types of work pools. Prefect Cloud provides a Prefect Managed work pool option that is the simplest way to run workflows remotely. A cloud-provider account, such as AWS, is not required with a Prefect Managed work pool.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#set-up-a-work-pool","title":"Set up a work pool","text":"

    Prefect Cloud

    This tutorial uses Prefect Cloud to deploy flows to work pools. Managed execution and push work pools are available in Prefect Cloud only. If you are not using Prefect Cloud, please learn about work pools below and then proceed to the next tutorial that uses worker-based work pools.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#create-a-prefect-managed-work-pool","title":"Create a Prefect Managed work pool","text":"

    In your terminal, run the following command to set up a work pool named my-managed-pool of type prefect:managed.

    prefect work-pool create my-managed-pool --type prefect:managed \n

    Let\u2019s confirm that the work pool was successfully created by running the following command.

    prefect work-pool ls\n

    You should see your new my-managed-pool in the output list.

    Finally, let\u2019s double check that you can see this work pool in the UI.

    Navigate to the Work Pools tab and verify that you see my-managed-pool listed.

    Feel free to select Edit from the three-dot menu on right of the work pool card to view the details of your work pool.

    Work pools contain configuration that is used to provision infrastructure for flow runs. For example, you can specify additional Python packages or environment variables that should be set for all deployments that use this work pool. Note that individual deployments can override the work pool configuration.

    Now that you\u2019ve set up your work pool, we can deploy a flow to this work pool. Let's deploy your tutorial flow to my-managed-pool.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#create-the-deployment","title":"Create the deployment","text":"

    From our previous steps, we now have:

    1. A flow
    2. A work pool

    Let's update our repo_info.py file to create a deployment in Prefect Cloud.

    The updates that we need to make to repo_info.py are:

    1. Change flow.serve to flow.deploy.
    2. Tell flow.deploy which work pool to deploy to.

    Here's what the updated repo_info.py looks like:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.from_source(\n        source=\"https://github.com/discdiver/demos.git\", \n        entrypoint=\"repo_info.py:get_repo_info\"\n    ).deploy(\n        name=\"my-first-deployment\", \n        work_pool_name=\"my-managed-pool\", \n    )\n

    In the from_source method, we specify the source of our flow code.

    In the deploy method, we specify the name of our deployment and the name of the work pool that we created earlier.

    You can store your flow code in any of several types of remote storage. In this example, we use a GitHub repository, but you could use a Docker image, as you'll see in an upcoming section of the tutorial. Alternatively, you could store your flow code in cloud provider storage such as AWS S3, or within a different git-based cloud provider such as GitLab or Bitbucket.

    Note

    In the example above, we store our code in a GitHub repository. If you make changes to the flow code, you will need to push those changes to your own GitHub account and update the source argument of from_source to point to your repository.

    Run the script again and you should see a message in the CLI that your deployment was created with instructions for how to run it.

    Successfully created/updated all deployments!\n\n                       Deployments                       \n\u250f\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2533\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2513\n\u2503 Name                              \u2503 Status  \u2503 Details \u2503\n\u2521\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2547\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2529\n\u2502 get-repo-info/my-first-deployment  | applied \u2502         \u2502\n\u2514\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2534\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2518\n\nTo schedule a run for this deployment, use the following command:\n\n        $ prefect deployment run 'get-repo-info/my-first-deployment'\n\n\nYou can also run your flow via the Prefect UI: https://app.prefect.cloud/account/\nabc/workspace/123/deployments/deployment/xyz\n

    Navigate to your Prefect Cloud UI and view your new deployment. Click the Run button to trigger a run of your deployment.

    Because this deployment was configured with a Prefect Managed work pool, Prefect Cloud will run your flow on your behalf.

    View the logs in the UI.

    Now that you've updated your script, you can run it to register your deployment on Prefect Cloud:

    python repo_info.py\n
    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#schedule-a-deployment-run","title":"Schedule a deployment run","text":"

    Now everything is set up for us to submit a flow-run to the work pool. Go ahead and run the deployment from the CLI or the UI.

    prefect deployment run 'get_repo_info/my-deployment'\n

    Prefect Managed work pools are a great way to get started with Prefect. See the Managed Execution guide for more details.

    Many users will find that they need more control over the infrastructure that their flows run on. Prefect Cloud's push work pools are a popular option in those cases.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#push-work-pools-with-automatic-infrastructure-provisioning","title":"Push work pools with automatic infrastructure provisioning","text":"

    Serverless push work pools scale infinitely and provide more configuration options than Prefect Managed work pools.

    Prefect provides push work pools for AWS ECS on Fargate, Azure Container Instances, and Google Cloud Run. You will need to have an account with sufficient permissions on the cloud provider that you want to use. We'll use GCP for this example.

    Setting up the cloud provider pieces for infrastructure can be tricky and time consuming. Fortunately, Prefect can automatically provision infrastructure for you and wire it all together to work with your push work pool.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#create-a-push-work-pool-with-automatic-infrastructure-provisioning","title":"Create a push work pool with automatic infrastructure provisioning","text":"

    In your terminal, run the following command to set up a push work pool.

    Install the gcloud CLI and authenticate with your GCP project.

    If you already have the gcloud CLI installed, be sure to update to the latest version with gcloud components update.

    You will need the following permissions in your GCP project:

    • resourcemanager.projects.list
    • serviceusage.services.enable
    • iam.serviceAccounts.create
    • iam.serviceAccountKeys.create
    • resourcemanager.projects.setIamPolicy
    • artifactregistry.repositories.create

    Docker is also required to build and push images to your registry. You can install Docker here.

    Run the following command to set up a work pool named my-cloud-run-pool of type cloud-run:push.

    prefect work-pool create --type cloud-run:push --provision-infra my-cloud-run-pool \n

    Using the --provision-infra flag will allow you to select a GCP project to use for your work pool and automatically configure it to be ready to execute flows via Cloud Run. In your GCP project, this command will activate the Cloud Run API, create a service account, and create a key for the service account, if they don't already exist. In your Prefect workspace, this command will create a GCPCredentials block for storing the service account key.

    Here's an abbreviated example output from running the command:

    \u256d\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256e\n\u2502 Provisioning infrastructure for your work pool my-cloud-run-pool will require:                           \u2502\n\u2502                                                                                                          \u2502\n\u2502     Updates in GCP project central-kit-405415 in region us-central1                                      \u2502\n\u2502                                                                                                          \u2502\n\u2502         - Activate the Cloud Run API for your project                                                    \u2502\n\u2502         - Activate the Artifact Registry API for your project                                            \u2502\n\u2502         - Create an Artifact Registry repository named prefect-images                                    \u2502\n\u2502         - Create a service account for managing Cloud Run jobs: prefect-cloud-run                        \u2502\n\u2502             - Service account will be granted the following roles:                                       \u2502\n\u2502                 - Service Account User                                                                   \u2502\n\u2502                 - Cloud Run Developer                                                                    \u2502\n\u2502         - Create a key for service account prefect-cloud-run                                             \u2502\n\u2502                                                                                                          \u2502\n\u2502     Updates in Prefect workspace                                                                         \u2502\n\u2502                                                                                                          \u2502\n\u2502         - Create GCP credentials block my--pool-push-pool-credentials to store the service account key   \u2502\n\u2502                                                                                                          \u2502\n\u2570\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u256f\nProceed with infrastructure provisioning? [y/n]: y\nActivating Cloud Run API\nActivating Artifact Registry API\nCreating Artifact Registry repository\nConfiguring authentication to Artifact Registry\nSetting default Docker build namespace\nCreating service account\nAssigning roles to service account\nCreating service account key\nCreating GCP credentials block\nProvisioning Infrastructure \u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501\u2501 100% 0:00:00\nInfrastructure successfully provisioned!\nCreated work pool 'my-cloud-run-pool'!\n

    After infrastructure provisioning completes, you will be logged into your new Artifact Registry repository and the default Docker build namespace will be set to the URL of the repository.

    While the default namespace is set, any images you build without specifying a registry or username/organization will be pushed to the repository.

    To take advantage of this functionality, you can write your deploy script like this:

    example_deploy_script.py
    from prefect import flow                                                       \nfrom prefect.deployments import DeploymentImage                                \n\n\n@flow(log_prints=True)\ndef my_flow(name: str = \"world\"):\n    print(f\"Hello {name}! I'm a flow running on Cloud Run!\")\n\n\nif __name__ == \"__main__\":                                                     \n    my_flow.deploy(                                                            \n        name=\"my-deployment\",\n        work_pool_name=\"above-ground\",\n        image=DeploymentImage(\n            name=\"my-image:latest\",\n            platform=\"linux/amd64\",\n        )\n    )\n

    Running this script will build a Docker image with the tag <region>-docker.pkg.dev/<project>/<repository-name>/my-image:latest and push it to your repository.

    Tip

    Make sure you have Docker running locally before running this script.

    Note that you only need to include an object of the DeploymentImage class with the argument platform=\"linux/amd64 if you're building your image on a machine with an ARM-based processor. Otherwise, you could just pass image=\"my-image:latest\" to deploy.

    See the Push Work Pool guide for more details and example commands for each cloud provider.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/work-pools/#next-step","title":"Next step","text":"

    Congratulations! You've learned how to deploy flows to work pools. If these work pool options meet all of your needs, we encourage you to go deeper with the concepts docs or explore our how-to guides to see examples of particular Prefect use cases.

    However, if you need more control over your infrastructure, want to run your workflows in Kubernetes, or are running a self-hosted Prefect server instance, we encourage you to see the next section of the tutorial. There you'll learn how to use work pools that rely on a worker and see how to customize Docker images for container-based infrastructure.

    ","tags":["work pools","orchestration","flow runs","deployments","schedules","tutorial"],"boost":2},{"location":"tutorial/workers/","title":"Workers","text":"","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#prerequisites","title":"Prerequisites","text":"

    Docker installed and running on your machine.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#why-workers","title":"Why workers","text":"

    In the previous section of the tutorial, you learned how work pools are a bridge between the Prefect orchestration layer and infrastructure for flow runs that can be dynamically provisioned. You saw how you can transition from persistent infrastructure to dynamic infrastructure by using flow.deploy instead of flow.serve.

    Work pools that rely on client-side workers take this a step further by enabling you to run work flows in your own Docker containers, Kubernetes clusters, and serverless environments such as AWS ECS, Azure Container Instances, and GCP Cloud Run.

    The architecture of a worker-based work pool deployment can be summarized with the following diagram:

    graph TD\n    subgraph your_infra[\"Your Execution Environment\"]\n        worker[\"Worker\"]\n    subgraph flow_run_infra[Flow Run Infra]\n     flow_run_a((\"Flow Run A\"))\n    end\n    subgraph flow_run_infra_2[Flow Run Infra]\n     flow_run_b((\"Flow Run B\"))\n    end      \n    end\n\n    subgraph api[\"Prefect API\"]\n    Deployment --> |assigned to| work_pool\n        work_pool([\"Work Pool\"])\n    end\n\n    worker --> |polls| work_pool\n    worker --> |creates| flow_run_infra\n    worker --> |creates| flow_run_infra_2

    Notice above that the worker is in charge of provisioning the flow run infrastructure. In context of this tutorial, that flow run infrastructure is an ephemeral Docker container to host each flow run. Different worker types create different types of flow run infrastructure.

    Now that we\u2019ve reviewed the concepts of a work pool and worker, let\u2019s create them so that you can deploy your tutorial flow, and execute it later using the Prefect API.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#set-up-a-worker-and-work-pool","title":"Set up a worker and work pool","text":"

    For this tutorial you will create a Docker type work pool via the CLI.

    Using the Docker work pool type means that all work sent to this work pool will run within a dedicated Docker container using a Docker client available to the worker.

    Other work pool types

    There are work pool types for serverless computing environments such as AWS ECS, Azure Container Instances, Google Cloud Run, and Vertex AI. Kubernetes is also a popular work pool type.

    These options are expanded upon in various How-to Guides.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#create-a-work-pool","title":"Create a work pool","text":"

    In your terminal, run the following command to set up a Docker type work pool.

    prefect work-pool create --type docker my-docker-pool\n

    Let\u2019s confirm that the work pool was successfully created by running the following command in the same terminal. You should see your new my-docker-pool in the output list.

    prefect work-pool ls\n

    Finally, let\u2019s double check that you can see this work pool in your Prefect UI.

    Navigate to the Work Pools tab and verify that you see my-docker-pool listed.

    When you click into my-docker-pool you should see a red status icon signifying that this work pool is not ready to submit work.

    To get the work pool ready to submit flow runs, you need to start a worker.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#start-a-worker","title":"Start a worker","text":"

    Workers are a lightweight polling process that kick off scheduled flow runs on a certain type of infrastructure (such as Docker). To start a worker on your laptop, open a new terminal and confirm that your virtual environment has prefect installed.

    Run the following command in this new terminal to start the worker:

    prefect worker start --pool my-docker-pool\n

    You should see the worker start. It's now polling the Prefect API to request any scheduled flow runs it should pick up and then submit for execution. You\u2019ll see your new worker listed in the UI under the Workers tab of the Work Pools page with a recent last polled date.

    You should also be able to see a Ready status indicator on your work pool - progress!

    You will need to keep this terminal session active in order for the worker to continue to pick up jobs. Since you are running this worker locally, the worker will terminate if you close the terminal. Therefore, in a production setting this worker should run as a daemonized or managed process.

    Now that you\u2019ve set up your work pool and worker, we have what we need to kick off and execute flow runs of flows deployed to this work pool. Let's deploy your tutorial flow to my-docker-pool.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#create-the-deployment","title":"Create the deployment","text":"

    From our previous steps, we now have:

    1. A flow
    2. A work pool
    3. A worker

    Now it\u2019s time to put it all together. We're going to update our repo_info.py file to build a Docker image and update our deployment so our worker can execute it.

    The updates that you need to make to repo_info.py are:

    1. Change flow.serve to flow.deploy.
    2. Tell flow.deploy which work pool to deploy to.
    3. Tell flow.deploy the name to use for the Docker image it builds.

    Here's what the updated repo_info.py looks like:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-first-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=\"my-first-deployment-image:tutorial\",\n        push=False\n    )\n

    Why the push=False?

    For this tutorial, your Docker worker is running on your machine, so we don't need to push the image built by flow.deploy to a registry. When your worker is running on a remote machine, you will need to push the image to a registry that the worker can access.

    Remove the push=False argument, include your registry name, and ensure you've authenticated with the Docker CLI to push the image to a registry.

    Now that you've updated your script, you can run it to deploy your flow to the work pool:

    python repo_info.py\n

    Prefect will build a custom Docker image containing your workflow code that the worker can use to dynamically spawn Docker containers whenever this workflow needs to run.

    What Dockerfile?

    In this example, Prefect generates a Dockerfile for you that will build an image based off of one of Prefect's published images. The generated Dockerfile will copy the current directory into the Docker image and install any dependencies listed in a requirements.txt file.

    If you want to use a custom Dockerfile, you can specify the path to the Dockerfile using the DeploymentImage class:

    repo_info.py
    import httpx\nfrom prefect import flow\nfrom prefect.deployments import DeploymentImage\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-first-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        image=DeploymentImage(\n            name=\"my-first-deployment-image\",\n            tag=\"tutorial\",\n            dockerfile=\"Dockerfile\"\n        ),\n        push=False\n    )\n
    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#modify-the-deployment","title":"Modify the deployment","text":"

    If you need to make updates to your deployment, you can do so by modifying your script and rerunning it. You'll need to make one update to specify a value for job_variables to ensure your Docker worker can successfully execute scheduled runs for this flow. See the example below.

    The job_variables section allows you to fine-tune the infrastructure settings for a specific deployment. These values override default values in the specified work pool's base job template.

    When testing images locally without pushing them to a registry (to avoid potential errors like docker.errors.NotFound), it's recommended to include an image_pull_policy job_variable set to Never. However, for production workflows, always consider pushing images to a remote registry for more reliability and accessibility.

    Here's how you can easily set the image_pull_policy to be Never for this tutorial deployment without affecting the default value set on your work pool:

    repo_info.py
    import httpx\nfrom prefect import flow\n\n\n@flow(log_prints=True)\ndef get_repo_info(repo_name: str = \"PrefectHQ/prefect\"):\n    url = f\"https://api.github.com/repos/{repo_name}\"\n    response = httpx.get(url)\n    response.raise_for_status()\n    repo = response.json()\n    print(f\"{repo_name} repository statistics \ud83e\udd13:\")\n    print(f\"Stars \ud83c\udf20 : {repo['stargazers_count']}\")\n    print(f\"Forks \ud83c\udf74 : {repo['forks_count']}\")\n\n\nif __name__ == \"__main__\":\n    get_repo_info.deploy(\n        name=\"my-first-deployment\", \n        work_pool_name=\"my-docker-pool\", \n        job_variables={\"image_pull_policy\": \"Never\"},\n        image=\"my-first-deployment-image:tutorial\",\n        push=False\n    )\n

    To register this update to your deployment's parameters with Prefect's API, run:

    python repo_info.py\n

    Now everything is set for us to submit a flow-run to the work pool:

    prefect deployment run 'get_repo_info/my-deployment'\n

    Common Pitfall

    • Store and run your deploy scripts at the root of your repo, otherwise the built Docker file may be missing files that it needs to execute!

    Did you know?

    A Prefect flow can have more than one deployment. This can be useful if you want your flow to run in different execution environments or have multiple schedules.

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2},{"location":"tutorial/workers/#next-steps","title":"Next steps","text":"
    • Go deeper with deployments and learn about configuring deployments in YAML with prefect.yaml.
    • Concepts contain deep dives into Prefect components.
    • Guides provide step-by-step recipes for common Prefect operations including:
    • Deploying flows on Kubernetes
    • Deploying flows in Docker
    • Deploying flows on serverless infrastructure
    • Daemonizing workers

    Happy building!

    ","tags":["workers","orchestration","flow runs","deployments","schedules","triggers","tutorial"],"boost":2}]} \ No newline at end of file diff --git a/versions/unreleased/sitemap.xml b/versions/unreleased/sitemap.xml index 1ce778f58b..629435d4ad 100644 --- a/versions/unreleased/sitemap.xml +++ b/versions/unreleased/sitemap.xml @@ -2,1137 +2,1142 @@ https://docs.prefect.io/latest/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/faq/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/rest-api-reference/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/agent/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/artifacts/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/context/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/engine/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/events/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/exceptions/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/filesystems/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/flows/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/futures/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/infrastructure/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/logging/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/manifests/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/serializers/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/settings/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/software/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/states/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/task-runners/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/tasks/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/testing/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/variables/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/blocks/core/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/blocks/fields/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/blocks/kubernetes/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/blocks/notifications/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/blocks/system/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/blocks/webhook/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/agent/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/artifact/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/block/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/cloud-webhook/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/cloud/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/concurrency_limit/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/config/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/deploy/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/deployment/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/dev/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/flow/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/flow_run/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/kubernetes/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/profile/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/project/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/root/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/server/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/variable/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/work_pool/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/work_queue/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/cli/worker/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/client/base/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/client/cloud/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/client/orchestration/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/client/schemas/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/client/utilities/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/concurrency/asyncio/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/concurrency/common/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/concurrency/events/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/concurrency/services/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/concurrency/sync/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/deployments/base/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/deployments/deployments/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/deployments/runner/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/deployments/steps/core/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/deployments/steps/pull/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/deployments/steps/utility/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/input/actions/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/input/run_input/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/logging/configuration/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/logging/formatters/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/logging/handlers/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/logging/highlighters/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/logging/loggers/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/packaging/base/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/packaging/docker/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/packaging/file/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/packaging/serializers/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/runner/runner/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/runner/server/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/runner/storage/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/runner/utils/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/runtime/deployment/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/runtime/flow_run/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/runtime/task_run/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/annotations/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/asyncutils/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/callables/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/collections/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/compat/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/context/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/dispatch/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/dockerutils/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/filesystem/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/hashing/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/importtools/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/math/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/names/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/processutils/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/pydantic/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/render_swagger/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/services/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/slugify/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/templating/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/text/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/validation/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/utilities/visualization/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/workers/base/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/workers/block/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/workers/process/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/workers/server/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/prefect/workers/utilities/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/python/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/rest-api/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/admin/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/dependencies/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/deployments/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/flow_run_states/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/flow_runs/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/flows/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/run_history/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/saved_searches/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/server/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/task_run_states/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/api/task_runs/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/models/deployments/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/models/flow_run_states/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/models/flow_runs/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/models/flows/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/models/saved_searches/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/models/task_run_states/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/models/task_runs/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/orchestration/core_policy/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/orchestration/global_policy/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/orchestration/policies/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/orchestration/rules/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/schemas/actions/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/schemas/core/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/schemas/filters/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/schemas/responses/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/schemas/schedules/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/schemas/sorting/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/schemas/states/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/services/late_runs/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/services/loop_service/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/services/scheduler/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/utilities/database/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/utilities/schemas/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/api-ref/server/utilities/server/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/cloud-quickstart/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/connecting/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/incidents/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/rate-limits/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/workspaces/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/users/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/users/api-keys/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/users/audit-log/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/users/object-access-control-lists/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/users/roles/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/users/service-accounts/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/users/sso/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/cloud/users/teams/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/community/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/agents/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/artifacts/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/automations/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/blocks/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/deployments-block-based/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/deployments/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/events/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/filesystems/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/flows/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/infrastructure/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/results/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/schedules/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/states/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/storage/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/task-runners/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/tasks/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/concepts/work-pools/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/contributing/overview/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/contributing/style/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/contributing/versioning/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/getting-started/installation/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/getting-started/quickstart/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/ - 2024-01-08 + 2024-01-09 + daily + + + https://docs.prefect.io/latest/guides/automations/ + 2024-01-09 daily https://docs.prefect.io/latest/guides/big-data/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/ci-cd/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/creating-human-in-the-loop-workflows/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/dask-ray-task-runners/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/docker/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/global-concurrency-limits/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/host/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/logs/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/managed-execution/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/migration-guide/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/moving-data/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/prefect-deploy/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/runtime-context/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/settings/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/state-change-hooks/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/testing/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/troubleshooting/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/upgrade-guide-agents-to-workers/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/using-the-client/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/variables/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/webhooks/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/deployment/aci/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/deployment/daemonize/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/deployment/developing-a-new-worker-type/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/deployment/kubernetes/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/deployment/push-work-pools/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/deployment/serverless-workers/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/guides/deployment/storage-guide/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/integrations/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/integrations/contribute/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/integrations/usage/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/recipes/recipes/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/tutorial/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/tutorial/deployments/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/tutorial/flows/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/tutorial/tasks/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/tutorial/work-pools/ - 2024-01-08 + 2024-01-09 daily https://docs.prefect.io/latest/tutorial/workers/ - 2024-01-08 + 2024-01-09 daily \ No newline at end of file diff --git a/versions/unreleased/sitemap.xml.gz b/versions/unreleased/sitemap.xml.gz index ff868c3428bc833e1b5bef3e21a75fb2c7a1b4e6..bcd904600024797770d36018368ec3c8197fbc7f 100644 GIT binary patch delta 401 zcmV;C0dD@^4de|6ABzYGwX>a(2Ot4-ktjwix5po^zq}-G0>y`Xvkl)LhZx=OyWQn- zc?vqxC|!7B-|vDN@|&=$CwI?Jp_&DMAxsS`>OuFS#5P;!R9dL6c}P$5x@|F}QCgv` zhy#fdvKM7AYkP)JpP`#g8950En`VV>yEJE|T(ULArj&l`vAGI9aA z=}af+ol0G~mQ8~StotR=Rm>-v$K`X;z6j%4y&xLju(Q(EQk0_g3UL)TEQW1=^ZH=+ z!-(apMcIzHdE#>1^gOI$dQPSGWo}ah+~l&F?PyW5Vu{s>$YL!d(K>!8TDO?GbvLiZ zdDk5xn{U{VhOVtu0P7diY0Cr0L4DC`F z1b|=qGx?HCZA!Wt$f&Y~U*bSajZZ>!wvFp_DPTZez3d>3nnKIbKu=1$x$5Wg+A3#E vxjSiT+6P2Ps=30p?cKd%wjKPvVgd}ZA6WmF0d;-}6kq=V*|8CBH=qCjBrnA* delta 396 zcmV;70dxN34c`q1ABzYGDTle_1qp_&DM9ds{Bz_4{rrG@I6hx9Zrh89B_r4`zWIFKlD za8X8xw$1MP4Bc!W$VosL3M+Km^)4&rk^(6<_32v=cvbL$L-TspaNm48W~Gql2%VsJ zD#hAb9&anK?w9OQF`sB2m(NA}B8+GCf@pjf%t~A9JBrrJu2tNySc}c;gW0)%A(pQe zWsl$HiOX@*lbed^IhBHvxlIvplgnzhqeaPzB>*NOi?upK>-eE)-D2w2EwUQtUAJ9q zzG3Um-|d^ diff --git a/versions/unreleased/tutorial/deployments/index.html b/versions/unreleased/tutorial/deployments/index.html index 01ed688723..078419dff7 100644 --- a/versions/unreleased/tutorial/deployments/index.html +++ b/versions/unreleased/tutorial/deployments/index.html @@ -980,6 +980,8 @@ + + @@ -1236,6 +1238,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/tutorial/flows/index.html b/versions/unreleased/tutorial/flows/index.html index 62ce70c420..73b437efed 100644 --- a/versions/unreleased/tutorial/flows/index.html +++ b/versions/unreleased/tutorial/flows/index.html @@ -970,6 +970,8 @@ + + @@ -1226,6 +1228,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/tutorial/index.html b/versions/unreleased/tutorial/index.html index e3fe7b003d..66b392ed36 100644 --- a/versions/unreleased/tutorial/index.html +++ b/versions/unreleased/tutorial/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/tutorial/tasks/index.html b/versions/unreleased/tutorial/tasks/index.html index f9c38c51c9..2ef94def5e 100644 --- a/versions/unreleased/tutorial/tasks/index.html +++ b/versions/unreleased/tutorial/tasks/index.html @@ -957,6 +957,8 @@ + + @@ -1213,6 +1215,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/tutorial/work-pools/index.html b/versions/unreleased/tutorial/work-pools/index.html index 2e319b3467..15b67b5bf6 100644 --- a/versions/unreleased/tutorial/work-pools/index.html +++ b/versions/unreleased/tutorial/work-pools/index.html @@ -1018,6 +1018,8 @@ + + @@ -1274,6 +1276,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/tutorial/workers/index.html b/versions/unreleased/tutorial/workers/index.html index 834b653521..87eca39d32 100644 --- a/versions/unreleased/tutorial/workers/index.html +++ b/versions/unreleased/tutorial/workers/index.html @@ -1012,6 +1012,8 @@ + + @@ -1268,6 +1270,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • + + + + + + + Automations + + + + + + + + +
  • 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 z zxr?aM_YKV>UQL7V33dx~YI-8Ty6^MK=}pvlm9$6?O$7Tp5(>_Se?T4|3KJ0wOV~-X zZtb9%`2!jnh?8mjnIG3nLqLxrF}3?q-1Q?)KZ2{Q9D4G9^Utgp%eJ~nRZ~aJah7p5 z8B5YR{o2Zaajp6HmEje$u=a!zju&!Z9#b7Bx~64{qesvAoyoi*LMd{{mr_Lh&laHR zc%(}dW8pJdIp2GA8YO+km4GtoD!Lp(_Xp!B^3~0YO?FwE@feDZ6Znmv%dH8HeW93I zpzmI{JZcmg5RQ~dT&-KQ2!{zuNL3a+ajnoU;9+0k*-v-b_SeHcy)(UB<7XwIeao8+ zuTq=`iDG{_R3@QN<(o<#&RefZ1lMhkj>!n6ISY--34r+N|Bsjs#iCBa;6ONxi0jL()0=$%~U$GPK1wwUc3Efya8T zr!wScPC0I4LYBUX7qqg(QdDk2qRaKYcomRaLT2tP?~4@MC(G=orbKLcb+hIJ+6e55 zHPT>dihRy;+^VsWg!-|O^Ckta7ZJ+~Ca~cuV)RnC&HOi;B#?xvGzau{kN$L*QNeY^C_KY$E;f>g4C9o2ITx*%MDtp%Mok$sJniSRpTE49Qa7U zxo0hERrmH?OXX0D)am6`;$tmsOKR)M-c$^yv1$}SfMOdl{;gtM{B4$eDJTB?v5^IO zr@znSr>I{ooW<=MV(&?hd=kC?m&P{Y`G4fcJcca7JKLtvd07#q+z;~%=aBP$jU*XRt@dEB^7 z8%~%!?cIC`t*_E>fMax3`xoO-5g}&h$MTol&$Z=oq*Nb8QmRT6rKt~Sjzz}tr_Su& zP|>B9CeqS4iaL_ng<@BSv1z5<8@T33DV$-WaJYUv0+*7D`{MzOr3mC_=4%u>h-tm6 zrV)2|CV)zVVlmo#*=?tG`DH4ly?n#QQzZBXl#NHc?~aUtO4Y>gJtHKy{D`lJOS(uy zd*+fh<68mSM?il$9o;)RHN#jZvZFvT@Gbu%IZVeD0ruhO$?=-l=^(@7sF(|3(`z{? zDqU_fwKur8BueHSJ#JS@o$~(peuX5C>w&Z3&q=IkmgAP1hOj@68&z%?tb&-}}0Syn`2{$ZdH zg3M~7Gn;`$BI4?6)s7b^#+8y%kZD9H#XVY|7i3w->R$EC4)*iD@Rjmwh z@r~9N3<2X(XMJGA5Iv4FgD&&;B3wzOpTjb%_?pAi*WU9RdUi?(QzZ-GaMYaCZs8 zo#5^kEO>BtcM0x;-DdBz&p!7r+~+>?p})*bcfV5as#UA1bjr_N_jG(*3#*+oa~^<_ z*^%M^k8=WtNqcY!(RClVmDZ+U2oXU-OAOjXpr0Q(_D2ZN7*NOTqiX2*Ks1sq<)`7e z`E}OrRr1E^I(K0``#xO{95O^xY5dg5dOMy&Mm^R6B; z37Rw;C$YBU*_+ce1$9ERkfGP^N@qUTlN~8oj>1oALk=umJ5afI{0vHFlZviPIU<0P1;x@>{(u9$p6P9l28!hRbLdJd4%Z$9b zr0XzM{f?5Wi~Ha11j|WRcS_U2WHO3eR9$QQm1L4n7(za zIaY2>#fIdL$GS|o1_BSZJU}7>^X}CyWKj_cU1MxX(M|o1uZ;~^R|fQo)Jmg@@JfkP z<5!SQ#YCqY?tofd*QS&-wh8t??;qDKsRwY~JdXwCjSs5okb4S^Lp@ypa#1A;1ppVB zA7VzsS6(cCBMLigzZwRTY_?DCiylD_z+aV|W&6lvfO8_C`fGW#&vYXdMAmwl0aP*n zD0<_ALoYNGmLwh-J<5CPFC}!vx-SC)1QePR;*O|5(coMAmO~G0s9cnH0bIyITi9<- zr$k_n3<%zADEH6N4a}n#5-Z9#e*%>8pB9li812Z{6tem!dlZo~^NRI=qO>&80&e{r z#8wW3J^Qa#!#HraVxYfXm>?tgYfeo4^T>&IJrUGI&``SDt_Gga!{z}=ov`^;Z5h`-kgcwvH3umCkDBdSYXNz_ z${&F{!tTkp7=(6`L9Vo4V1>H?Vd*I#1z>o!BA`X4HF6ZrI+qdFIZj~#qj$NrSbOiv z`1c2TR1JKi0)4Fjrje+vG3X{o^HSv^zI1@G&Jod_a_&b=1KLj6txvI|)^5~xO2yP` z){v|`?~_dd6Uaz_u?hmj^l(z!$b9Jjd@s4nPG`U@6zuCsA1~NnvfJD#hG(tsvTX1` z&7NL!@ZUM)IYP?d4z)QF=>{e-w z980suY0Y`zK=oIOda)RK>Vi)h*1ezA2pi~!p;}}sf3vrty7zgTZwytTi-dXg?VDox z;Rp8!hBR9=b!iTVgE@Krr|I>m!4UV|l>SIU)@5GIUSTVrp_Y7h`;YWPs8xC$?}wis zj<@)}{#9L?NG6Uvfrg_$R_&+mUVh|t?oo(QQGaTk#;UPSzICeG^!;7<_Zk`7VXpe# z-Lw=y-dB_07gr~U7h)ksZZeX@^JFPLn=_k^^p?VKQL{8XxgE;ZT}If9_P|tQRj}tv z8Jl;R$_r}dqj$zc$KUt?#7`!kO_e@pHyBor8(RfsSyA1x;B}j|bp7YeWfT%lbu!Y3 zU57DOWzBG*0N|i}AsssU;|Kf46nvx8PA#JK7pO4u8d*9uig~XoG)HeT?2l*mb0_i1 z=ceL~On7cs;y1Kd^6YKmqHQA~5ol<%_n#4|s6yYOy(1^!+{+^t6dfj|4u?clfPBMA zZAc6s3i=RwMzg!Q54_v%9%()3q&d zj!MJEmFE!^p6$~1)WsKO|Q z&K_nwUurY^t-GNM&o7>18cfWz#Mz9w3T4a(5vm$Iw+G^?+D9FRfDr$CcOgr1X$6G%2pAn${5s8GDr_CX zeCH*iY*9c12J`q|U=oj3LVd@3f1SlJR|%5ED}1qZ>|mg=gTs{lw;nMTjKX%ATib&&GvS;fX$)q_7yC zwT;^;ofZuwXe0tAiTLQ8AKmQqP)YqqO%q`|de~6j+ig{da!;W&q{M@W(oXJ|yE8&a z%k`v@INZ84Z1yK&AS2fZHoZfJ^N5T_^<+bKr-2BshFWJ@83+8Tub9u3pD{u4aBls_ zjh|Oq)XIaDa_h_;W|SV~VYz1bs*$84MP z-_CTLXnKuWxkUFIhp%mAeU7No>e4xFXsv_>c8Ahl3>LZn+RbiK!9DsMPKl3=h`@8Q zljhK-6Gy7ei0nng+39MsG*h1<@LkuG7rvvB zP)vn#kphX=S~~sH@O@;!Wpbi%D4R($eM%k&zjM?LKEE~e^4$-E%U!yw;aZIDY?cRd z`s{3^iBmi6Y}!cgyNP}S^QtLklAN@5{;lJbKvCjAvz^9`cZv28-gjuYQ>kzRUI>>w zyE2#E6j}{dU@}~VO)oz-d|nUjWJw>nQ_&lyhcH>b;UxMjO;3sOo%_2j&Z~n*=oZ!= z*Kd~|Mq_B>zIi1MMZ$Yl(AKxU{0eFLbRqw0j{devldw>NNw<&QmNlYqR-?TXI?Tx@ zwKAl{wuil76}@`*G4gs9t=NU&RXZ}z- zBu4s<0i>I&<+C{z3P+#efB{`vjFQWKPTA>SG&-2fAGopZJzt4dS0doGXiA*WNMy%) zc8>}?^_%7da`F^hyyv9V4?-f(ctdN_Hrl^xQAuNNSTZ+_S)w&KV=kKpacO~iSotjn zM^cOHr%I13B6jCRJpRlx8oh_FGSMhj>}44G-gH1qG+Pb6KF()%H1zTSiqqsc_bZq< zkOGzoFi^1usBksINw^ovPvE&pI(GKA%-6SJiue~pZe(fI9^7Y zyxJ$GLYadQxcO33lyYu|T2hy-D5M!Enu&||H2a-?A}i`Fw7F=IWWUpwbw|mJ1R5}r zGH8~ag=T*hU^ClsclI8CFQ!j!?9%>BkpySfg1~y+`Y3P%zb6Ko=79of$)0Q zCy8oheIiZ~d|OLX(In%Dx9aH#gT?xr*{Y@nXCcPBPZgF~erX=M7!gJtmGnT>er>vA zqfu+!*8e*N6c|UGitK1S19Vb4Io zu--?--K*LR@QtA<7&p`2^}(Y)?i)ATVh-dpz#jbLj2P*-}3$Pw(XPk}~HR%0+S?xJuP=;Wd4<^$6 z&{;Lev07@dwy6Ge-fN4OJb;CPJzP1O@lf3GExs5S z667T8iQlxVPidlZv%`nw+_)colX5%YTf?CDF7RYBX>2+e&vuzQj9uI`5m7SVCFw;3 z;;F)+js8IRj|Ye0sOz623#G=3pdRmRy?53BgK72D|EFUV1#bHbM$hua-Vv?SmwpbH zwWHbZGSOiq$y90v6j~Aajms$8l9*6I3<_nB* za-%esx&pIW#J)ynH9&bk`oSxnu9I|kEw`k4O{Tx7FZf?owmk=Xtd-AP!k6)Gw!!?u z-D=Z&5p=rR|11*0)wp0c`S#7#Sj>3xtcXDE2*iqiPhItJIoI5o9yYZsjsPqUS zZ*ak7%Dhfc4DNOe%1@UV-m*(Zf~`j*W$5Pf0|tD0Q$!TQ*IZlN~{5h%}e^C zmHIXB!t);f$_o>!ob=2BCUrb zgG3lsNmOAhH=05wpVz~IY!KKV3`@ga_0AV!Xw=RbHfbOVcSXRtqE$$*pPv$qVRBKH zDvry*(#vz?BiY5ZnYd~v389gxdj2k}cv&N#g9Q@Wt@R1`u`DKKY}%4psyh6dyzaxY z#xH0V|K@6>4zlvv_4X*ZeN$PSw7H+IUo|Vo{wy~~6tuDE_{ev1znN_~m-Vaatd<;6 zy+RUlf1rQiLItscr0go<3Lnl|C{ct8mSspS6`I`bC*ixCen=~)aYO*pB)fxS}V zT3yZ&3qerE{%!}e^pNl$1NdTQ8QkE!Ey0veM7pk`Hj}ekxJ|5X4qgvH5oW3@)_d8G zZ)Q22y2mXsQ*T0p{F1<#0Tj+XwRI+%Df*n@6ReT28v*3ph9rI^dC(srOosrKLGxcL zhr}LDZlgm^__#YFlxYR1!@6UxA+NvbJY@C9Lqol-+Au!6q%X0J?H*KAm1|KtAW0F2 zxXq+dD;KL{IPtqD*ifM`;R9W;AZZDxy$q09Uvn|obqA9s5}?iDJnoy4UTbaJ;{A*b zFerldkQq3|ppcowd-7v!w^sKS_fUEXUbFG*c&nHSj3$0gB%8caGpbr{b~v!@ zF`d~hJmz*WqY@}Fvhknt&(E^|oi;VT*B=S=X6=L!e!DWwjzkO9bx0+l2j}g?D1r5g z?I=j*%%f(?$L>=)0ECw}?Iz5Nc4iGLtkcTpXS)E99v@bfU)+_5^{_e2LQ6%J9plrE zdAEZJ_KtZjxM1``1aq8^W{)`0tfx<^#jjl1XSF9cPRk23kCarpJmSZDp3R? zL5gR&6<&w_2C*0>T$dE1w#=!wH1Q9pOKfyUiWr$U?)*jmWqz>p+^!Cd%x)!&PJ*}p z#Z?=Dy(tEwYc%|QDY}oyJy8@~=ysH?l?W1^E)gT;Fv*wiy~S-0T{=e&KT{ncM@f2e z<%u6}mUQ#;uyH7~AIxo+V5zYdUtumwC`MFlzaw-X_dE1{6|ehL>*wvg#b^_~h#PIW zF#y9w_^U37eG zP^mWTedm7g5rYet$1Rz#)+CFCzrcD6RrM>QH`% z7*l{GsK;3sk7a8ef30K9dby~>95^gb+;!3zj3qDA-niSZdtBN!lFg#(zYkz1v*#JG zZTtEm@MF_Gp^-6_K5Dv1r-UVYI-$7BSde4css0KYwmZ_Cu2M`YTbn558w0dVEspH+ zd5pIe3$$hPMynD>-v&YwycPO(6$cqXdyB)XZC~qa29-T8Gy{uiC&J{PYSAF=0R&NF z?uYK4d?aya?UDb>7Swq0tA%*bw@T_`Ty-3t-_};^i?86KGxq3*{}}8%d6567%ltZ? z-kR)qtIl%EL-%a8!^!%eDv2An`S;JJyfJ`{7&`<1$90S*R|1njEe;~#aHV~taYw>$ zIc|S~39JeahOS&_FGJMRkB1IXH8vUl^s&V$GUS;gfUaUg~8!dDyS6(n@1;U zlM-KS-2CF)?*HNQB-%X9Ma_>l2e^!#*nxDw%P~b_`1y4HDj#VZ>+QQV&-&IVG{QcO(~%;IaS%l(O7^Aio-5{2Hg-}( zfTy4H`9~C4o;?}VX6D>|+|<=!O7?Z)U6gG@JUpdR45Qh&!V0}T(XX+lz{WDdK{8x0 z6jZy`v&ve~BffUp^F`^RKkq17>!D$6ALp)pl(Pp%zxo>zENxr;j_% zBgFZAcykE-H-mT|*kDPAQ<&+8iPaZsjIYuIzp7J0p!~+MI%Qk>I)KD&>iw8D$2fed z7C9DA`GRyU8o44E5MuVmA{|+rf2&W1elHB}F7DQ9K0~Els5e>7ZBp?g;$=@1?rL^I zrvgk`j!Nl6QNiK|U%9z@EtIF+^#`ly$*2K;(Tc@-iy$~p30UTlDyUks3G`8lKl9no z&V%^oD^)D&d1={YuwAJ*P-UhYDJGqAK<*cdf(-{k$Xoa-{E&=~XGf!u+tf0n1;5Q? zs0WA;G;O48<;KMKKx1jq+SguVWSVwJx&}TtAgF$HE?7TkmhkLAuu9|@i?#9I^ zu*d)x%G%>@TK=JNqMFtq^Y4HLustqcjl9cL5ZoFwwC*3;XcHV{f0B1g(utmur>)!( zwG-m%UR}r`QF!sG--7YB7waw2JZtzbhkL4=s<53+FS(0E!Zve~P<30h8m)f-oe`Bz zFBmXHsNeKohI1D>5?bG-7ED-2Ov>OoYR)+A9zP;kdJqrPoDKERYvBynwL13K3H0BR zJg!W%rg+39!L^Clp$J_MNj2%yWZ?5pGGdYvPX?U(9F@y{?;XiCSje%Z(`pbp&Gws7 zFl7h|ClikjM1w!yT*?I_&!0Ec=`JlDCGStPl~P<8dUqk8OF+&lxIcT`wrSM4ohOk% zP86eteMdPlu=16OTqhSjib(X)W{+0@~H92(}aG^xrbkN!`fe4 zs1!>j3|h1%jsF~OU#pWuFcc0(obIaBu#wv{9^GTH>93;#%p*e5zMUyN`lo>=lQYFR zvoXPT@1>H0nppd>71y9$B#K0_Luu>~J9!md4bU%6sP8&IU5-_`$xSI!Y;zO%(dD=< zAs#OQJDES+kID-WIJ_q^-#FKp`O#NgZb>VW+J8G@ZU6Q=47mRteO#|1;Z334@5ALo zBG|wvT*rrY?zQ(hvC~nhW~_^qXZ5{X$d`#q`L$>8o3+wb^G%Yy|Bz6?#cmf30Dt|K zDgf8!_s4lo=Z8+$IWDNepd8{PQ6Cx-ayFMlI2CdY#A>Np7@nk^%cE{Ig~_!`vry5b zIpvq|PD`V;OjZQEyY67ucF#_9{$#KHQEk1?d;`dTlOC}F@lahXHrf`>Vj`I1AX{Sc z+pdpG@!pDJbObmLW^lq-p9rft;2EiIBUZPr7lkuHvz0(veneGA3z6Li*j@Rugr;k-8VHKKgx?v z83?+Huv&QuKh?JBX(U!1pmGvK>#n{`%pzJJ4a6Bq@pGYl`WjCWRbMKKNL!BdGK6I`onkhfwlsG|Cw^MIyJd<#D>E4Uf(QOF z#mbP>J@hzWF0g5_(xQ@ljvl!~!0nwQ8DBuYQ9zuLW&f6DyK3^cZpz*L-Dv0hTB|#P z9y~$Yz9C0knq16jca6uxO-vA6h0KN1k>fJ#@*>mZV-Qc{!B6(1YQkxR?V48c>age*C$4`WzrkpF5a%o&ARn*$V^f+7o?@pZN(-~#q)0)h||p^lHI-f zrk2-ar=(f84}H~71bd)nOn_(9ISm)PB$f0jpLWK0*0M*gbR{k`g-jf|7nVxu0p&5X z%odZUJ3=O1=iqST8pU|6cCV2HD`QP1Et4(%oh7&~ zG6{EU;OG&~&<-s>DFiK5&*dbN`r&~fY2S6e=j!(5xj^ss5!WPg2+5JZCE1LNH})jE za=x(j#(ig|DS>(R2(Si<7OlOSnZ!K-pJb;0gmu8&gqv31ot)2!~2_9BtC(O9A18TK^dsQ zJ3Hai9YALn;+}KD8N-UkkO$eHW)E0N2Q@ zcTk1J)h4%|qa6WSbd8UTk@YV+FW`|rJw@>*O?l1%u(zr2p1&;LX^3b6#OT8_4+QM;jeQ- z;?G?+n!!cgXdo)D=al|SaUH+@oQ5nI505oZEXoE>#p?--bhTM`khAs=^ZH82{1O!~ zYF}@N`Y!YLOAzy+#`vK-o@{>JE6bl zET{1eAF3sDwauIiz#{JM$8ha6m932rgL;SFeQRpKS)j`=B7>Zka@wr9b>< zqX9o64j?0D(t|mLQ1trueiFBTc@x=iXyd`*+8F6qNkU}Apy z-gcg~R^!B~*Mrkr<92v;VM%+afY(i~>Lv@~IV!6SeL?*lC4fPmZwCa@ik_Y4ENYg& z)~^2r4jZ#i? zBdEv8P#RUhJ3i(E;~P@04?o8=J>-LtmR*-uWzg22bCUGjN4~-kDo%fH>5WK;KK~iD zc2sYOBFQU$4K9E`X zFW$HuW@h`|9(<5V`M??M^j^>X+8Ihu!k<`Q_!X86?UjWmlstv#;NETduHaJ2rH>0? z=yfwnNNto>ln9TUniUK>MS755F-71MUE+uDESG60iPrI?jjD8+-6ISg_#KOZXq{Hw zr)y1Jt?j8ua$Kh~m~}EwM0&&D=T~;8;VkY&gFOlfJmMDWg>a<`;Ur& z7Rr@k4=DKAJ40&Z%|zc3pb;?hfZAn&19YPqVhs=MD^6r9c3y9|vJctx*JG6{rP@l5 zCV@(=!g@ZJNCZ4?R5MvgaYPWMI(PFT^LI}asZ2ZfPAIS#Asu&yj3XW|0m+BlR`dE@ z%OD|)P}y^It1Zs*!IMT?=p7yC%}2UMmK@$gGX4E3wCzSd^B~vp%z&68pm*@TtHUi} z+rH1sgWh;Jb~Vr6ivo>?lKtNk;1z61Z1Wy~{yk5L;qrVy{Z8sdw~p8QlEES##&VM~lHga~k(heXH%I)Ak>GSy`CLrh@(aCRcDegl=^Q zt8Jzw-;u;PaHl)u@ZP0%=o_FDH0WhGm%L*eWr_;6wos;|dvLV)&QG@YHdA=3FRCtw z9uJxL%a`cIr=!Y-C36R)^kks#rwl*BtYHlPeY|R=hH$Nkmw=ZCrGg(6^i|uc7kFx; zJ7g+P#`J>3{bD-@h!!4_`Xatn%mWE_2EQ7D#4(K-OKou0F$~Oh^_PB@U*V^)k{|Kk zIY58G7(!Eh7FY~3LW|XFx!j}Ai{X1Fw`lMVD(NTmO0E(%1(3@@5SE_QMfz*mTz%Ha zlm$sr&^DB=6Y0(j%70YcV1wirIsheYNLu%tzbV%zh&I0&C0rCzn8miM0N^U;E zeCecIXav3=DEwe`AAF;_>Tk<7<6;`nWhG@$9waLiw<#YJR%6Do$y4a_?P5LuFT7C+ z7uuSntEGDQ;a{-f-)a6jr8`4O@>*P66%uCjvOXWea#E)1CGMF?pePJIR-EU6BctAI zwe`N(1Syhzcq?q40fo>H=kc&?goo+SoyAJ&4c0Ds9&|M;>tNRBORLpKmzI*zgq*|f zvk9Oya;1-+(`R>T<)T^8X++Zk-b1Rt>BaZG$34;g zf^-g$0a1JEN1g?nz@7(%g!|ETvVuFC~oqQ;T(cWo6#Ic=gU7pw@)S< z5+!5!G;-V?52vovEFkt%c)&kMj^c5fTEdGK*yaoA zQYMB!gmT$FG)Y-Z5l@H8JrOTBsbQe^+%8b>uJlM(R8-}WXK#06*M6?(xy5D$%ut$c zk}9!RedBe4((zam-wp=jL1Dg6_^MN9FJR%bAP>7rr79~L{S54E`I3Hl zK-eXqmZ?{%rnW$SP(uq>u+>L$KOhv^o&tL&(|A6-vvJe;n8@$ry_j3o{gYIyK}6}=ZV0K1 zAw=Q{)U-0c4+EAyHc8`sAd?e{n>@R0YcGxclz30B4Ck!a;`H%&u`$o1@yUDLRkPWm z-sSwn)6_Bb+XpFyLOFtKZLm|4jpu?Ukz@KMK`6L0iedL^rgY9sI}S4R2w1}f5%et* z9SCeix8#Ec2xi=WM0#N17h%#6&upGFoAX^9tB2hB7Be5UYl%z-)5Ba&7t8nX*)R8~ zw8~EHzrQ`j&y%#g#nk5S5a$=#N{PLHoRyyhsF%O>mcS);0u4Z+KW-t4JN`vUMxa+q zS-pC8>hqifv?Al5#&0b^ej=MaELANQnz!>(&?xdB^>Zf60riDB9^-q^qDsY%A_75i z?*>`5?SeP!o;hT|{Gh8SfwO2+YN?*q#58NRX*~9GV98owZd&-F_U^qflM-TD8#;6k z(y+;t*sr55)M4c9Rlc84`uIJ1M0u|)UZY(%r?W=`WeaIC)T**)1-IC*#MF=0TCBN& z00(|j-3B~%)gAaK|6RnWCMDBziU-$K99>$Y884I@f^NV9=DxHI7Yzp4K8M9zi6rpP zWR{6#1|ICDK2B3eX*-cBjW`Gm>6IFl>Q2NGNt0fP9;+FHN<>atou_#gL{}lF-{r;& zruM2s-LtbDV80$1*B2}Q{#NJqJ1M6h*FFC#85aK4UFX#@892VDvt_$wqnk za?^lzmw|D#idHN(MSt2~oRGIYh`Lp~%^y^s_;yDksK29}lbvbrrqb#@=DVm3Z`mYG zZa{uD?ZiiVB3{K3Pw@V_^YMx8YFsq@T>*Uum@CgGIuWak(oommo3P~zNG_luhUElp z4rt|>33GM%dbqCLJQfUn1vD)7NQeR-L;wN`z>rsE7w@TqYoDyD<#ePi^BE}CZFIOT z9a1%DlbLi{oH&1Z(an$YqNA7Z+vCU9qNBHjgj#tL^m{$rOr4>T#kYoL&!5icv=PRd zc&2aRY-OdOim?pkXF{@f1R+qSs>oB?S^;_FMLYA9XOYdB{$$v`!6)UYb8#=-;t z%Q086si)0iIUS+7N(KMP40w!IrwmBro`^@H>+^TgXN$LK%Xy(jb8~a_K({i256PR( z!DAijd2h2uk94a9G*3$1IOmk=F3%i6?mgY&v-NY+Q%1i`Gc&NIEJ{&glxX37 zL2yDPIDE-ZB**jT2=zXX)#OiW4W)(Z)rMD1xHK}f?3Y3*Au-tbzo&Pd`H#b{P?xQD zy4rS1);ga1v^bEU*H5o`SIbLN=iVQ96>HABDK!(tBmta{jAaVFZq;{(|4^8JInk&h z0PBNbSSpe7f5AbGzs(sc>&czboLO>$M{NT{Me z8)vUwseaiMtv(CngUqz|UDIM}ss@Y-Z zyFnq30r*(Log_w=_v^Sjz;?p~@;#n?$)GC?V>x*mmIk!lIe~;LUAt%%u!v*J4$`EB z6gj3uYMUSXlfLx*Wq*Weszd}B7B?ucFy2wD0akvJp4-)8NR{G#6ws2q7wx*dPsG@l3%~*Vj1)I*|yvQ(98fpsH{@ECBJV#Ih0Oh3tpj!>*1lEO5{N>=w-z;#^RxqC>K+>#~sVq1c0T zGxFS4v)G_i@Apx8i*IqSFw3Vv&-HYITaY1zl;dp_@euiygp(e;0nV+#htu#O`nfTv zKCcHPM6_ptKagK=4Q3yRPzT&in%WkuQxPr5vWPR6yeOa?bUXL7dRJe>(PYjI{nvfZ zf=S)aK4Crv*Ne2$>eHb+ua>#w3Y^iZRUjr%tH6J*6YVk0lq6rwMuLA&{jSRc;JH~R z&Vlp1Kr|w8|4u)ff93w;imAd2ka70hWi)QO*3yJ{4tUcY=2U$=qtqC7E9rRam!%$DTf;z39J3OJu433W7XF z(Y`%is@(f7{i&IXCD=NTtmjqfZNt826CKR1E&m$ry(d)sS-?gWkp{&0W`DNH7$yO^ z2MhW2#X+R0B|#sxC=Q)&Vh0B{r*#O1Ot1ns-}+! z@)C;{afD2F57Ik!$&GI>-4Tvk|4zrxJ}g6`N;yP_1V8;WT;i$68m%+2@AT*37`Fel zc;-*WTm42O)%u8<)fs2!r;M$TqgCddw9JUrv}W;`wSpBoWus1?@3l56U#mC|7lF-&gOQ}SyvZD08+$gu%Ux=R;q>Xlg|1UJ`G2|eSUEI6_ilQKl)@8avd(IjE}T{4{p3;p z#?O^{_S#%>!-UZ};tbFF@P9gE&uSpFc zzvMtm-ICw?56AIm-HYr}fpw;RqC-8F4Ay?jnG0QO0Z>Pd^pp1#1()8z)r8zQnf?fo zN1%FSL9#Vk#OPpg#~a22%KSG0qsSiKwfud#cwz*thiqSd&6cT^am7v_e%WW<=Vp+B zr18D~zS=5u4-caE=`@=^i!*>B9(ISg{q=GFM_HaA<#YL@9(dws>M{BdkDrTQP;+8Q zC5dk#1c4b_et*}U2?3Twc|l?AXR?qZ56H`FDZLm*5RLPlFvNKrFAR9b=P=Un4pv9M z)kEBhfLD)ZE!r5hWy7PL2o(ay5#f22TwhgW;GcI_`RV_f5aFM zi8a6t&E10->A80;7b6h4u2R_VFGnbNo3q3aZ4tNc$PBrbzib$uY6_0AdQ9Qqc{#+*2tl9z1=i&W$*=t0}!;Pkw8*#;c#@P zmN|6xAK)3Q44qp>_3zf)_XxOv4uRU(jGmzs2d($BOM~BbmUB+=&(ZAxt&9 zx7w{N-^ke#a`IJhJ}9UYv4h(eREf8*4!8x9f-ZNSp=kvIjY*bJFw?Ov#Il6fwjeHW zB+THd65%4z9@o3K25#&NnOUEMHqg80#U}x3&isNfuwA$9AqWh6vf_Hmy9iA^I>N?a z&$dU{gU93s;rAH0-pZcsLnLbvY%HM;&GKd-KAAG6IA;9hkSywrcBMad2*l;IQkGAg z?Z;9p&KaAyRT6{W@4gEx;;|HUL?bX|m~$e3sgt)&P8%(J5HB1J3)B@cfnq*`h`XJy z{*trDI01i~9QntYaD8~W&+%tJ;l#Gd)da~PI^gNcZ0Sck@BJiIlT~|C6+?-6IEcA;4TG>}sN44RHSB|C^ zy5L-40H_aIyuE&WuAuz!$G@H|IfkEG+^vFa-2JzOc`=8$TSSI!<$q|a0SGvd#xC=9 zFdHfHl@&2;ipOBPmxp;qKs=n`H~RJRsJWtv9=0~qI@2X5Q5!B_gKij0p;%cx6I^}`5&LWvx4dxIxQan*S8QdZ7}?C`sUXx7Y&|RgJ~xm%3=68f#q>9AKY9MB1eph zsrBJOI*jIe2CTb3qS-9M{ZHshE*d2tQ_Z5S$ESUV$9=XR26%aVFjm<3WFVktv)b%(p{?szNajTq3VeqGymfJ~N zzRtFAD}_r>yGiRD3*eq4^}O%%8=68ay|1$*@T8l{i#Ij29i7p6t!d3UqlwB!>(nzmiKm`q6rV@1K4;oW-g=Km8)$U3Z>H@o>D zf=aqx+!q({9ulDF=zesfLxv#ZGHM zah2!Z-x;M;-vr{obU)S){4G8Z)BHlzUp>q>Yg7-_&^~R<1k(F_lm1}k7JUSKfiNfu zVwvI`BU$!q_2Xw2h{fh4op8W+@e`n87tVWN|G2q=G{-(Zv4M=EN;U#9IjAoh3UzNj zxn1nxig@C<=0HZ+MUCr1#u;=VAtJCHv1bGn5)bRNg5Mz4^`pi9Xm{Uw<>k#us)vVJ zpwIGI!$05yH^@5?i&pM-CNGaI-j3U4nYqq-$$Zma4FE@xJKeoU^C2^wb(CGNX}h(N zA(qK_Fi3wY_ddl>_jCLp3fMQh+0(Ky&wU#sar$T{#4qX-WQRyvfRoH~@Gf}Xig!{V zmf`JZ30rvjf^3lM2x|Jom`8~dF?Q-k>m|HJL~>IfIRgfY=}!_}8{?oV!6%VPHd=DS zz4R=1;z_sNm>qe(6zNO8jF|plXjG!h-H|kl3cq;3n_~)I22?BWT^`rnd&H<3qTUl0 z0u<(F^2a!X=IWo9K;A6Pq}@x^*Vdt4Nm5br;bOOT90t|CNWvWY6e48r#i>6b#LNMR zW-zVdXSBl~WF)R02u*neRoQe})5Z`sn~8E5>B4-?VjOlpsgfT#2EgfQCmoz^2p{TP zdnx0PR?vJ5`d6U-NAE~Ucckzc}J3j!G+L+oA^KN5m%HctFWQ5Dri(Iq&9np#f0h( z!mUIE%D!~ly@4u2>N?EEk?~5A1YOng9p7r0xTo*6HrS%?s1Q-$|!bz~M2f>~AI?k-x`)`nSwXc-qxADjobeowFzhaNvzA@90m zDKfAeAefwqix>LQuOHjK?GtJEY)%K&e1fm=Tl4W@6E2N0S!vzZY1UZ=S2$)05DDuA zoM6C^$?(#_L%B8s9b=O}VeFDEo9e=T;tKQS%6&~ZE+mTx<<}@|vt9GXWx1~rxFxgy?6#aDBEZ)2y=d*E96V=ECRYS=OP`(ww${=O?IuA-8@FD5NJ*Nf z3Y$tsND7eby+e9xQ?eaA-=_KtEEGY1gM-5=Bwi$8*yz!4%lrdm1q8mQz~qo^bN~bL zX6jNMx~}D&E)UEyvy>AW&i1AAxR50a*OVp{wJyDnQ!@l0!7-JnEaQ>Nzud? zd&xL^xrYka1-SNQAQfGh|D%Hj%eS=A&bxjiib5^PX!HX<3`oZ2G$O2YOz{&887Gzp z%)oH|m_Q7aIGE%dBI}iwsJe^6ucdtSLc*0KF>m43u9lG$?3$?lI|A- zIlw@Yu_B{(j?f1Z$O1!gKZm;zA~~%M*K@-u(xRXOLDycOC=@f7cd3A;<^ZCEx(%7< z|Iqc8QB{Ro`?nwn2+}D?r=)Z@(w)*scXxL;(%s$Nozl`E-Q5lE+Mc85{eO7A?%^1? z_qx}-=e*`Me{-2vYya1P6;`5~9x~W-sq`bDCHFY|DI_LFdn5}A*yTj)EIL&8O|nu# z(FvnxqT)x^gbYW2XY#f(+H0uC+hD|#>}B)5cWD-9X*c6~nEaMDyGlaBmh?OsN;5#E zUfMh=GXJhDvI$Fx{+vG8wlvh`<=%Ms)2SmVXR~?RfyU@=FT3>ef4jqV9*#Fhk=?^y zcu5R~@J-cB*WqTZ;MQvgE(_IYwX)8-wjbrY+yT`kHZ`gpj9r$^4+eR)<5k=E9^t zJ|cG^LagC-8Npr_#QdC*EAAV4FUz*_gTy9_!@8Qbsv{v^Aq2I`E7Uc1Wt6~H_!+x{ zSKG&F2SijV(MGqvBb8NjcN2-2ND*d&?oB{cU^~BNze!;8Ow z&2&DF+KXu@pSgbL5#+@Nr(9~)6l&e?UDm(gFmIg-lwk6JZ>GmV;%e`=P*WN;KQlOH zfo_TA;hbeAMuB93%Gp{iAXof($K(ON*nk`S*x_79yhT%!Xt4^!7Tc%s^Q&_Dpj8oU zio{D8Zi`A-$WRD_P8>MXI)+temQq@L_+_r1fX~r|zI46yzb=}2exiLCo zr++*sE)cIYI>X%JOU7pY>a7@r-gqeB+fe|P)*7C%@lfUg-r6><^ld}U4MzDNBS)uk zhh~OeNaVkefO1yKZmS^}WNDBX-}NR2N0*!R0XeTn2KBgT=&N4iF69k=-5R=3?8J?e$gZ%lNgOA*Cq4nB8yDclSvVi@Yp;&KO#&B4 zBw6Dr70=s${wWFciPciDU$5u5YF|6Se6YSjG`y5r7JGU;a>F)8SMDv}i*^U1dP3jz z5vpo!Co}+wimx=Wcg*=l&RkVwJ%17*5}9Qa$>dDMebpfA62yBRL0*SfUC$Yj_2*lQO*X(dJnUjf zo7R>hZXY$lE5W>qCV~i{#LF=Y@+1sD+AcdgE8l+-2-Ecjzu)iw8otM0hXQXA$>!zu z9%sJ$b16%A2obot_~wf4J41K7)&dR+5)IjE_d-_^a3ylB^u;G!UdC$-M)QNp>Yfbo zDYytKEmp*s5K>7aT*auO|7YXDpMbvmMl4Yhjz@Tzr!j}*OnY1zPxi0ZX$THt%Xkbr zF%xEr#_cVXRLu4X_AHLqn6tl`eNFs|n>ubF00$!HJk+ilvW@7z)bFnm?&n7_LksfG zPJg<;lz#XYT|(#&EL|hYeXZ=f%&7_Si`LSJz*{jOCe)sIm`C6tCt{DypHIfmb!D(` zW7a?=+o@U;Iz2E<*A^givS8$zdV@cOgbpDrk0 ziXt_yGIU^ao*GrKetVtcLZo2p8(xgilu5GL>|_F3ZLoIVB*N_AG;>4Zc=i6RH3;ML zAh?Id;?WVTeKdeI0Y70$lnBW1x|r4r9QdU!)h+i%#!o-RQ+L8y49Q2LN6z;uJ2?r| z0n02V<3*1*=xvJjf58$V#24!q`~mDB@vL~)%22CJ^PX0Ln3Kh3Aa_U};K<^o4h4y_ zkn)w8CPqX{iMHe9!q7reuDAF?ZPQI5}ThrMVw<+B5adyv>Z72 zc*nOr+ZfNX0}Zr@J(G}XSWT|DzdHu zJvPdrl~aak!Zc77h1!FmE~7pqK{5pn?i|3cJy-te!1ooSi~p~rfe0~00M7U0_D$WO zoAKN_;8o`+-4^|uh;=6M0;yI;5yw8z2Sem+eP=d3Hy`9BGHARE=j}(&W501@rfC|X z2qjajAnI~31kbL_g1}ok0EAVK*@9>8k#CNd4d&(<_Ma@+eei*q0qF+4p@gfkmYScR zow;v;7a{=ZR+zRKhX=@?1UoaYtAN^3HlzJ*OszC93~oHp>G9A!6s-+se(A;L(YB|- zHm$w-ZS5A1YuOS4}xhlJ}w z<5-OqP6Xf&NeRFXPyMHZL(4s&?_-TF;ELm?_!RKFv%)E{%TS7sVCkgcC1{O0A>;?Ho6ik?W{_mUzaOmuF@`6xv%druHEY~A$>GE zUVLjU?HLF}0{cQR-EPaWVwk~@S_U03*_uQ$j~@EFAc>Go;=7mo^G?rTtdg$f@O{1a|- z-xkGH8ep=)ZH_=8MOydw!tlkCkRPuCARt%n=vL?IwJ3`ta5E*!M2-4FnDlmjfc~4p zed`NUP$np-fDTG)m%>{FYx#7^enX%~;E97#_=y0|bZe0?e4e*Zwpvhf-<=ZwGTWYG2g z?phHKmo%w9yL)3@UWIiP?J2^0;eMwT zwNQYCw;Z^CLk)Pei+_|bI(V#ZIxc%)&nRwP8{vI>I?BX(h7JcHU#)xD!*Lvr>6x8z zCnMHdgF1Dh1_I3$1qw2MU+%#mXb8pM0&2kMg#(w|R$Fb4YC7>>XI?4{NE1nN=>ZrS z&d1C7yZ(99%i}Rjux(Gk_;*y;QjPO5x6-FKv(Z%s5oZA6>FRvIwmW&@S3;5tKhNQ` zM+TY_{s6*%^1~RiKxm^&3qwlZZxShE$#ctDwg(k&g-b%hGgxz=3SxTi(A|^Lqe(}V-oCEH4BLU8^SV@e& z+ouCS?)aWYHt&me<*`PHZim`Qi6){ zI!N<6LkFw>ZWCCZH_^;5*nkzH9?g@n^7%$6wXkZ4b$8NZy6#0HFj10;(0Coo9vmC%x=Kb#$;7;Ch)e zp8E{zN=qT`TIFXHTOnD<{}-P&L+b_jYWxpI#(PJOgKv(nqznAnh^+s7FHpqEvl5JijKT|K((k z=Y9hHfN+xxckEd<3ZMcXw~Imu#w6r-WS4#1&<}UxOYah&s{I)3q)c&}J^CX>7D5H) z2?Tlpr>KO;ht6=LPuOv1*UH$$i`C(^@l&L7SPFK<#{pwdvGi5bL$@-WD=!M7q(&41 zkUG%F#HHjfchk}kgqTn~?irBkV?7^W$sT_rpie#&OUj)2MoMPd9g1rXa9jDb7$B-5 zX}Ugt*rs$yxw`g!l~EnAx#dnxQ|Mta#ek1@&d;C$O?*5`H@Z zjsIPc9zKr^9S^2NHsh4e@@CHaZnU40Ill*cC=?ppE149_)kO_qZs$gR=qrY~LyyjL zL2cBDo?Wve2x5;@wdTCRtPxM3p5}Lkyw-uv!q)$4)lSK4)fdPCUfKnMM=|ey=vJ{7 zdiqKS>GV{8P2F+&xk4A3wK_#hB`l488*=4ThNn?y_creSyiYe|Rin@s%j`OsQK}6Y z?A{mR8I0KV2w9+U?dfvAsDRz?XFXx^;p9Jnpsl;7S!9So%OVqNyGBXp_!r&ELeOf_{4)^>^M3@XPSZ zLHK+=&B|qp`3&WcGHA*7Bww#zD6x)H_{ z3#+J(=3GH^$EqPg8TI_^4H1xC2Py--7bG*BZUe7QrFS)@%xO$lkiZ=~+g!NP@gg=Z znOgcm7GDFnC-=@BB2Chc>j6XVQKRALQR%Lxz0xGzMezt<*5A%a0Q+}|2>D}gW?zpg zsWi%3Z-QmDI+t)}BX3)X6q?WHQX8H;F3J^`5W~=xzChzYVs<^^(`}eTLMZaqs4I>A z(U~8!Amp z3(Z+9V*Z3Z*Dg!!-GD<322F)nyi5gNZBMV37F9qknBr#6k9I+!RGN6P|DvE04@^ks zv_0PJnC<_j_(=bjx1`y{>8aN2oXi3p@EU`^5FxHcfa2W<9H~Gzwsvy~hw3j2;AzS16*Rx3W?|9Y zP;2EHVRIeKZSW32c!8sw+~U!Rtp5;E?Xxtz|H;15UUORG6-saY$xOwoafldq%XiI9 z383^{E2d;d;kf0KH*8ojg|E#fSdEX&3%k{RD2=b`^lnE0F9^mf)jRq9OqqSy35q~x z4~K_`8Wd9exID1J^1NM^SGxdC#aIXGIP~H>4O6gJ5R^`XA^)w620r6=nza{bu=f_e zlNGFMnzt4jg;wjm8kn7%dy_fIZ-Q9wo|kQ71$s>@E_rLu!weI1X`Xhx3_TmHb37th zcNGy?CdryCQr`(f8zt^W;pPvroy{{mqMzjbYWci~X|nR*QJnA4L6B)?6v^pq6X>V% zGY#+!&fYE@u`FWy$@V(FQ*$Q_m#+}y)t(n^GWd$e&x@@$Z0E4YHMadF*sF1;fXjRn z`EF}0jq+E&>1-)UBhFV|H=a}LlgNBngEXDa*Ly$%_JLMuaY*wb+HG?J-c?20t<_n- zD9RGWveHrsj64t}L9t|3bu(|L4~?>2Z1jl4@_*x3tOB6C<}J>??2fPc05)5iWQ64= zAgEK0+`Se{8kTf(er4Uv@3#cRueXXr|GGfX@4)%u?od2YdzklH&>kQ4_9#eu2t<3b@sEmz(fk5pa2GB8$dLkTJ$kD=ZUd+@9abZ zPWWN=Y27Emci(wT>0Lx%PRiIv^*4LpSa}n?q5xUfT@V`%AYbM3%wlM~WKjcspllt# z0jHV3he`7dw&+U0rpI7E&MK_6Qw8HI4hTfQTpWMFbmfPNwZ`79Dm56nDCb_$OXIi8 zL%mU*F1Mp<>W?|zh=2@V3ft|IM(riJd{nR2 zlMYD;=aA|}r~3{%72$J2n*K=q7uv(2txf5j35;Q`90q&~z_Nn5_55%%HN;E$R-9E; zJFBz)65QB%4opI&C$0|5bpn7MBpoWztMd^gP+E^P}NrN z)9>sXMt?dD2@xtf_9o!PeE5|Og&p94FIR+o;Q{QILS}3@V%;=+1V%loxXV-M)T#=x z%~r=C@2>Sd@uGXL?swclZrix)c4l+cKsw%U4yV5*8RAMbv4;@%Qy7g)CBb$_Q!se! zd7Ed7wYn8+&A(uV*-E@nL0>D8>NJ(lX*hS$J*v{ zE6#s*K6NVzcAMt-TU-$oI~*K2B)|2j7nFXtX*^-Kywheu<{p-4XsU1Tz2Dr^KO*9O zpxq%&B-(8c(HFQ1?zn;`WMHt?XVz|i`9qOvS8Zh`4Kt<{ zZCD2lK_g)Nr#%IwSr<$=e7I?)J8-e6!G~32Z>z!#w*tKy|4X*vU#Kx8YCQ07Iov45 ztpw3&2tQDHac3@jUE%~Xt2BMj`>g{GhGuR8OmMlw-V^U?Z0mD(;ds|KJza2F$R2`cUE_b*6LKqiS(2 z_lBeaC+ls2X(pUfR&Vutah4R?U?&`9PJn@kz^S!fY1a*`+hQAam2!0>?>EltT=0Y( zRAb3qhpwq*bA%lhD=3Uzvp;DEcR{I8)csV7+}-&AUiQP>%=Cp?3KSl~A+r6UX3X9q}QGvKdYGO4w!NQBy|nN;KdxNBo@pa|tejW?JG0Bt`q=0If&YY8x2JPq0X<&% z27OB3Btzmfdl0u_{lka2$>44sbNC!gTJjOqd|6b04BLc}rpOq^>ItYexlO?7Y?jOo z_M=|C&3?<59CxHkN^mq7O++BA=mdY2s21k|oxU6KNu1bbgPlLu8UIUwV;f~bsD1j{ z9bK-C@TEb>p$uW~hfc1&<#mgv0r}s=Y#u`~B#d}@yx!n$qIgJn4`zruqlhIhD8||Q z<)dzCyL*&$Zg$&f%6>C%&u;(=s1vp2^f-^D1L<@}^la#V+haW%w4Cev%~50!8ds(7QTouHDBwal55 zauC3iWX{W`aH%;g7HQr|=}d&}a*h|3mY~rpNyMroifo7JzVr*q7EU-P%fcNxJp%cJ zB^ldzxd>b(+`A2=Wvy=Hcr~E;EXdA$3_FE8=xU6FZ>B5fhaVI?iNzZ~2tdbNtdFMW zfg~S&4;FP$dbcED*Wg<2i+ei9Ww)hv7zSzL?Eu>OJ7NOT3f4^l6%OkQ!5}LzzQc!i z*TyNKFndO~Y4?|wzbF9M$|ApX=`$>?SR*1=Y3ZBn3PIRH76_F_Rs-bBq<0WhwUB4p zB|$sxmpHw7Jc?R1br=Jod8fc%KFa5NgI@XT4zPrLiDar!vBGM@XZyX+_J0Xta8oHebDBzTDv8t>CDL!(z;XohTvy> zr~*#Mt#z#%hajkmO%Z^pZ zET~f}3^P}MKkFt1Og%-tp0DdPSF&8fTmaS+0G4rs4lK{+xB|T4r=F$U#GbIBjK_V! zs`Xuc!g#JBHz$&DA2Lxr4o8i%AWL(EU#HOhmUyfeH;h!i7TpP;_7p*v>rwx$g;M!I zH4|8j6A^fPMc zaA$vlj&M(Iz%z>%yIym z^i5`umj@QRSV5Jq$apsw?csD#Myesfa}u=5B}CH>!gd{1vgW+_eWn)p$%N;LT!7m5 znT5slbsKdmua>k4Jyy3i=JzDiq0;mwU_NK})u|+)U6$;lgkcwTp}?Jev!4 z)_xX`JkdZsuMWC6eL|%=fNCn+V*%N(5ZvnO-dOWIoHimyLJ=?M$Un zrJq4$@EF(CPpMQL%?7eF(px*=lYiy!CSWwi94z`YZ(7j zDK2MWM?jTpq^8bE)h3u44}T6K*fBN7W?8~7_!g?G15*YgV0fg5P@@R7gIevkA@6=G-tNw?E#ZaQ`swl9c`Jk;Fr|dppnbJH;Z>IAAaElt_hx{V5jRz zyKgd!Wid;6ePe^U%G(X>63f*S6I)Xcmz$)aVbxw^SXf#~S1C(OFz>Tqcnv~}Ww;SW ze@|evx!X(3;jgIA>jmPHW8f#|k^>(_i9xdldlg+_4l%zanm7wMECT9aW1!uzD3${w z#H3GBS$A;}aAucq;&PehtThJdeFY$809b@02%sa{Wh!yRP~)f1-{@iwYlWv?Z-xa7 zt!LRVAh$oxQ%kC9^27>unpPAE-(T$J0Mpqus?H$}Jk-`9frc5d3!-OHKh)qsS=HkZ1n{cX|Mf~kB-!w9$7 zcxs;&X-SJ_QP?g#cbry75n?Kz#dNw)5ifo({%IEH=zqvUwGB4Jgqd8BNR+WUXglRC zYAr(Iv4`21QYeYr1pj`PWFkZbX9V6{GD7X#0DZRem0JVKTLDYqkmTZ9pf956L}L55 z1X&CFgMSDECqe*>Xx@x`yg^>Z;I;wo&~PSQD(K8>T{Bm}n;h4&U#1{~HCbGw9;jR> zqq*vPj)KV-4Zs4F|D0(mKVSnC31mf=iNt6!(^YTpLbv5pz=zd~abn>{89?8Czle4^ zPV=%9ODxs{YZwEa7%=k_F%p?jJHTkiPD8YD=y05Nbv6{-kLm4!<6Mznwp2eIhILwmFb%t3pD$S`HX|0trN;cp@cu&aq0!*j1z?g< zx?IGWHUSA&qxm8#mm*Q_kK2lI%HbvZ^-=tZjiSg0lr|T zUKd%gE^lT4;Zt2}h5yTJgAkA19wHa0htu)ubzBedh^8Px796!cf8YQJl?w^JY2Q&> zwoP!brPn`zMcN!5)h4nU>cgP85YPac;4JQU>jPahrEyY-L0)f4!8P-K3Sm&lC00}9 z{XB(&p8!UlD3B*uLf~qN_{Eyg1R-*{~n?JoB8d1c%2j`)-R|%Z=u51G8pwQSj$pBXt3H}ta`pq9y`|l>HwO~;vWvQ z1fURbvzZaNrA94Ul1^GqX$+B{AA!wR1;5?jp{PZC>saSFn4{gNsqgZ2neM|-mN_YW z&39F;t=ie%&L$6N%OtV>+j{rD`_FTgfFWGV-R(E3USSbemI*&lq)@9b)d_z)G9qoa zC+2#40^0&F4!+s{A!nm6#sY`K`GVN~cs8NlVkLk2SD~!rH#{eHDz4WzlV?r4$lF7X ztImoek-7t1!T@Mh(z5mZyU{e%z^-n|BVJa%i7xaOzaNxpgZRb@xDrCX6jsG)$MtN> z)y>au56h=K(YH^8Wc&Y~>KYM%ccH!86NLf()A&u$jD?<`01QohJdIj8vtF6Fg+q)c zVkIF^=S$`78Vl-+C1ktZUr=sBdzwl+r!<&l`NQ%Q0EnEqWGR~W;QNpuvvb+W{{_0* z1pGIW_E&pa0MzB3Q0gPzOw+|hR7@gT9S#Dz{#Q7>%pZ+1;5{E#y#a*4@DG|S^`^1O z$(>O(5rBl3JFdf*?{xls_}D?<8Cz^;d)Q?SK#2XgV42jY1w?XKpGriP-Qkg6P^Ow` zCpnQQWwGVzO#y&l7!MfrbK^Fd_IiU+>~=K}_E?_~LJk`YCCF5$pAJg$7PH}2FI`dl zDz&!$2g^e=`;*Vq#Sw$sP(EEv8!VQpLv+2r^2dmuNpNO(H_&Cx`wCHqO}_yccS2^i!3S=j28tLwJ+6XO0F&Tp ziqXYpvz4i=m6V;q;(9|SS_%%#1-|`NsPne1DzC2%;ot-Q10VNmaO*A#t0?K;RI3*Z zR6T9H;iky`pP?Tb+*=qls9X~Fe-O<2oruBty|RE|m2AWTuk+jM3B9jLM<6>!fwwZ)qHnQXyI|e~{e6pnNrE zvz|NA+7Bjc-gCOc5OwHr2klfsBpeFX9e~S=Aqh3T?L(+H&q!SRr2O4q+MS#eQ`+GJ1A^u_iIm2K7&odki&H9owRBJSupi{wZMxj)4BE>FgZBjh8>@J%T1W0SB7vOwdX%)o2|}Yw^nG=HasAi1C?2 zKtjao;B>DhrM7rjCleFd!*fB0_Q8Muaz4Ok^Epf)nlnA<`#h00vc}&AXAsJt!`KcS zMqGBgMAZgkjM>jA zI<728pg{5AZ$*6+?0k1!K`i$W6ak>)e_4^<0JtHM}|_#n(4wJA-O z@7|U9wI80$UY&NlA`b>xwy{a76*=#&An>0Arfpw6sG;^~BE{bQeR{lZ_@KmQFOL`r zF+N$RY5~ij{h3~WI4)SzXvT1iGOPc37j8r;lM< zy@a%uD`#LU);Rkim5Xb?okl-UbY4hEQArQLN6NLiC>vAGM7@a4-opvG5W{%Z6uPP2uCws*_dR< zEpmp>fNM;4Bp!i`fK?UtQ;(1=+`yX{!T&u;MDNUDS1)>!V6TxllHK*;>>36yy#zod zqa(y`IZLC{`Rop45x`ZK0^s9WRB9j?0yRE?tyVtjo2Zn7o zi}_j3M@%!U&|&X`G1gv9)Dz26SDt(GAW z@n8i+2gYC_enYml7xcbL65i(v@#AaTb&Z3iDvkbXB8A%%_bG@z$e zy~>M-RTbuqqBE7vnEnp`BO`@0TU%AL`aa9C~{ufRQT$CPv zDhnQRG10$wJ&5B8P4nlGn}f1lArQy)e=w0hk38RyP&-d7VZ!SLrrCb|Bc{@As?Jqm zmqN3hsOq1z!qRBZ!33~h!g`f0)e<#U`@%MzZ<0-B_i6d3XDp-0!jWO{iPHYP1|U#C z50&>&0tt*v3P*Amg95~POZ^(*LMqy$GY!eCg})31`_uC|Z(KeUz)~)Nuf>pP{h`fO zkC;q$`34q_`bHia7W4v-PYzsYNZ@a{fhGlAFh0SFabkZz0=@Vk^e1q^Wc^715ZTU0T@%J-6$N-H;K4(o!iD6@4G*?c| zROl&*tZ%`nPk%8OM}g9Iy~F74$CL;n#s#C0?TxkI(&0rP*#ci5n}6sZ0T!D0kDqHc zi4ff3EWd4P}zohbP4U~fSG z35gInsEYspU#1hl?f$M&{Q7@|L>E%<>~G)BLoWe?fRybnO{V9|eIQM9At%Yzo*;k- zm=fC*#&wT*x!c6g-aqTU9G81}x;J`#d3)vbSYZFLJpl(;ses9vNv2rp^u3sXhT$}NG`FzRJ^~V!Svd2PGC}VI`{y0yh z&CL*ZNNa)_G|Ah)6(W8m=oW>gGLobF`w{2~Z9xCIMP2wI7-hVf;$z9AKPqoAk|4e* zS6O}nG9P(Dj-ei(&ZgaTaR4|p!}@x)DQ&kk&O=?bC1-<#m>L_ZZ8c>$6!Q(qNeZHX z*IE_X*|J}_2h>pGKte=UA(j3p4v_RF!lBt9>9u!&1uPw|Hy=e(n4d9c2N8Il-Y}Wm zf%Z_2nQXvBdgi@{mdkaNNoGXD7ODyjnTI70K||XY4&2e3n{0^lcsDO%;M+8kM z*({e!PEP)GyVMf=Xp_>Ku-vk_vhwrPF`-rOupwoX_)>odd^d5Qf;304GvI@Co`<_D zt3E(6hy);xyH61V5tsp+-q~^gSgypl;K9YGU`07X5P~bCDH-BvxXl734Y zWM!&#$uTY;>^yX#e&?*ZT#_6YB}q#pT~xg|L3A+iLkN28W#)4k1_IS}JEJtxeUF<8C}I;Li=_xWb^lQfir|!H!c9xwjB%NtmVVuKN+sb*aIDWMg41Qp`#`pG7 z_IIoE{>&UH1p*c9&uOgR6&i=A?2kt^Mw|VTjL*|pUlQi3x1AO8Z1e#+gn(MJm3YVP zTm-YCS(Pvx>iDlrfqTsJ$C9MmN) z+>Sz9Z3zbeUyacKgHXFnm25gH&f{Ya^W1VWe2JQ6oA;u+yZ=S?1yo(jF%JFj*qtNcQBp7B`fsnJ#9z*P9Vz-{-3eT=E_U#>P7~7*~g#1EidXDc0aS7Dq`9ERO2Mt_HG+NNp+-H;aS&@TH9*0eW z%oBx71-SKjj9d>#*%x}1HFE%u!`BhQT~G;$O@R1ywZ@*7AlOSXIhBuP)y}u9Eqwsj zdCOY0#p075CgS$$>;gyv#aLizq{Q#i+Q{#6M$;7i#XZD`Q1T7B=4ZC&7TH9SI zFXLqsnJ%hJjwYv-9D0)xMZswTXi(LxovMsD8ja+SOlG_szfzxG z9 z3la}zN~V5ux$Y|q;e-WdN?=y4|F8l&S65H46}>t0LG-J?q%*%%HBuY#KK> z9I=dZ3iG@)66OcEEckpMPVEeCHM17<#`ki zR|hom+aWW+6@oY`H{g!;wf@C_wr%}8Yh9(cx2>eh)&k* zTi;p6?~ZIi5KMAHe7K6^#ozpeoA%qEOn?o9>gMY8O@NwTs`6!242ti#FM+iGxX)Pq z9VyCtZ%1x7No0fyw<<499~hYP;%|)AiqAg z>bS7y;e)0hRkNfHCzs=vSz)pTlUMr;)yqr881$M5p2C-^g5*bAd_{)>auw}%Yn@;P z_et)ncAsO?8u1ao8vg{u_Qp49UHtRpa20;Xr;QmmCo2hE$kT=xJytFfTS0%5&L|W_ z5e0XIdD8btbV&VJdcDe8&R<<~G;570K1hi^i%_SVqqx(i8;O1U7b(8~c_`lruC+|J zxNbaEdR;BIg0hB*9JrIO`qda6PdwABH#MYNM?wo$7OPgFkz^32?{^Ct+F+0-Jl`$R zU5@h+3w9bo4>{u_hQkt)lhAbFt7d{8D_b-khDfY~u zQ};cB44kmm0>o2Rohhz8p3^e+w;63&-bin-KcuASc(7ToNy;d2MN-Za(?1Tcl*

    100
    +101
    +102
    +103
    +104
    +105
    +106
    +107
    +108
    +109
    +110
    +111
    +112
    def keyset_from_paused_state(state: "State") -> Keyset:
         """
         Get the keyset for the given Paused state.
     
    diff --git a/versions/unreleased/api-ref/prefect/logging/configuration/index.html b/versions/unreleased/api-ref/prefect/logging/configuration/index.html
    index c6ccd4a6ba..cb61cceed5 100644
    --- a/versions/unreleased/api-ref/prefect/logging/configuration/index.html
    +++ b/versions/unreleased/api-ref/prefect/logging/configuration/index.html
    @@ -848,6 +848,8 @@
             
           
             
    +      
    +        
           
             
           
    @@ -1104,6 +1106,32 @@
       
       
       
    +    
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/logging/formatters/index.html b/versions/unreleased/api-ref/prefect/logging/formatters/index.html index 12a739c6e0..ddbe2b9914 100644 --- a/versions/unreleased/api-ref/prefect/logging/formatters/index.html +++ b/versions/unreleased/api-ref/prefect/logging/formatters/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/logging/handlers/index.html b/versions/unreleased/api-ref/prefect/logging/handlers/index.html index 0db93765e6..25ea719aa0 100644 --- a/versions/unreleased/api-ref/prefect/logging/handlers/index.html +++ b/versions/unreleased/api-ref/prefect/logging/handlers/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/logging/highlighters/index.html b/versions/unreleased/api-ref/prefect/logging/highlighters/index.html index d3341efaf5..25093b6857 100644 --- a/versions/unreleased/api-ref/prefect/logging/highlighters/index.html +++ b/versions/unreleased/api-ref/prefect/logging/highlighters/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/logging/index.html b/versions/unreleased/api-ref/prefect/logging/index.html index b4b867369c..a5c8962b2a 100644 --- a/versions/unreleased/api-ref/prefect/logging/index.html +++ b/versions/unreleased/api-ref/prefect/logging/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/logging/loggers/index.html b/versions/unreleased/api-ref/prefect/logging/loggers/index.html index 6afb41550f..2690e6b20e 100644 --- a/versions/unreleased/api-ref/prefect/logging/loggers/index.html +++ b/versions/unreleased/api-ref/prefect/logging/loggers/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/manifests/index.html b/versions/unreleased/api-ref/prefect/manifests/index.html index 3138d8b648..3d142adff3 100644 --- a/versions/unreleased/api-ref/prefect/manifests/index.html +++ b/versions/unreleased/api-ref/prefect/manifests/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/packaging/base/index.html b/versions/unreleased/api-ref/prefect/packaging/base/index.html index 8d9cc38193..fd286d3a34 100644 --- a/versions/unreleased/api-ref/prefect/packaging/base/index.html +++ b/versions/unreleased/api-ref/prefect/packaging/base/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/packaging/docker/index.html b/versions/unreleased/api-ref/prefect/packaging/docker/index.html index 55e0dcf3a1..c36b85b53c 100644 --- a/versions/unreleased/api-ref/prefect/packaging/docker/index.html +++ b/versions/unreleased/api-ref/prefect/packaging/docker/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/packaging/file/index.html b/versions/unreleased/api-ref/prefect/packaging/file/index.html index 44969dd6c5..552fb2fea8 100644 --- a/versions/unreleased/api-ref/prefect/packaging/file/index.html +++ b/versions/unreleased/api-ref/prefect/packaging/file/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/packaging/serializers/index.html b/versions/unreleased/api-ref/prefect/packaging/serializers/index.html index 3cd0de14d5..714f1c061d 100644 --- a/versions/unreleased/api-ref/prefect/packaging/serializers/index.html +++ b/versions/unreleased/api-ref/prefect/packaging/serializers/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/runner/runner/index.html b/versions/unreleased/api-ref/prefect/runner/runner/index.html index e8bba3d5dd..203340358c 100644 --- a/versions/unreleased/api-ref/prefect/runner/runner/index.html +++ b/versions/unreleased/api-ref/prefect/runner/runner/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/runner/server/index.html b/versions/unreleased/api-ref/prefect/runner/server/index.html index 6249e98252..b884aa1cd8 100644 --- a/versions/unreleased/api-ref/prefect/runner/server/index.html +++ b/versions/unreleased/api-ref/prefect/runner/server/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/runner/storage/index.html b/versions/unreleased/api-ref/prefect/runner/storage/index.html index 7ede63adcd..e44da1115b 100644 --- a/versions/unreleased/api-ref/prefect/runner/storage/index.html +++ b/versions/unreleased/api-ref/prefect/runner/storage/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/runner/utils/index.html b/versions/unreleased/api-ref/prefect/runner/utils/index.html index 283f123ada..b7043c0fd8 100644 --- a/versions/unreleased/api-ref/prefect/runner/utils/index.html +++ b/versions/unreleased/api-ref/prefect/runner/utils/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/runtime/deployment/index.html b/versions/unreleased/api-ref/prefect/runtime/deployment/index.html index 156cf54899..87f2c408af 100644 --- a/versions/unreleased/api-ref/prefect/runtime/deployment/index.html +++ b/versions/unreleased/api-ref/prefect/runtime/deployment/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/runtime/flow_run/index.html b/versions/unreleased/api-ref/prefect/runtime/flow_run/index.html index 1f397628d1..bcd08d63b4 100644 --- a/versions/unreleased/api-ref/prefect/runtime/flow_run/index.html +++ b/versions/unreleased/api-ref/prefect/runtime/flow_run/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/runtime/task_run/index.html b/versions/unreleased/api-ref/prefect/runtime/task_run/index.html index 4ed0430b15..1b78d96abe 100644 --- a/versions/unreleased/api-ref/prefect/runtime/task_run/index.html +++ b/versions/unreleased/api-ref/prefect/runtime/task_run/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/serializers/index.html b/versions/unreleased/api-ref/prefect/serializers/index.html index e96f6ba9e6..1b8437b0b3 100644 --- a/versions/unreleased/api-ref/prefect/serializers/index.html +++ b/versions/unreleased/api-ref/prefect/serializers/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/settings/index.html b/versions/unreleased/api-ref/prefect/settings/index.html index e392fe1ed4..2f4c1cf27d 100644 --- a/versions/unreleased/api-ref/prefect/settings/index.html +++ b/versions/unreleased/api-ref/prefect/settings/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/software/index.html b/versions/unreleased/api-ref/prefect/software/index.html index ccb85f8fa3..00cb2a105f 100644 --- a/versions/unreleased/api-ref/prefect/software/index.html +++ b/versions/unreleased/api-ref/prefect/software/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/states/index.html b/versions/unreleased/api-ref/prefect/states/index.html index 843a6c8417..b365e94af9 100644 --- a/versions/unreleased/api-ref/prefect/states/index.html +++ b/versions/unreleased/api-ref/prefect/states/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/task-runners/index.html b/versions/unreleased/api-ref/prefect/task-runners/index.html index 924c9e682e..96e94a8afc 100644 --- a/versions/unreleased/api-ref/prefect/task-runners/index.html +++ b/versions/unreleased/api-ref/prefect/task-runners/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/tasks/index.html b/versions/unreleased/api-ref/prefect/tasks/index.html index ce03c0459a..35a6d80e4c 100644 --- a/versions/unreleased/api-ref/prefect/tasks/index.html +++ b/versions/unreleased/api-ref/prefect/tasks/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/testing/index.html b/versions/unreleased/api-ref/prefect/testing/index.html index 6e789e90fb..d955d39d37 100644 --- a/versions/unreleased/api-ref/prefect/testing/index.html +++ b/versions/unreleased/api-ref/prefect/testing/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/annotations/index.html b/versions/unreleased/api-ref/prefect/utilities/annotations/index.html index 03549fe113..5f51cb6eca 100644 --- a/versions/unreleased/api-ref/prefect/utilities/annotations/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/annotations/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/asyncutils/index.html b/versions/unreleased/api-ref/prefect/utilities/asyncutils/index.html index f5b14c5017..d9c2e02d1c 100644 --- a/versions/unreleased/api-ref/prefect/utilities/asyncutils/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/asyncutils/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/callables/index.html b/versions/unreleased/api-ref/prefect/utilities/callables/index.html index 21f93d21ee..3ce01cf18a 100644 --- a/versions/unreleased/api-ref/prefect/utilities/callables/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/callables/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/collections/index.html b/versions/unreleased/api-ref/prefect/utilities/collections/index.html index cb67372ad3..62d354b75f 100644 --- a/versions/unreleased/api-ref/prefect/utilities/collections/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/collections/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/compat/index.html b/versions/unreleased/api-ref/prefect/utilities/compat/index.html index 95b330d986..21065255ad 100644 --- a/versions/unreleased/api-ref/prefect/utilities/compat/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/compat/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/context/index.html b/versions/unreleased/api-ref/prefect/utilities/context/index.html index 9b39b43566..93a4fb0e3e 100644 --- a/versions/unreleased/api-ref/prefect/utilities/context/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/context/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/dispatch/index.html b/versions/unreleased/api-ref/prefect/utilities/dispatch/index.html index 1577528c66..e8ec55b1cc 100644 --- a/versions/unreleased/api-ref/prefect/utilities/dispatch/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/dispatch/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/dockerutils/index.html b/versions/unreleased/api-ref/prefect/utilities/dockerutils/index.html index 37a7ea7c13..952b5d8c4a 100644 --- a/versions/unreleased/api-ref/prefect/utilities/dockerutils/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/dockerutils/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/filesystem/index.html b/versions/unreleased/api-ref/prefect/utilities/filesystem/index.html index 8f3233edc6..5dfa6d9c6b 100644 --- a/versions/unreleased/api-ref/prefect/utilities/filesystem/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/filesystem/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/hashing/index.html b/versions/unreleased/api-ref/prefect/utilities/hashing/index.html index a6140f0ef1..ad04c52d58 100644 --- a/versions/unreleased/api-ref/prefect/utilities/hashing/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/hashing/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/importtools/index.html b/versions/unreleased/api-ref/prefect/utilities/importtools/index.html index c4ef3d89a8..ca79f20ea4 100644 --- a/versions/unreleased/api-ref/prefect/utilities/importtools/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/importtools/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/math/index.html b/versions/unreleased/api-ref/prefect/utilities/math/index.html index 4f2c5c26f1..34d671e10d 100644 --- a/versions/unreleased/api-ref/prefect/utilities/math/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/math/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/names/index.html b/versions/unreleased/api-ref/prefect/utilities/names/index.html index dbdae0939e..a26dd0520d 100644 --- a/versions/unreleased/api-ref/prefect/utilities/names/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/names/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/processutils/index.html b/versions/unreleased/api-ref/prefect/utilities/processutils/index.html index 6b0ddcc37e..25f79f4584 100644 --- a/versions/unreleased/api-ref/prefect/utilities/processutils/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/processutils/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/pydantic/index.html b/versions/unreleased/api-ref/prefect/utilities/pydantic/index.html index ce6ef47224..41a2b82803 100644 --- a/versions/unreleased/api-ref/prefect/utilities/pydantic/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/pydantic/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/render_swagger/index.html b/versions/unreleased/api-ref/prefect/utilities/render_swagger/index.html index 76753e33a7..6b49ba06a2 100644 --- a/versions/unreleased/api-ref/prefect/utilities/render_swagger/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/render_swagger/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/services/index.html b/versions/unreleased/api-ref/prefect/utilities/services/index.html index 9b0f7de6de..0dfe0fd392 100644 --- a/versions/unreleased/api-ref/prefect/utilities/services/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/services/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/slugify/index.html b/versions/unreleased/api-ref/prefect/utilities/slugify/index.html index 04750686f1..654ad7093b 100644 --- a/versions/unreleased/api-ref/prefect/utilities/slugify/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/slugify/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/templating/index.html b/versions/unreleased/api-ref/prefect/utilities/templating/index.html index 38e3f4c7c1..cf147678f9 100644 --- a/versions/unreleased/api-ref/prefect/utilities/templating/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/templating/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/text/index.html b/versions/unreleased/api-ref/prefect/utilities/text/index.html index 5388754acf..199321a45a 100644 --- a/versions/unreleased/api-ref/prefect/utilities/text/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/text/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/validation/index.html b/versions/unreleased/api-ref/prefect/utilities/validation/index.html index 3bc18d4fd3..1f3ddbf643 100644 --- a/versions/unreleased/api-ref/prefect/utilities/validation/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/validation/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/utilities/visualization/index.html b/versions/unreleased/api-ref/prefect/utilities/visualization/index.html index 2a5ae57ee9..3d1a7c6e37 100644 --- a/versions/unreleased/api-ref/prefect/utilities/visualization/index.html +++ b/versions/unreleased/api-ref/prefect/utilities/visualization/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/variables/index.html b/versions/unreleased/api-ref/prefect/variables/index.html index 11fb90ff2b..f548e25fc5 100644 --- a/versions/unreleased/api-ref/prefect/variables/index.html +++ b/versions/unreleased/api-ref/prefect/variables/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/workers/base/index.html b/versions/unreleased/api-ref/prefect/workers/base/index.html index a53bc8ecb3..1345f9a443 100644 --- a/versions/unreleased/api-ref/prefect/workers/base/index.html +++ b/versions/unreleased/api-ref/prefect/workers/base/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/workers/block/index.html b/versions/unreleased/api-ref/prefect/workers/block/index.html index 32876b5dc5..e486afb79d 100644 --- a/versions/unreleased/api-ref/prefect/workers/block/index.html +++ b/versions/unreleased/api-ref/prefect/workers/block/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/workers/process/index.html b/versions/unreleased/api-ref/prefect/workers/process/index.html index caa13196e5..0ac03ce1fa 100644 --- a/versions/unreleased/api-ref/prefect/workers/process/index.html +++ b/versions/unreleased/api-ref/prefect/workers/process/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/workers/server/index.html b/versions/unreleased/api-ref/prefect/workers/server/index.html index e0b3b673c3..80dcc789a3 100644 --- a/versions/unreleased/api-ref/prefect/workers/server/index.html +++ b/versions/unreleased/api-ref/prefect/workers/server/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/prefect/workers/utilities/index.html b/versions/unreleased/api-ref/prefect/workers/utilities/index.html index 29e47e5d19..a396a79a59 100644 --- a/versions/unreleased/api-ref/prefect/workers/utilities/index.html +++ b/versions/unreleased/api-ref/prefect/workers/utilities/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/python/index.html b/versions/unreleased/api-ref/python/index.html index ff24e019d0..21f50d2ac0 100644 --- a/versions/unreleased/api-ref/python/index.html +++ b/versions/unreleased/api-ref/python/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/rest-api-reference/index.html b/versions/unreleased/api-ref/rest-api-reference/index.html index 0d92743b60..2924656238 100644 --- a/versions/unreleased/api-ref/rest-api-reference/index.html +++ b/versions/unreleased/api-ref/rest-api-reference/index.html @@ -845,6 +845,8 @@ + + @@ -1101,6 +1103,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/rest-api/index.html b/versions/unreleased/api-ref/rest-api/index.html index 64282f61d6..51aa57eff9 100644 --- a/versions/unreleased/api-ref/rest-api/index.html +++ b/versions/unreleased/api-ref/rest-api/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/admin/index.html b/versions/unreleased/api-ref/server/api/admin/index.html index 33aa5c35c9..67dabf81af 100644 --- a/versions/unreleased/api-ref/server/api/admin/index.html +++ b/versions/unreleased/api-ref/server/api/admin/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/dependencies/index.html b/versions/unreleased/api-ref/server/api/dependencies/index.html index a11df0d204..11f9f01a27 100644 --- a/versions/unreleased/api-ref/server/api/dependencies/index.html +++ b/versions/unreleased/api-ref/server/api/dependencies/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/deployments/index.html b/versions/unreleased/api-ref/server/api/deployments/index.html index 208faa4d75..f928ed5c28 100644 --- a/versions/unreleased/api-ref/server/api/deployments/index.html +++ b/versions/unreleased/api-ref/server/api/deployments/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/flow_run_states/index.html b/versions/unreleased/api-ref/server/api/flow_run_states/index.html index 56f5b982c0..5de8c1e958 100644 --- a/versions/unreleased/api-ref/server/api/flow_run_states/index.html +++ b/versions/unreleased/api-ref/server/api/flow_run_states/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/flow_runs/index.html b/versions/unreleased/api-ref/server/api/flow_runs/index.html index b70bdc30ab..812a5ab316 100644 --- a/versions/unreleased/api-ref/server/api/flow_runs/index.html +++ b/versions/unreleased/api-ref/server/api/flow_runs/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/flows/index.html b/versions/unreleased/api-ref/server/api/flows/index.html index 88777d9a91..b9eb8f753c 100644 --- a/versions/unreleased/api-ref/server/api/flows/index.html +++ b/versions/unreleased/api-ref/server/api/flows/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/run_history/index.html b/versions/unreleased/api-ref/server/api/run_history/index.html index c09b45d8d9..79ebefbc2b 100644 --- a/versions/unreleased/api-ref/server/api/run_history/index.html +++ b/versions/unreleased/api-ref/server/api/run_history/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/saved_searches/index.html b/versions/unreleased/api-ref/server/api/saved_searches/index.html index 2443d7d2fe..4d926bde5a 100644 --- a/versions/unreleased/api-ref/server/api/saved_searches/index.html +++ b/versions/unreleased/api-ref/server/api/saved_searches/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/server/index.html b/versions/unreleased/api-ref/server/api/server/index.html index b1efdbf590..17f302a7f0 100644 --- a/versions/unreleased/api-ref/server/api/server/index.html +++ b/versions/unreleased/api-ref/server/api/server/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/task_run_states/index.html b/versions/unreleased/api-ref/server/api/task_run_states/index.html index 291c0ff5f3..e6695625f3 100644 --- a/versions/unreleased/api-ref/server/api/task_run_states/index.html +++ b/versions/unreleased/api-ref/server/api/task_run_states/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/api/task_runs/index.html b/versions/unreleased/api-ref/server/api/task_runs/index.html index 6fb5405e69..2771103e44 100644 --- a/versions/unreleased/api-ref/server/api/task_runs/index.html +++ b/versions/unreleased/api-ref/server/api/task_runs/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/index.html b/versions/unreleased/api-ref/server/index.html index d29fa61b3a..737f1e36aa 100644 --- a/versions/unreleased/api-ref/server/index.html +++ b/versions/unreleased/api-ref/server/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/models/deployments/index.html b/versions/unreleased/api-ref/server/models/deployments/index.html index 1b7c7b9227..0c6b194774 100644 --- a/versions/unreleased/api-ref/server/models/deployments/index.html +++ b/versions/unreleased/api-ref/server/models/deployments/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/models/flow_run_states/index.html b/versions/unreleased/api-ref/server/models/flow_run_states/index.html index fabdeeed64..b71dc607ec 100644 --- a/versions/unreleased/api-ref/server/models/flow_run_states/index.html +++ b/versions/unreleased/api-ref/server/models/flow_run_states/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/models/flow_runs/index.html b/versions/unreleased/api-ref/server/models/flow_runs/index.html index b874043fa7..eee12038d1 100644 --- a/versions/unreleased/api-ref/server/models/flow_runs/index.html +++ b/versions/unreleased/api-ref/server/models/flow_runs/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/models/flows/index.html b/versions/unreleased/api-ref/server/models/flows/index.html index cd86cfa88b..b48cc3b08a 100644 --- a/versions/unreleased/api-ref/server/models/flows/index.html +++ b/versions/unreleased/api-ref/server/models/flows/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/models/saved_searches/index.html b/versions/unreleased/api-ref/server/models/saved_searches/index.html index 6f9a8bce98..19b2e9edc3 100644 --- a/versions/unreleased/api-ref/server/models/saved_searches/index.html +++ b/versions/unreleased/api-ref/server/models/saved_searches/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/models/task_run_states/index.html b/versions/unreleased/api-ref/server/models/task_run_states/index.html index 272ee5e8f5..29ea7e6ceb 100644 --- a/versions/unreleased/api-ref/server/models/task_run_states/index.html +++ b/versions/unreleased/api-ref/server/models/task_run_states/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/models/task_runs/index.html b/versions/unreleased/api-ref/server/models/task_runs/index.html index 7b5ab780b9..eab1c93c00 100644 --- a/versions/unreleased/api-ref/server/models/task_runs/index.html +++ b/versions/unreleased/api-ref/server/models/task_runs/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/orchestration/core_policy/index.html b/versions/unreleased/api-ref/server/orchestration/core_policy/index.html index 838fa97ac1..6eb6e504cb 100644 --- a/versions/unreleased/api-ref/server/orchestration/core_policy/index.html +++ b/versions/unreleased/api-ref/server/orchestration/core_policy/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/orchestration/global_policy/index.html b/versions/unreleased/api-ref/server/orchestration/global_policy/index.html index 29a7e5589f..bae3e78ba5 100644 --- a/versions/unreleased/api-ref/server/orchestration/global_policy/index.html +++ b/versions/unreleased/api-ref/server/orchestration/global_policy/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/orchestration/policies/index.html b/versions/unreleased/api-ref/server/orchestration/policies/index.html index 2ca2c0dea4..fa8536dc15 100644 --- a/versions/unreleased/api-ref/server/orchestration/policies/index.html +++ b/versions/unreleased/api-ref/server/orchestration/policies/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/orchestration/rules/index.html b/versions/unreleased/api-ref/server/orchestration/rules/index.html index 16f7ff5bc5..058359c44d 100644 --- a/versions/unreleased/api-ref/server/orchestration/rules/index.html +++ b/versions/unreleased/api-ref/server/orchestration/rules/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/schemas/actions/index.html b/versions/unreleased/api-ref/server/schemas/actions/index.html index cbe5bc0441..419b13c0dc 100644 --- a/versions/unreleased/api-ref/server/schemas/actions/index.html +++ b/versions/unreleased/api-ref/server/schemas/actions/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/schemas/core/index.html b/versions/unreleased/api-ref/server/schemas/core/index.html index 6fdf0c941e..3e428b3421 100644 --- a/versions/unreleased/api-ref/server/schemas/core/index.html +++ b/versions/unreleased/api-ref/server/schemas/core/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/schemas/filters/index.html b/versions/unreleased/api-ref/server/schemas/filters/index.html index 40d4c39e12..153465faa9 100644 --- a/versions/unreleased/api-ref/server/schemas/filters/index.html +++ b/versions/unreleased/api-ref/server/schemas/filters/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/schemas/responses/index.html b/versions/unreleased/api-ref/server/schemas/responses/index.html index 7ea028c772..171133e3bb 100644 --- a/versions/unreleased/api-ref/server/schemas/responses/index.html +++ b/versions/unreleased/api-ref/server/schemas/responses/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/schemas/schedules/index.html b/versions/unreleased/api-ref/server/schemas/schedules/index.html index 05108bc172..3173a3aa73 100644 --- a/versions/unreleased/api-ref/server/schemas/schedules/index.html +++ b/versions/unreleased/api-ref/server/schemas/schedules/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/schemas/sorting/index.html b/versions/unreleased/api-ref/server/schemas/sorting/index.html index 99e487feb6..a435f321ea 100644 --- a/versions/unreleased/api-ref/server/schemas/sorting/index.html +++ b/versions/unreleased/api-ref/server/schemas/sorting/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/schemas/states/index.html b/versions/unreleased/api-ref/server/schemas/states/index.html index 9b285886f8..826e32e057 100644 --- a/versions/unreleased/api-ref/server/schemas/states/index.html +++ b/versions/unreleased/api-ref/server/schemas/states/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/services/late_runs/index.html b/versions/unreleased/api-ref/server/services/late_runs/index.html index a5e56008a2..41b9351887 100644 --- a/versions/unreleased/api-ref/server/services/late_runs/index.html +++ b/versions/unreleased/api-ref/server/services/late_runs/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/services/loop_service/index.html b/versions/unreleased/api-ref/server/services/loop_service/index.html index 9490d49864..cebc9e01ef 100644 --- a/versions/unreleased/api-ref/server/services/loop_service/index.html +++ b/versions/unreleased/api-ref/server/services/loop_service/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/services/scheduler/index.html b/versions/unreleased/api-ref/server/services/scheduler/index.html index 949f9f39b2..4d1403f4dc 100644 --- a/versions/unreleased/api-ref/server/services/scheduler/index.html +++ b/versions/unreleased/api-ref/server/services/scheduler/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/utilities/database/index.html b/versions/unreleased/api-ref/server/utilities/database/index.html index 881e3c6186..44f22d3840 100644 --- a/versions/unreleased/api-ref/server/utilities/database/index.html +++ b/versions/unreleased/api-ref/server/utilities/database/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/utilities/schemas/index.html b/versions/unreleased/api-ref/server/utilities/schemas/index.html index 7833e41504..4356412191 100644 --- a/versions/unreleased/api-ref/server/utilities/schemas/index.html +++ b/versions/unreleased/api-ref/server/utilities/schemas/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/api-ref/server/utilities/server/index.html b/versions/unreleased/api-ref/server/utilities/server/index.html index 8ac8c8f2f7..caec5f1018 100644 --- a/versions/unreleased/api-ref/server/utilities/server/index.html +++ b/versions/unreleased/api-ref/server/utilities/server/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/assets/images/social/guides/automations.png b/versions/unreleased/assets/images/social/guides/automations.png new file mode 100644 index 0000000000000000000000000000000000000000..611fb7b554384b42f66802002d779ef52554c7a4 GIT binary patch literal 50061 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 literal 0 HcmV?d00001 diff --git a/versions/unreleased/cloud/cloud-quickstart/index.html b/versions/unreleased/cloud/cloud-quickstart/index.html index 7845572d67..58f903e0f8 100644 --- a/versions/unreleased/cloud/cloud-quickstart/index.html +++ b/versions/unreleased/cloud/cloud-quickstart/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/connecting/index.html b/versions/unreleased/cloud/connecting/index.html index f3ea19ad44..f4da9e7eaf 100644 --- a/versions/unreleased/cloud/connecting/index.html +++ b/versions/unreleased/cloud/connecting/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/incidents/index.html b/versions/unreleased/cloud/incidents/index.html index c54e8cce85..e063a6d26d 100644 --- a/versions/unreleased/cloud/incidents/index.html +++ b/versions/unreleased/cloud/incidents/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/index.html b/versions/unreleased/cloud/index.html index 686157d94a..e06327f632 100644 --- a/versions/unreleased/cloud/index.html +++ b/versions/unreleased/cloud/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/rate-limits/index.html b/versions/unreleased/cloud/rate-limits/index.html index 578f102d46..f8dd0997d8 100644 --- a/versions/unreleased/cloud/rate-limits/index.html +++ b/versions/unreleased/cloud/rate-limits/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/users/api-keys/index.html b/versions/unreleased/cloud/users/api-keys/index.html index 39a1317e7c..051ecdda79 100644 --- a/versions/unreleased/cloud/users/api-keys/index.html +++ b/versions/unreleased/cloud/users/api-keys/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/users/audit-log/index.html b/versions/unreleased/cloud/users/audit-log/index.html index d467406aee..5a2530a70a 100644 --- a/versions/unreleased/cloud/users/audit-log/index.html +++ b/versions/unreleased/cloud/users/audit-log/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/users/index.html b/versions/unreleased/cloud/users/index.html index 7adadf5a51..91bcb3f1a2 100644 --- a/versions/unreleased/cloud/users/index.html +++ b/versions/unreleased/cloud/users/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/users/object-access-control-lists/index.html b/versions/unreleased/cloud/users/object-access-control-lists/index.html index ebd7d7006f..e319a3e238 100644 --- a/versions/unreleased/cloud/users/object-access-control-lists/index.html +++ b/versions/unreleased/cloud/users/object-access-control-lists/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/users/roles/index.html b/versions/unreleased/cloud/users/roles/index.html index e251bf66af..771fc93bcd 100644 --- a/versions/unreleased/cloud/users/roles/index.html +++ b/versions/unreleased/cloud/users/roles/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/users/service-accounts/index.html b/versions/unreleased/cloud/users/service-accounts/index.html index e22365b38f..e57c0ebdbe 100644 --- a/versions/unreleased/cloud/users/service-accounts/index.html +++ b/versions/unreleased/cloud/users/service-accounts/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/users/sso/index.html b/versions/unreleased/cloud/users/sso/index.html index d623aa2f4b..463bf18961 100644 --- a/versions/unreleased/cloud/users/sso/index.html +++ b/versions/unreleased/cloud/users/sso/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/users/teams/index.html b/versions/unreleased/cloud/users/teams/index.html index c466af263f..a34a89a6ca 100644 --- a/versions/unreleased/cloud/users/teams/index.html +++ b/versions/unreleased/cloud/users/teams/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/cloud/workspaces/index.html b/versions/unreleased/cloud/workspaces/index.html index 221350c916..d5b8ebe036 100644 --- a/versions/unreleased/cloud/workspaces/index.html +++ b/versions/unreleased/cloud/workspaces/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/community/index.html b/versions/unreleased/community/index.html index ac0cade261..e545f050d1 100644 --- a/versions/unreleased/community/index.html +++ b/versions/unreleased/community/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/agents/index.html b/versions/unreleased/concepts/agents/index.html index 78bdcc5467..bf3d2dd26a 100644 --- a/versions/unreleased/concepts/agents/index.html +++ b/versions/unreleased/concepts/agents/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/artifacts/index.html b/versions/unreleased/concepts/artifacts/index.html index f7de88ffc4..654eb6eb25 100644 --- a/versions/unreleased/concepts/artifacts/index.html +++ b/versions/unreleased/concepts/artifacts/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/automations/index.html b/versions/unreleased/concepts/automations/index.html index d4d53cf35a..3dcaef4c5a 100644 --- a/versions/unreleased/concepts/automations/index.html +++ b/versions/unreleased/concepts/automations/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/blocks/index.html b/versions/unreleased/concepts/blocks/index.html index d314c546d5..65a1d26f48 100644 --- a/versions/unreleased/concepts/blocks/index.html +++ b/versions/unreleased/concepts/blocks/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/deployments-block-based/index.html b/versions/unreleased/concepts/deployments-block-based/index.html index dd719d4f01..0dc53536d3 100644 --- a/versions/unreleased/concepts/deployments-block-based/index.html +++ b/versions/unreleased/concepts/deployments-block-based/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/deployments/index.html b/versions/unreleased/concepts/deployments/index.html index 1e49800c39..9f64e29b73 100644 --- a/versions/unreleased/concepts/deployments/index.html +++ b/versions/unreleased/concepts/deployments/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/events/index.html b/versions/unreleased/concepts/events/index.html index f9eab479e6..cef6bac721 100644 --- a/versions/unreleased/concepts/events/index.html +++ b/versions/unreleased/concepts/events/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/filesystems/index.html b/versions/unreleased/concepts/filesystems/index.html index 99cba8576c..20b107fb0c 100644 --- a/versions/unreleased/concepts/filesystems/index.html +++ b/versions/unreleased/concepts/filesystems/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/flows/index.html b/versions/unreleased/concepts/flows/index.html index 479a2e7442..37d07735ad 100644 --- a/versions/unreleased/concepts/flows/index.html +++ b/versions/unreleased/concepts/flows/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/index.html b/versions/unreleased/concepts/index.html index adbeb8d845..4d71d3b858 100644 --- a/versions/unreleased/concepts/index.html +++ b/versions/unreleased/concepts/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/infrastructure/index.html b/versions/unreleased/concepts/infrastructure/index.html index 01b27c5b01..a34babbcd7 100644 --- a/versions/unreleased/concepts/infrastructure/index.html +++ b/versions/unreleased/concepts/infrastructure/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/results/index.html b/versions/unreleased/concepts/results/index.html index a484d25ccb..38cf9e1167 100644 --- a/versions/unreleased/concepts/results/index.html +++ b/versions/unreleased/concepts/results/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/schedules/index.html b/versions/unreleased/concepts/schedules/index.html index c0af25691e..e79762a1c9 100644 --- a/versions/unreleased/concepts/schedules/index.html +++ b/versions/unreleased/concepts/schedules/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/states/index.html b/versions/unreleased/concepts/states/index.html index b30f738163..6a7aba6f47 100644 --- a/versions/unreleased/concepts/states/index.html +++ b/versions/unreleased/concepts/states/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/storage/index.html b/versions/unreleased/concepts/storage/index.html index f33fa2d73e..aeb7ffb06d 100644 --- a/versions/unreleased/concepts/storage/index.html +++ b/versions/unreleased/concepts/storage/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/task-runners/index.html b/versions/unreleased/concepts/task-runners/index.html index f34a7a6720..e4b64b335c 100644 --- a/versions/unreleased/concepts/task-runners/index.html +++ b/versions/unreleased/concepts/task-runners/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/tasks/index.html b/versions/unreleased/concepts/tasks/index.html index 96ded3b960..5df0be247b 100644 --- a/versions/unreleased/concepts/tasks/index.html +++ b/versions/unreleased/concepts/tasks/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/concepts/work-pools/index.html b/versions/unreleased/concepts/work-pools/index.html index 193464f2a0..159b8e72ce 100644 --- a/versions/unreleased/concepts/work-pools/index.html +++ b/versions/unreleased/concepts/work-pools/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/contributing/overview/index.html b/versions/unreleased/contributing/overview/index.html index d0dd47861c..dd1b4ae825 100644 --- a/versions/unreleased/contributing/overview/index.html +++ b/versions/unreleased/contributing/overview/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/contributing/style/index.html b/versions/unreleased/contributing/style/index.html index 11bcad8588..094a1a5e07 100644 --- a/versions/unreleased/contributing/style/index.html +++ b/versions/unreleased/contributing/style/index.html @@ -848,6 +848,8 @@ + + @@ -1104,6 +1106,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/contributing/versioning/index.html b/versions/unreleased/contributing/versioning/index.html index c13009daa7..1d9fbf71b0 100644 --- a/versions/unreleased/contributing/versioning/index.html +++ b/versions/unreleased/contributing/versioning/index.html @@ -846,6 +846,8 @@ + + @@ -1102,6 +1104,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/faq/index.html b/versions/unreleased/faq/index.html index 0a7d44cc98..290be17c3c 100644 --- a/versions/unreleased/faq/index.html +++ b/versions/unreleased/faq/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/getting-started/installation/index.html b/versions/unreleased/getting-started/installation/index.html index 6dac4dd357..8df4dd8679 100644 --- a/versions/unreleased/getting-started/installation/index.html +++ b/versions/unreleased/getting-started/installation/index.html @@ -1090,6 +1090,8 @@ + + @@ -1346,6 +1348,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/getting-started/quickstart/index.html b/versions/unreleased/getting-started/quickstart/index.html index 738ae43d89..be8b1c6800 100644 --- a/versions/unreleased/getting-started/quickstart/index.html +++ b/versions/unreleased/getting-started/quickstart/index.html @@ -987,6 +987,8 @@ + + @@ -1243,6 +1245,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/automations/index.html b/versions/unreleased/guides/automations/index.html new file mode 100644 index 0000000000..ee8faf0b44 --- /dev/null +++ b/versions/unreleased/guides/automations/index.html @@ -0,0 +1,9294 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + Automations and Common Use Cases - Prefect Docs + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
    + +
    + + + + + + + + + +
    + + + + +
    + +
    + + + + +
    +
    + + + +
    +
    +
    + + + + + +
    +
    +
    + + + + + + + +
    + + + + + + + + + +
    + + + + + + + + + + + + +

    Using Automations for Dynamic Responses

    +

    From the Automations concept page, we saw what an automation can do and how to configure one within the UI.

    +

    In this guide, we will showcase the following common use cases:

    +
      +
    • Create a simple notification automation in just a few UI clicks
    • +
    • Build upon an event based automation
    • +
    • Combine into a multi-layered responsive deployment pattern
    • +
    +
    +

    Available only on Prefect Cloud

    +
    Automations are a Prefect Cloud feature.
    +
    + +
    +

    Prerequisites

    +

    Please have the following before exploring the guide:

    + +

    Creating the example script

    +

    Automations allow you to take actions in response to triggering events recorded by Prefect.

    +

    For example, let's try to grab data from an API and send a notification based on the end state.

    +

    We can start by pulling hypothetical user data from an endpoint and then performing data cleaning and transformations.

    +

    Let's create a simple extract method, that pulls the data from a random user data generator endpoint.

    +
    from prefect import flow, task, get_run_logger
    +import requests
    +import json
    +
    +@task
    +def fetch(url: str):
    +    logger = get_run_logger()
    +    response = requests.get(url)
    +    raw_data = response.json()
    +    logger.info(f"Raw response: {raw_data}")
    +    return raw_data
    +
    +@task
    +def clean(raw_data: dict):
    +    print(raw_data.get('results')[0])
    +    results = raw_data.get('results')[0]
    +    logger = get_run_logger()
    +    logger.info(f"Cleaned results: {results}")
    +    return results['name']
    +
    +@flow
    +def build_names(num: int = 10):
    +    df = []
    +    url = "https://randomuser.me/api/"
    +    logger = get_run_logger()
    +    copy = num
    +    while num != 0:
    +        raw_data = fetch(url)
    +        df.append(clean(raw_data))
    +        num -= 1
    +    logger.info(f"Built {copy} names: {df}")
    +    return df
    +
    +if __name__ == "__main__":
    +    list_of_names = build_names()
    +
    +

    The data cleaning workflow has visibility into each step, and we are sending a list of names to our next step of our pipeline.

    +

    Create notification block within the UI

    +

    Now let's try to send a notification based off a completed state outcome. We can configure a notification to be sent so that we know when to look into our workflow logic.

    +
      +
    1. +

      Prior to creating the automation, let's confirm the notification location. We have to create a notification block to help define where the notification will be sent. +List of available blocks

      +
    2. +
    3. +

      Let's navigate to the blocks page on the UI, and click into creating an email notification block. +Creating a notification block in the Cloud UI

      +
    4. +
    5. +

      Now that we created a notification block, we can go to the automations page to create our first automation. +Automations page

      +
    6. +
    7. +

      Next we try to find the trigger type, in this case let's use a flow completion. +Trigger type

      +
    8. +
    9. +

      Finally, let's create the actions that will be done once the triggered is hit. In this case, let's create a notification to be sent out to showcase the completion. +Notification block in automation

      +
    10. +
    11. +

      Now the automation is ready to be triggered from a flow run completion. Let's run the file locally and see that the notification is sent to our inbox after the completion. It may take a few minutes for the notification to arrive. +Final notification

      +
    12. +
    +
    +

    No deployment created

    +

    Keep in mind, we did not need to create a deployment to trigger our automation, where a state outcome of a local flow run helped trigger this notification block. We are not required to create a deployment to trigger a notification.

    +
    +

    Now that you've seen how to create an email notification from a flow run completion, let's see how we can kick off a deployment run in response to an event.

    +

    Event-based deployment automation

    +

    We can create an automation that can kick off a deployment instead of a notification. Let's explore how we can programmatically create this automation. We will take advantage of Prefect's REST API to help create this automation.

    +

    See the REST API documentation as a reference for interacting with the Prefect Cloud automation endpoints.

    +

    Let's create a deployment where we can kick off some work based on how long a flow is running. For example, if the build_names flow is taking too long to execute, we can kick off a deployment of the with the same build_names flow, but replace the count value with a lower number - to speed up completion. +You can create a deployment with a prefect.yaml file or a Python file that uses flow.deploy.

    +
    +
    +
    +

    Create a prefect.yaml file like this one for our flow build_names:

    +
      # Welcome to your prefect.yaml file! You can use this file for storing and managing
    +  # configuration for deploying your flows. We recommend committing this file to source
    +  # control along with your flow code.
    +
    +  # Generic metadata about this project
    +  name: automations-guide
    +  prefect-version: 2.13.1
    +
    +  # build section allows you to manage and build docker images
    +  build: null
    +
    +  # push section allows you to manage if and how this project is uploaded to remote locations
    +  push: null
    +
    +  # pull section allows you to provide instructions for cloning this project in remote locations
    +  pull:
    +  - prefect.deployments.steps.set_working_directory:
    +      directory: /Users/src/prefect/Playground/automations-guide
    +
    +  # the deployments section allows you to provide configuration for deploying flows
    +  deployments:
    +  - name: deploy-build-names
    +    version: null
    +    tags: []
    +    description: null
    +    entrypoint: test-automations.py:build_names
    +    parameters: {}
    +    work_pool:
    +      name: tutorial-process-pool
    +      work_queue_name: null
    +      job_variables: {}
    +    schedule: null
    +
    +
    +
    +

    To follow a more python based approach to create a deployment, you can use flow.deploy as in the example below.

    +
    # .deploy only needs a name, valid work pool 
    +# and a reference to where the flow code exists
    +
    +if __name__ == "__main__":
    +build_names.deploy(
    +    name="deploy-build-names",
    +    work_pool_name="tutorial-process-pool"
    +    image="my_registry/my_image:my_image_tag",
    +)
    +
    +
    +
    +
    +

    Now let's grab our deployment_id from this deployment, and embed it in our automation. There are many ways to obtain the deployment_id, but the CLI is a quick way to see all of your deployment ids.

    +
    +

    Find deployment_id from the CLI

    +

    The quickest way to see the ID's associated with your deployment would be running prefect deployment ls + in an authenticated command prompt, and you will be able to see the id's associated with all of your deployments

    +
    +

    prefect deployment ls
    +                                          Deployments                                           
    +┏━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┳━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┓
    +┃ Name                                                   ID                                   ┃
    +┡━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━╇━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━┩
    +│ Extract islands/island-schedule                        d9d7289c-7a41-436d-8313-80a044e61532 │
    +│ build-names/deploy-build-names                         8b10a65e-89ef-4c19-9065-eec5c50242f4 │
    +│ ride-duration-prediction-backfill/backfill-deployment  76dc6581-1773-45c5-a291-7f864d064c57 │
    +└───────────────────────────────────────────────────────┴──────────────────────────────────────┘
    +
    +We can create an automation via a POST call, where we can programmatically create the automation. Ensure you have your api_key, account_id, and workspace_id.

    +
    def create_event_driven_automation():
    +    api_url = f"https://api.prefect.cloud/api/accounts/{account_id}/workspaces/{workspace_id}/automations/"
    +    data = {
    +    "name": "Event Driven Redeploy",
    +    "description": "Programmatically created an automation to redeploy a flow based on an event",
    +    "enabled": "true",
    +    "trigger": {
    +    "after": [
    +        "string"
    +    ],
    +    "expect": [
    +        "prefect.flow-run.Running"
    +    ],
    +    "for_each": [
    +        "prefect.resource.id"
    +    ],
    +    "posture": "Proactive",
    +    "threshold": 30,
    +    "within": 0
    +    },
    +    "actions": [
    +    {
    +        "type": "run-deployment",
    +        "source": "selected",
    +        "deployment_id": "YOUR-DEPLOYMENT-ID", 
    +        "parameters": "10"
    +    }
    +    ],
    +    "owner_resource": "string"
    +        }
    +
    +    headers = {"Authorization": f"Bearer {PREFECT_API_KEY}"}
    +    response = requests.post(api_url, headers=headers, json=data)
    +
    +    print(response.json())
    +    return response.json()
    +
    +

    After running this function, you will see within the UI the changes that came from the post request. Keep in mind, the context will be "custom" on UI.

    +

    Let's run the underlying flow and see the deployment get kicked off after 30 seconds elapsed. This will result in a new flow run of build_names, and we are able to see this new deployment get initiated with the custom parameters we outlined above.

    +

    In a few quick changes, we are able to programmatically create an automation that deploys workflows with custom parameters.

    +

    Using an underlying .yaml file

    +

    We can extend this idea one step further by utilizing our own .yaml version of the automation, and registering that file with our UI. This simplifies the requirements of the automation by declaring it in its own .yaml file, and then registering that .yaml with the API.

    +

    Let's first start with creating the .yaml file that will house the automation requirements. Here is how it would look like:

    +
    name: Cancel long running flows
    +description: Cancel any flow run after an hour of execution
    +trigger:
    +  match:
    +    "prefect.resource.id": "prefect.flow-run.*"
    +  match_related: {}
    +  after:
    +    - "prefect.flow-run.Failed"
    +  expect:
    +    - "prefect.flow-run.*"
    +  for_each:
    +    - "prefect.resource.id"
    +  posture: "Proactive"
    +  threshold: 1
    +  within: 30
    +actions:
    +  - type: "cancel-flow-run"
    +
    +

    We can then have a helper function that applies this YAML file with the REST API function. +

    import yaml
    +
    +from utils import post, put
    +
    +def create_or_update_automation(path: str = "automation.yaml"):
    +    """Create or update an automation from a local YAML file"""
    +    # Load the definition
    +    with open(path, "r") as fh:
    +        payload = yaml.safe_load(fh)
    +
    +    # Find existing automations by name
    +    automations = post("/automations/filter")
    +    existing_automation = [a["id"] for a in automations if a["name"] == payload["name"]]
    +    automation_exists = len(existing_automation) > 0
    +
    +    # Create or update the automation
    +    if automation_exists:
    +        print(f"Automation '{payload['name']}' already exists and will be updated")
    +        put(f"/automations/{existing_automation[0]}", payload=payload)
    +    else:
    +        print(f"Creating automation '{payload['name']}'")
    +        post("/automations/", payload=payload)
    +
    +if __name__ == "__main__":
    +    create_or_update_automation()
    +

    +

    You can find a complete repo with these APIs examples in this GitHub repository.

    +

    In this example, we managed to create the automation by registering the .yaml file with a helper function. This offers another experience when trying to create an automation.

    +

    Custom webhook kicking off an automation

    +

    We can use webhooks to expose the events API which allows us to extend the functionality of deployments and ways to respond to changes in our workflow through a few easy steps.

    +

    By exposing a webhook endpoint, we can kick off workflows that can trigger deployments - all from a simple event created from an HTTP request.

    +

    Lets create a webhook within the UI. +Here is the webhook we can use to create these dynamic events. +

    {
    +    "event": "model-update",
    +    "resource": {
    +        "prefect.resource.id": "product.models.{{ body.model_id}}",
    +        "prefect.resource.name": "{{ body.friendly_name }}",
    +        "run_count": "{{body.run_count}}"
    +    }
    +}
    +
    +From a simple input, we can easily create an exposed webhook endpoint.

    +

    webhook-simple

    +

    Each webhook will correspond to a custom event created, where you can react to it downstream with a separate deployment or automation.

    +

    For example, we can create a curl request that sends the endpoint information such as a run count for our deployment. +

    curl -X POST https://api.prefect.cloud/hooks/34iV2SFke3mVa6y5Y-YUoA -d "model_id=adhoc" -d "run_count=10" -d "friendly_name=test-user-input"
    +
    +From here, we can make a webhook that is connected to pulling in parameters on the curl command, and then it kicks off a deployment that uses these pulled parameters. +Webhook created

    +

    Let us go into the event feed, and we can automate straight from this event. +Webhook automate

    +

    This allows us to create automations that respond to these webhook events. From a few clicks in the UI, we are able to associate an external process with the Prefect events API, that can enable us to trigger downstream deployments. +Automation custom

    +

    In the next section, we will explore event triggers that automate the kickoff of a deployment run.

    +

    Using triggers

    +

    Let's take this idea one step further, by creating a deployment that will be triggered when a flow run takes longer than expected. +We can take advantage of Prefect's Marvin library that will use an LLM to classify our data. +Marvin is great at embedding data science and data analysis applications within your pre-existing data engineering workflows. In this case, we can use Marvin'd AI functions to help make our dataset more information rich.

    +

    Install Marvin with pip install marvin and set you OpenAI API key as shown here

    +

    We can add a trigger to run a deployment in response to a specific event.

    +

    Let's create an example with Marvin's AI functions. We will take in a pandas DataFrame and use the AI function to analyze it.

    +

    Here is an example of pulling in that data and classifying using Marvin AI. We can help create dummy data based on classifications we have already created.

    +
    from marvin import ai_classifier
    +from enum import Enum
    +import pandas as pd
    +
    +@ai_fn
    +def generate_synthetic_user_data(build_of_names: list[dict]) -> list:
    +    """
    +    Generate additional data for userID (numerical values with 6 digits), location, and timestamp as separate columns and append the data onto 'build_of_names'. Make userID the first column
    +    """
    +
    +@flow
    +def create_fake_user_dataset(df):
    +  artifact_df = generate_synthetic_user_data(df)
    +  print(artifact_df)
    +
    +  create_table_artifact(
    +      key="fake-user-data",
    +      table=artifact_df,
    +      description= "Dataset that is comprised of a mix of autogenerated data based on user data"
    +  )
    +
    +if __name__ == "__main__":
    +    create_fake_artifact()  
    +
    +

    Let's kick off a deployment with a trigger defined in a prefect.yaml file. Let's specify what we want to trigger when the event stays in a running state for longer than 30 seconds. +

    # Welcome to your prefect.yaml file! You can use this file for storing and managing
    +# configuration for deploying your flows. We recommend committing this file to source
    +# control along with your flow code.
    +
    +# Generic metadata about this project
    +name: automations-guide
    +prefect-version: 2.13.1
    +
    +# build section allows you to manage and build docker images
    +build: null
    +
    +# push section allows you to manage if and how this project is uploaded to remote locations
    +push: null
    +
    +# pull section allows you to provide instructions for cloning this project in remote locations
    +pull:
    +- prefect.deployments.steps.set_working_directory:
    +    directory: /Users/src/prefect/Playground/marvin-extension
    +
    +# the deployments section allows you to provide configuration for deploying flows
    +deployments:
    +- name: create-fake-user-dataset
    +  triggers:
    +    - enabled: true
    +      match:
    +        prefect.resource.id: "prefect.flow-run.*"
    +      after: "prefect.flow-run.Running",
    +      expect: [],
    +      for_each: ["prefect.resource.id"],
    +      parameters:
    +        param_1: 10
    +      posture: "Proactive"
    +  version: null
    +  tags: []
    +  description: null
    +  entrypoint: marvin-extension.py:create_fake_user_dataset
    +  parameters: {}
    +  work_pool:
    +    name: tutorial-process-pool
    +    work_queue_name: null
    +    job_variables: {}
    +  schedule: null
    +

    +

    Next steps

    +

    You've seen how to create automations via the UI, REST API, and a triggers defined in a prefect.yaml deployment definition.

    +

    To learn more about events that can act as automation triggers, see the events docs. To learn more about event webhooks in particular, see the webhooks guide

    + + + + + + + + + + + + + + + + + + + + +
    +
    + + + + + +
    + +
    + +
    + + +
    + +
    +
    +
    +
    + + + + + + + + + + + + + + \ No newline at end of file diff --git a/versions/unreleased/guides/big-data/index.html b/versions/unreleased/guides/big-data/index.html index b3fcbd2767..0335314701 100644 --- a/versions/unreleased/guides/big-data/index.html +++ b/versions/unreleased/guides/big-data/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/ci-cd/index.html b/versions/unreleased/guides/ci-cd/index.html index ac3ba32414..f8fb6d2a04 100644 --- a/versions/unreleased/guides/ci-cd/index.html +++ b/versions/unreleased/guides/ci-cd/index.html @@ -852,6 +852,8 @@ + + @@ -1108,6 +1110,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/creating-human-in-the-loop-workflows/index.html b/versions/unreleased/guides/creating-human-in-the-loop-workflows/index.html index 30a861c6eb..39b112dd31 100644 --- a/versions/unreleased/guides/creating-human-in-the-loop-workflows/index.html +++ b/versions/unreleased/guides/creating-human-in-the-loop-workflows/index.html @@ -16,7 +16,7 @@ - + @@ -852,6 +852,8 @@ + + @@ -1193,6 +1195,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/dask-ray-task-runners/index.html b/versions/unreleased/guides/dask-ray-task-runners/index.html index cfba78c174..83ae97d8f9 100644 --- a/versions/unreleased/guides/dask-ray-task-runners/index.html +++ b/versions/unreleased/guides/dask-ray-task-runners/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/deployment/aci/index.html b/versions/unreleased/guides/deployment/aci/index.html index 325c908f95..3431db3658 100644 --- a/versions/unreleased/guides/deployment/aci/index.html +++ b/versions/unreleased/guides/deployment/aci/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/deployment/daemonize/index.html b/versions/unreleased/guides/deployment/daemonize/index.html index d11428b574..c0078b948a 100644 --- a/versions/unreleased/guides/deployment/daemonize/index.html +++ b/versions/unreleased/guides/deployment/daemonize/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/deployment/developing-a-new-worker-type/index.html b/versions/unreleased/guides/deployment/developing-a-new-worker-type/index.html index c7730f7da6..028adb1977 100644 --- a/versions/unreleased/guides/deployment/developing-a-new-worker-type/index.html +++ b/versions/unreleased/guides/deployment/developing-a-new-worker-type/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/deployment/kubernetes/index.html b/versions/unreleased/guides/deployment/kubernetes/index.html index 58ac2551c6..2bfce980f1 100644 --- a/versions/unreleased/guides/deployment/kubernetes/index.html +++ b/versions/unreleased/guides/deployment/kubernetes/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/deployment/push-work-pools/index.html b/versions/unreleased/guides/deployment/push-work-pools/index.html index cb425177c5..bc5cf1553a 100644 --- a/versions/unreleased/guides/deployment/push-work-pools/index.html +++ b/versions/unreleased/guides/deployment/push-work-pools/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/deployment/serverless-workers/index.html b/versions/unreleased/guides/deployment/serverless-workers/index.html index a91dbcd711..af06fc2aec 100644 --- a/versions/unreleased/guides/deployment/serverless-workers/index.html +++ b/versions/unreleased/guides/deployment/serverless-workers/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/deployment/storage-guide/index.html b/versions/unreleased/guides/deployment/storage-guide/index.html index b6ee57944c..ae4c7cdcd8 100644 --- a/versions/unreleased/guides/deployment/storage-guide/index.html +++ b/versions/unreleased/guides/deployment/storage-guide/index.html @@ -844,6 +844,8 @@ + + @@ -1100,6 +1102,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/docker/index.html b/versions/unreleased/guides/docker/index.html index 7e4bc8ba7f..e03c6bdc57 100644 --- a/versions/unreleased/guides/docker/index.html +++ b/versions/unreleased/guides/docker/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/global-concurrency-limits/index.html b/versions/unreleased/guides/global-concurrency-limits/index.html index c84d46787d..e6713df290 100644 --- a/versions/unreleased/guides/global-concurrency-limits/index.html +++ b/versions/unreleased/guides/global-concurrency-limits/index.html @@ -852,6 +852,8 @@ + + @@ -1296,6 +1298,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/host/index.html b/versions/unreleased/guides/host/index.html index 45bf59e959..d2ceeb79a2 100644 --- a/versions/unreleased/guides/host/index.html +++ b/versions/unreleased/guides/host/index.html @@ -852,6 +852,8 @@ + + @@ -1309,6 +1311,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • diff --git a/versions/unreleased/guides/index.html b/versions/unreleased/guides/index.html index 367bab9f08..a0eab33576 100644 --- a/versions/unreleased/guides/index.html +++ b/versions/unreleased/guides/index.html @@ -850,6 +850,8 @@ + + @@ -1106,6 +1108,32 @@ +
  • + + + + + + + Automations + + + + + + + + +
  • + + + + + + + + +
  • @@ -8582,6 +8610,10 @@

    DevelopmentCreate human-in-the-loop workflows by pausing flow runs for input.

  • AutomationsConfigure actions that Prefect executes automatically based on trigger conditions.
    Webhooks Receive, observe, and react to events from other systems.